java.lang.SecurityException: Prohibited package name: java.com.xx异常抛出

本文解析了在IDEA中创建工程时,使用java作为包名导致的异常现象。深入探讨了Java类加载机制中的双亲委派模型,并给出了避免此类异常的安全实践建议。

使用idea创建了一个新的工程,然后建了一个名为“java”的文件夹,当启动里面的主函数时,出现如下异常:

根据异常信息,定位到java.lang.ClassLoader.preDefineClass进行排查,发现以下代码片断:

/* Determine protection domain, and check that: 
        - not define java.* class, 
        - signer of this class matches signers for the rest of the classes in package. 
*/  
    private ProtectionDomain preDefineClass(String name,  
    ProtectionDomain protectionDomain)  
    {  
    if (!checkName(name))  
        throw new NoClassDefFoundError("IllegalName: " + name);  
    if ((name != null) && [color=red]name.startsWith("java.")[/color]) {  
        throw new SecurityException("Prohibited package name: " +  
            name.substring(0, name.lastIndexOf('.')));  
    }  
    if (protectionDomain == null) {  
        protectionDomain = getDefaultDomain();  
    }  
  
    if (name != null)  
        checkCerts(name, protectionDomain.getCodeSource());  
  
    return protectionDomain;  
    }  
  
......  
  
// true if the name is null or has the potential to be a valid binary name  
    private boolean checkName(String name) {  
    if ((name == null) || (name.length() == 0))  
        return true;  
    if ((name.indexOf('/') != -1)  
        || (!VM.allowArraySyntax() && (name.charAt(0) == '[')))  
        return false;  
    return true;  
    }  

可以看出preDefineClass方法首先对类名进行了检查,发现以java作为一级包名,则抛出安全异常:禁止使用的包名!

这条安全异常是由Java类加载的“双亲委派模型”(详见这里)所导致的。在双亲委派模型中,由父加载类加载的类,下层加载器是不能加载的。本例中最高层加载器BootstrapClassLoader加载了classpath路径下所定义的java.*包内的类,而java.research包就不能由BootstrapClassLoader的下层加载器AppClassLoader加载了。这也是java安全机制中对于恶意代码所采取的防护措施。

解决办法就是:不使用java作为包名。

Error: A JNI error has occurred, please check your installation and try again Exception in thread "main" java.lang.SecurityException: Prohibited package name: java at java.base/java.lang.ClassLoader.preDefineClass(ClassLoader.java:910) at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1025) at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:150) at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:862) at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:760) at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:681) at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:639) at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188) at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:526) at java.base/java.lang.Class.forName0(Native Method) at java.base/java.lang.Class.forName(Class.java:534) at java.base/java.lang.Class.forName(Class.java:513) at java.base/sun.launcher.LauncherHelper.loadMainClass(LauncherHelper.java:811) at java.base/sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:706) PS D:\vscode> ^C PS D:\vscode> PS D:\vscode> d:; cd 'd:\vscode'; & 'C:\Program Files\Java\jdk-21\bin\java.exe' '-XX:+ShowCodeDetailsInExceptionMessages' '-cp' 'C:\Users\Lenovo\AppData\Roaming\Code\User\workspaceStorage\4a14cca3ab822f5de7fc05dfef06c938\redhat.java\jdt_ws\vscode_71559de3\bin' 'java.test1' Error: A JNI error has occurred, please check your installation and try again Exception in thread "main" java.lang.SecurityException: Prohibited package name: java at java.base/java.lang.ClassLoader.preDefineClass(ClassLoader.java:910) at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1025) at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:150) at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:862) at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:760) at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:681) at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:639) at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188) at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:526) at java.base/java.lang.Class.forName0(Native Method) at java.base/java.lang.Class.forName(Class.java:534) at java.base/java.lang.Class.forName(Class.java:513) at java.base/sun.launcher.LauncherHelper.loadMainClass(LauncherHelper.java:811) at java.base/sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.j
08-01
评论 4
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值