使用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可能就足够了。

相关推荐
蝎子莱莱爱打怪3 小时前
OpenClaw 从零配置指南:接入飞书 + 常用命令 + 原理图解
java·后端·ai编程
狼爷5 小时前
Go 没有 override?别硬套继承!用接口+嵌入,写更清爽的“覆盖”逻辑
java·go
小兔崽子去哪了7 小时前
Java 自动化部署
java·后端
ma_king7 小时前
入门 java 和 数据库
java·数据库·后端
后端AI实验室8 小时前
我用Cursor开发了3个月,整理出这套提效4倍的工作流
java·ai
码路飞12 小时前
GPT-5.3 Instant 终于学会好好说话了,顺手对比了下同天发布的 Gemini 3.1 Flash-Lite
java·javascript
SimonKing12 小时前
OpenCode AI编程助手如何添加Skills,优化项目!
java·后端·程序员
Seven9714 小时前
剑指offer-80、⼆叉树中和为某⼀值的路径(二)
java
怒放吧德德1 天前
Netty 4.2 入门指南:从概念到第一个程序
java·后端·netty
雨中飘荡的记忆1 天前
大流量下库存扣减的数据库瓶颈:Redis分片缓存解决方案
java·redis·后端