使用Collections.singletonList()遇到的问题

示例代码

java 复制代码
    @RequestMapping("/list")
    public Result list() {
        startPage();
        List<Byte> statusList = Collections.singletonList(DynamicContentConstants.BE_APPROVED);
        //List<Byte> statusList = new ArrayList<>();
        //statusList.add(DynamicContentConstants.BE_APPROVED);
        List<User> userList = userService.selectListByStatus(statusList);
        return Result.ok(userList);
    }

在JDK8中使用Collections.singletonList()方法构建集合时没有任何问题,而在JDK17中就会诱发一下问题:

java 复制代码
nested exception is org.apache.ibatis.exceptions.PersistenceException: 
\r\n### Error querying database.  
Cause: java.lang.reflect.InaccessibleObjectException: Unable to make public int java.util.Collections$SingletonList.size() accessible: module java.base does not \"opens java.util\" to unnamed module @1530c739\r\n### 
Cause: java.lang.reflect.InaccessibleObjectException: Unable to make public int java.util.Collections$SingletonList.size() accessible: module java.base does not \"opens java.util\" to unnamed module @1530c739

经过大量的搜索,诱发此问题的原因是由Java模块系统的访问控制引起的。在Java 9及以上版本中,引入了模块系统(Project Jigsaw),这增加了对JVM内部API访问的限制。错误信息中的"Unable to make public int java.util.CollectionsSingletonList.size() accessible: module java.base does not "opens java.util" to unnamed module"表明,尝试访问`java.util`包中的`CollectionsSingletonList.size()accessible:modulejava.basedoesnot"opensjava.util"tounnamedmodule"表明,尝试访问'java.util'包中的'CollectionsSingletonList类的size`方法时,因为模块化系统的访问限制而失败。

具体到你的情况,当你使用Collections.singletonList方法时,它返回的是一个不可变的单元素列表,这个列表是Collections$SingletonList的实例。这个实例在运行时可能因为Java平台模块系统的限制而无法被MyBatis正常访问和处理。

而当你使用new ArrayList<>()并手动添加元素时,返回的是一个标准的ArrayList实例,它没有受到Java模块系统访问控制的影响,因此MyBatis可以正常处理这个列表。

解决方案

一般来说,有几种方法可以解决或绕过这个问题:

  1. 使用ArrayList:就像你已经发现的那样,使用new ArrayList<>()而不是Collections.singletonList可以避免这个问题。如果你只需要添加一个元素,这是一个简单有效的解决方案。

  2. 配置JVM参数 :如果你出于某些原因需要使用Collections.singletonList或其他可能受模块系统限制的方法,你可以尝试通过添加JVM启动参数来放宽访问限制。例如,可以尝试添加**--add-opens java.base/java.util=ALL-UNNAMED**参数给JVM。这会指示JVM放宽对java.util包内类的访问限制。请注意,这种方法可能会降低应用程序的安全性,因为它放宽了模块化系统的访问控制。

  3. 升级依赖库:确保你的应用依赖的库(如MyBatis)是最新版本,因为一些库可能已经修复了与Java模块系统相关的兼容性问题。

选择哪种方法取决于你的具体需求和约束。在大多数情况下,简单地使用ArrayList可能就足够了。

相关推荐
可乐加.糖10 分钟前
一篇关于Netty相关的梳理总结
java·后端·网络协议·netty·信息与通信
s91236010112 分钟前
rust 同时处理多个异步任务
java·数据库·rust
9号达人13 分钟前
java9新特性详解与实践
java·后端·面试
cg501717 分钟前
Spring Boot 的配置文件
java·linux·spring boot
啊喜拔牙25 分钟前
1. hadoop 集群的常用命令
java·大数据·开发语言·python·scala
anlogic1 小时前
Java基础 4.3
java·开发语言
非ban必选1 小时前
spring-ai-alibaba第七章阿里dashscope集成RedisChatMemory实现对话记忆
java·后端·spring
A旧城以西1 小时前
数据结构(JAVA)单向,双向链表
java·开发语言·数据结构·学习·链表·intellij-idea·idea
杉之2 小时前
选择排序笔记
java·算法·排序算法
Naive_72 小时前
蓝桥杯准备(前缀和差分)
java·职场和发展·蓝桥杯