自定义一个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;
    }

}
public class Test2 {
    public static void main(java.lang.String[] args) {
        String string = new String();
    }
}

输出

静态代码块--自定义String
代码块--自定义String

可以正常调用,因为此时以包名区分了String类

这里实际上是使用AppClassLoader应用程序类加载器进行加载的,与普通类是一样的(也可以自定义加载器加载)

public class Test2 {
    public static void main(java.lang.String[] args) {
        ClassLoader classLoader = String.class.getClassLoader();
        System.out.println(classLoader);
    }
}
输出:
sun.misc.Launcher$AppClassLoader@18b4aac2

包名为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类
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值