浅谈Java类加载

1.类加载是啥?

当一个程序需要运行时,那么它所涉及相关类需要被加载到JVM中,这个过程称之为类加载。

2.类加载器是啥?

类加载器简而言之就是加载类的对象,java中的类加载器可分为三类,引导类加载器,扩展类加载器,应用程序类加载器。这三类加载器各有各的分工后面再说。

3.sun.misc.Launcher 是啥?

这个launcher就是类加载的源头,我们称之为启动器(JVM启动器)(我们能扒代码起点),只有启动器启动了才会调用类加载器去加载类。至于如何创建的,其实是c++调用咱们Java代码创建的其中细节有兴趣的自己去瞅瞅,这里就不多说了。

4.一个类的加载到卸载会经历什么?

这一块算是从书面上获取的知识点,总的概括为以下这些步骤,加载,验证(校验字节码文件的正确性)比如说下图的是否以cafe babe开头等

,准备(给一些静态变量分配内存,并赋予默认值),解析(个人理解是把一些引用指向已经加载好的内存地址),初始化(初始化静态变量,执行静态代码块),使用,卸载。

关于类加载需要注意一点,当我们运行的程序用到某个类的时候这个类才会被加载,jar包和war包的类不是一次性全部加载的,有点类似spring中的懒加载。

5.什么是双亲委派机制?

这个双亲委派机制,说的是类加载时候遵循的一个规则。首先我们需要知道的是上面2中所说的类加载器是有一定亲戚关系的,我们可以理解为儿子是应用程序类加载器,爸爸是扩展类加载器,爷爷是引导类加载器,至于为啥这么规定呢,源码里是这么写的。双亲委派机制的正式说明为,加载某个类时会先委托父加载器寻找目标类,找不到再委托上层父加载器加载,如果所有父加载器在自己的加载类路径下都找不到目标类,则在自己的类加载路径中查找并载入目标类。

 

 源码展示

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;
        }
    }

 这个是ClassLoader类中的加载方法,我们可以看到 如果找到了直接返回,没有的话则会用父加载器加载,当都没有在成功 最后会用自身类加载路径下的findClass()方法去加载。

6.为何要设计双亲委派机制?

首先是沙箱安全机制,如果自己写一个String类,Java是不会加载你的string,这样防止了核心类库不会被修改;其次是避免类的重复加载,当父亲加载过了之后,就没必要子加载器再加载一遍了,保证被加载的唯一性!

以上就是对类加载这一块一些知识点的学习心得,如有不正确的地方欢迎指正!

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java,浅拷贝是指创建一个新的对象,并将原有对象的所有非引用型的数据(如基本数据型和对象的引用)复制到新对象。如果原对象包含引用型的对象,那么新对象只是复制了这些引用,而不是它们所指向的实际对象。浅拷贝通常用于快速复制对象,但可能引发一些预期之外的问题,比如修改子对象会影响原对象。 以下是一个简单的浅拷贝的代码示例,使用`Cloneable`接口和`Object`的`clone()`方法: ```java public class Person implements Cloneable { private String name; private Address address; // 构造函数和getter/setter省略 public Person shallowCopy() { try { return (Person) this.clone(); } catch (CloneNotSupportedException e) { throw new RuntimeException("Failed to clone object", e); } } } class Address { private String street; // 其他属性和getter/setter省略 } // 使用浅拷贝 public void modifyShallowCopy(Person person) { person.getAddress().setStreet("New Address"); // 修改子对象 } public static void main(String[] args) { Person original = new Person(); Address address = new Address(); address.setStreet("Original Address"); original.setName("John"); original.setAddress(address); Person copy = original.shallowCopy(); // 假设copy不会被修改 // 修改原始对象的地址 address.setStreet("Modified Address"); System.out.println("Original: " + original.getName() + ", " + original.getAddress().getStreet()); System.out.println("Copy: " + copy.getName() + ", " + copy.getAddress().getStreet()); // 会看到两者地址都变了 } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值