先说下结论,可以自定义包名不为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)
总结
-
可以自定义包名不为java.lang的String类,并区别包名正常使用
-
自定义包名为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,欢迎关注~