先说下结论,可以自定义包名不为java.lang的String类,区别包名是可以正常使用的。

包名不为java.lang

package com.seven.jvm;

public final class String {
    /** The value is used for character storage. */
    private final char value[] = {};

    /** Cache the hash code for the string */
    private int hash; // Default to 0

    /** use serialVersionUID from JDK 1.0.2 for interoperability */
    private static final long serialVersionUID = -6849794470754667710L;

    public String(int hash) {
        this.hash = hash;
    }

    public String(){

    }

    static{
        System.out.println("静态代码块--自定义String");
    }

    {
        System.out.println("代码块--自定义String");
    }
    public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            char val[] = value;

            for (int i = 0; i < value.length; i++) {
                h = 31 * h + val[i];
            }
            hash = h;
        }
        return h;
    }

}

可以正常使用,但在使用时有有用到java.lang下的String,那就需要区别报名使用,即其中一个使用全类名表示,一般生产中不会去自定义一个与JDK类库中同名的类,这里只作为拓展了解即可~

包名为java.lang.String

package java.lang;


public final class String {

    /**
     * The value is used for character storage.
     */
    private final char value[] = {};

    /**
     * Cache the hash code for the string
     */
    private int hash; // Default to 0

    /**
     * use serialVersionUID from JDK 1.0.2 for interoperability
     */
    private static final long serialVersionUID = -6849794470754667710L;

    public String(int hash) {
        this.hash = hash;
    }

    public String() {

    }

    static {
        System.out.println("静态代码块--自定义String");
    }

    {
        System.out.println("代码块--自定义String");
    }

    public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            char val[] = value;

            for (int i = 0; i < value.length; i++) {
                h = 31 * h + val[i];
            }
            hash = h;
        }
        return h;
    }

}

String类下写main方法

package java.lang;

public final class String {

    public static void main(String[] args) {
        System.out.println("aaa");;
    }
}

输出:

错误: 在类 java.lang.String 中找不到 main 方法, 请将 main 方法定义为:
   public static void main(String[] args)
否则 JavaFX 应用程序类必须扩展javafx.application.Application

原因在于双亲委派模型,先从父类加载器寻找能不能加载此类,如果没有则再到子类;因此在加载String类时,会最终委派给Bootstrap ClassLoader去加载,加载的是rt.jar包中的那个java.lang.String,而rt.jar包中的String类是没有main方法的,因此报错误

同包下新建一个类写main方法

package java.lang;

public class Main {
    public static void main(String[] args) {
       String str =  new String();
    }
}

输出:

java.lang.SecurityException: Prohibited package name: java.lang
    at java.lang.ClassLoader.preDefineClass(ClassLoader.java:662)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:761)
    at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
    at java.net.URLClassLoader.defineClass(URLClassLoader.java:467)
    at java.net.URLClassLoader.access$100(URLClassLoader.java:73)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:368)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:362)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:361)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:335)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
    at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:495)
Error: A JNI error has occurred, please check your installation and try again
Exception in thread "main" 

限制包名,不能自定义这个包名,与java类库冲突,安全管理器不通过,这里不管用不用到String都会有这个报错

原因:java.lang 是java 自带类库包,是属于rt.jar 包下的文件,而rt.jar 是通过启动类加载器(Bootstrap ClassLoader)加载的,由于双亲委派,因此java.lang 包肯定早于自定义的java.lang 包的加载,就会冲突。

调用方法不在java.lang包中

package com.seven;

public class Test {
    public static void main(String[] args) {
        String string = new String();
    }
}

无输出

原因,由于双亲委派,这里加载的String包是rt.jar中的java.lang.String类。因此这里并没有用到自定义的String类,因为不会加载到自定义的String(即便改自定义String的包名也叫java.lang)

总结

  1. 可以自定义包名不为java.lang的String类,并区别包名正常使用

  2. 自定义包名为java.lang的String类

    • String类下写main方法:由于双亲委派模型,在加载String类时,会最终委派给Bootstrap ClassLoader去加载,加载的是rt.jar包中的那个java.lang.String,而rt.jar包中的String类是没有main方法的,因此报错误

    • 启动类也在java.lang包下:这里与是否用到String类无关,会报 Prohibited package name: java.lang错误。由于双亲委派,java.lang 包肯定早于自定义的java.lang 包的加载,就会冲突.

    • 调用方法不在java.lang包中:此时由于双亲委派模型的存在,并不会加载到自定义的String类

关于作者

来自一线程序员Seven的探索与实践,持续学习迭代中~

公众号:seven97,欢迎关注~