dll已加载但找不到入口点dllregisterserver_Java 是如何加载类的?

a52a05c3a1d3583c7defd24980de1b44.png

本文只是从 Java 的角度出发,并不涉及 Android 的类加载方式。


从上一篇解析类加载机制的文章:

我们已经知道了 ClassLoader 的委托机制。

本篇文章我们来详细分析下 ClassLoader 是如何加载 Java 类的。


一、ClassLoader 使用

dcbb1a51d779b78d7cc1b3529a8e7971.png

流程简单说是这样的:

  • 我们用 ClassLoader 的 loadClass() 方法获取到了对应类的 class 文件;
  • 随后通过 class 文件调用 newInstance() 方法创建了对应类的实例;
  • 然后调用实例的方法;

那么其实 loadClass() 就是我们加载类的关键一步。

二、loadClass() 源码解析

从上例中,点击 loadClass() 方法,进入 ClassLoader 源码

e36c2565ee0f2c1fe758a9ecad7bd30a.png

发现实际起作用的还是 loadClass(name,false) 这个方法,我们点进去看看

a40b54401b0831aa9747722746cdfe4f.png

源码的解释已经很清楚了,我们再来看下实际代码

e5e4d4d435cd227ac4416817ddc3b551.png

用流程图可以这样概括一下

dc7f93948ee6eead91dce7b922247e18.png

我们逐个步骤看下吧

① 保证线程安全

synchronized (getClassLoadingLock(name))

给整个 loadClass 的过程加了一把同步锁,避免了多线程共同加载相同名字的 class 的类加载问题。

② 查看是否已加载

8fdf3c267ce98c7384e64ab08ba22756.png
findLoadedClass(name)

看了源码,发现查看对应名字的 class 是否已被加载是调用的 native 方法:

findLoadedClass0(name)

如果 class 已经被加载,那么就直接返回加载的 Class 文件;

如果 class 并未被加载,那么继续进行下面的步骤。

③ 查找「父'类加载器」

785234a573d956c37e76969aa273cc26.png

我们再看下全局变量 parent 的声明

000f0ad17c576a334c8d1573c437e5eb.png
952e043c770c0c71b88248217b427f39.png

所以,在加载类时,起初 parent 不为 null,所以会调用

parent.loadClass(name,false)

依次往父级上推,直到 parent 为 null,即追溯到的「父'类加载器」是

BootStrap,则会调用

findBootstrapClassOrNull(name)

方法,我们来看下这个方法的定义

a673c3d6bcf064d12a29073bc8573d6c.png

这个方法会返回一个“被 bootstrap 加载过的类,如果没有找到,则会返回 null ”

而真正的逻辑处理也是一个 native 方法:

findBootstrapClass(name)

如果这个方法返回值不为 null ,loadClass 流程会进入:

parent.loadClass(name, false)

阶段,依次往父级上推,直到出现下方情况之一

  • 加载成功,返回加载好的类
  • 加载失败,返回 null
  • 加载异常,抛出 ClassNotFoundException 异常

则结束此过程。

④ 如果加载失败,返回为 null,则会调用:

findClass(name)
068a37550bae03f815e03451590511a3.png

即如果我们没有自定义类加载器,默认则会抛出

ClassNotFoundException 异常。

ok ,这样整个过程就结束了。

我们可以看到,整个 loadClass() 的方法会有两种情况:

  • 加载成功,返回加载好的类
  • 加载异常,抛出 ClassNotFoundException 异常

三、单纯了解了 ClassLoader 中的 loadClass() 不够,我们来自定义一个类加载器吧

我的例子的思路大致是:

① 创建一个需要被自定义的 ClassLoader 加载的 java 文件,并编译成为 class 文件

391adeb364c69e032d52c905241fae1b.png

很简单,就是打印一句话,但此例是我们要创建 java 文件的基类。

下面是我们的需要加载的 java 文件,即上面基类的子类。

d92c247577314dad0436a5a4a3a0994d.png

运行,得到 class 文件

2dd3053954349713f0362498eaa2330d.png

红框内即我们得到的class 文件

② 在我们工程目录下创建一个新的目录,用来存放我们创建好的 class 文件

777ebfeb088269692adb244a8f450663.png

③ 编写我们的自定义 ClassLoader 文件

69f64c36d624d55ce8a5bffe93433b51.png

对,没有可扩展性,路径都是定的。

因为上文中我们解析 loadClass() 方法的源码时,得知我们需要重写

findClass

方法,所以这里就重写了下,主要功能就是找到我们放到 myclasses 文件夹下的 class 文件,并且调用 defineClass 方法去解析出来。

④ 创建运行类,查看类加载器的加载规则

747677c6ed27bd4ec695a3232897a286.png

打印结果是

52706a6824ae6ea12e7fc8abef834f7e.png

我们发现,同样是通过 myClassLoader 的实例加载的类,但是当我们加载

MyClassLoaderTest 时:

Class loadClass = classLoader.loadClass("com.guaju.classloadertest.MyClassLoaderTest");

真正的类加载器是 AppClassLoader

而当我们加载 PrintUtil 时:

 Class> loadClass = myClassLoader.loadClass("PrintUtil");

真正的类加载器是 MyClassLoader。

当我们修改 PrintUtils 的加载方式时

4c177b14e1b805017be7faebfbdaa021.png

真正的类加载器也是 AppClassLoader

0009b4aeb0b3632b0d9df498bf707f36.png

总结得到这两点:

  • 自定义类加载器时,如果传入完整类名,会优先使用系统类加载器去加载类,如果系统类加载器找不到该类,则会调用自定义的类加载器。
  • 自定义类加载器时,需要使用 findClass 去定位需要加载的类,读取并调用 defineClass 方法去解析类

好了,本篇完~~~

后续会继续针对 Android 项目的类加载进行解析,如果有兴趣想继续看,点个关注吧~

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值