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

相关推荐
bemyrunningdog3 分钟前
IntelliJ IDEA导出WAR包全指南
java·ide·intellij-idea
虫师c11 分钟前
List vs Set:深入剖析Java两大集合的核心区别与实战应用
java·数据结构·list·set·java集合
emiya_saber12 分钟前
Linux 硬盘分区管理
java·linux·网络
量子炒饭大师25 分钟前
【一天一个计算机知识】—— 【编程百度】预处理指令
java·开发语言
q***518929 分钟前
Spring Boot 条件注解:@ConditionalOnProperty 完全解析
java·spring boot·后端
任子菲阳33 分钟前
学Java第四十四天——Map实现类的源码解析
java·开发语言
听风吟丶37 分钟前
Java 11+ HttpClient 实战:从 HttpURLConnection 到现代 HTTP 客户端的全面升级
java·开发语言·http
Felix_XXXXL1 小时前
集成RabbitMQ+MQ常用操作
java·后端
hygge9991 小时前
JVM GC 垃圾回收体系完整讲解
java·开发语言·jvm·经验分享·面试