使用Collections.singletonList()遇到的问题
示例代码
@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中就会诱发一下问题:
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可以正常处理这个列表。
解决方案
一般来说,有几种方法可以解决或绕过这个问题:
-
使用ArrayList:就像你已经发现的那样,使用new ArrayList<>()而不是Collections.singletonList可以避免这个问题。如果你只需要添加一个元素,这是一个简单有效的解决方案。
-
配置JVM参数:如果你出于某些原因需要使用
Collections.singletonList
或其他可能受模块系统限制的方法,你可以尝试通过添加JVM启动参数来放宽访问限制。例如,可以尝试添加--add-opens java.base/java.util=ALL-UNNAMED参数给JVM。这会指示JVM放宽对java.util包内类的访问限制。请注意,这种方法可能会降低应用程序的安全性,因为它放宽了模块化系统的访问控制。 -
升级依赖库:确保你的应用依赖的库(如MyBatis)是最新版本,因为一些库可能已经修复了与Java模块系统相关的兼容性问题。
选择哪种方法取决于你的具体需求和约束。在大多数情况下,简单地使用ArrayList可能就足够了。