双亲委派模型是Java加载类的机制.采用双亲委派模型的好处是Java类随着它的类加载器一块儿具有了一种带有优先级的层级关系,经过这种层级关系能够避免类的重复加载.java
1.双亲委派模型最大的好处就是让Java类同其类加载器一块儿具有了一种带优先级的层次关系。这句话可能很差理解,咱们举个例子。好比咱们要加载java.lang.Object类,不管咱们用哪一个类加载器去加载Object类,这个加载请求最终都会委托给Bootstrap ClassLoader,这样就保证了全部加载器加载的Object类都是同一个类。若是没有双亲委派模型,那就乱了套了,彻底可能搞出多个不一样的Object类。
2.自上而下每一个类加载器都会尽力加载.算法
1.首先加载类调用的loadClass方法,咱们找到ClassLoader的loadClass():性能优化
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}复制代码
protected Class<?> findClass(String name) throws ClassNotFoundException {
throw new ClassNotFoundException(name);
}复制代码
1.首先须要一个编译好的class文件,笔者用了一个以前写的斐波那契的类Fib.class(所在路径:C:/Users/Think/crabapple),下面是用idea经过反编译方式打开的class文件,注意记下class文件的包名,在后续代码中须要使用类的全限定名称.app
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package crabapple;
public class Fib {
public static int fib(int num) {
return num < 2 ? num : fib(num - 2) + fib(num - 1);
}
}复制代码
2.继承ClassLoader,重写findClass方法:ide
class MyClassLoader extends ClassLoader {
private String classPath; // 保存的地址
/** * 传入地址构造函数 * @param classPath */
public MyClassLoader(String classPath) {
this.classPath = classPath;
}
/** * 读取class文件 * @param name * @return * @throws Exception */
private byte[] loadByte(String name) throws Exception {
String inPath = classPath + "/" + name + ".class";
FileInputStream fis = new FileInputStream(inPath);
int len = fis.available();
byte[] data = new byte[len];
fis.read(data);
fis.close();
return data;
}
/** * 重写findClass方法,让加载的时候调用findClass方法 * @param name * @return * @throws ClassNotFoundException */
protected Class<?> findClass(String name) throws ClassNotFoundException {
try {
byte[] data = loadByte(name);
// 将字节码载入内存
return defineClass(name, data, 0, data.length);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}复制代码
3.测试,一下将使用一些反射机制和class类的方法.函数
public class ClassLoaderTest extends ClassLoader {
//main函数本该抛出异常有 ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException,为了好看,简写成Exception
public static void main(String[] args) throws Exception {
//初始化类加载器
MyClassLoader myClassLoader=new MyClassLoader("C:/Users/Think/crabapple");
//加载Fib类,笔者class文件包名为crabapple
Class myClass=myClassLoader.loadClass("crabapple.Fib");
//获取加载类的实例
Object object=myClass.newInstance();
//获取该类一个名为fib,且参数为int的方法
Method method=myClass.getMethod("fib",int.class);
//执行这个方法
int result=method.invoke(object,4);
//打印结果
System.out.print(result);
//output
/** * 3 * Process finished with exit code 0 */
}
}复制代码
后续会持续更新性能优化专题知识,写的很差的地方也但愿大牛能指点一下,你们以为不错能够点个赞在关注下,之后还会分享更多文章!