浅谈JAVA 类加载器

类加载机制

类加载器负责加载所有的类,系统为所有被载入内存中的类生成一个 java.lang.Class 实例。一旦一个类被载入 JVM 中,同个类就不会被再次载入了。现在的问题是,怎么样才算“同一个类”?

正如一个对象有一个唯一的标识一样,一个载入 JVM 中的类也有一个唯一的标识。在 Java 中,一个类用其全限定类名(包括包名和类名)作为标识:但在 JVM 中,一个类用其全限定类名和其类加载器作为唯一标识。例如,如果在 pg 的包中有一个名为 Person 的类,被类加载器 ClassLoader 的实例 k1 负责加载,则该 Person 类对应的 Class 对象在 JVM 中表示为(Person、pg、k1)。这意味着两个类加载器加载的同名类:(Person、pg、k1)和(Person、pg、k12)是不同的,它们所加载的类也是完全不同、互不兼容的。

当 JVM 启动时,会形成由三个类加载器组成的初始类加载器层次结构。

  • Bootstrap ClassLoader:根类加载器。
  • Extension ClassLoader:扩展类加载器。
  • System ClassLoader:系统类加载器。

Bootstrap ClassLoader 被称为引导(也称为原始或根)类加载器,它负责加载 Java 的核心类。在Sun 的 JVM 中,当执行 java.exe 命令时,使用 -Xbootclasspath 或 -D 选项指定 sun.boot.class.path 系统属性值可以指定加载附加的类。

JVM的类加载机制主要有如下三种。

  • 全盘负责。所谓全盘负责,就是当一个类加载器负责加载某个 Class 时,该 Class 所依赖的和引用的其他 Class 也将由该类加载器负责载入,除非显式使用另外一个类加载器来载入。
  • 父类委托。所谓父类委托,则是先让 parent(父)类加载器试图加载该 Class,只有在父类加载器无法加载该类时才尝试从自己的类路径中加载该类。
  • 缓存机制。缓存机制将会保证所有加载过的 Class 都会被缓存,当程序中需要使用某个 Class 时,类加载器先从缓存区中搜寻该 Class,只有当缓存区中不存在该 Class 对象时,系统才会读取该类对应的二进制数据,并将其转换成 Class 对象,存入缓存区中。这就是为什么修改了 Class 后,必须重新启动 JVM,程序所做的修改才会生效的原因。

除了可以使用 Java 提供的类加载器之外,开发者也可以实现自己的类加载器,自定义的类加载器通过继承 ClassLoader 来实现。JVM 中这4种类加载器的层次结构如下图所示。

注意:类加载器之间的父子关系并不是类继承上的父子关系,这里的父子关系是类加载器实例之间的关系

下面程序示范了访问 JVM 的类加载器。

public class ClassLoaderPropTest {
 public static void main(String[] args) throws IOException {
 // 获取系统类加载器
 ClassLoader systemLoader = ClassLoader.getSystemClassLoader();
 System.out.println("系统类加载器:" + systemLoader);
 /*
  * 获取系统类加载器的加载路径——通常由CLASSPATH环境变量指定 如果操作系统没有指定CLASSPATH环境变量,默认以当前路径作为
  * 系统类加载器的加载路径
  */
 Enumeration<URL> em1 = systemLoader.getResources("");
 while (em1.hasMoreElements()) {
  System.out.println(em1.nextElement());
 }
 // 获取系统类加载器的父类加载器:得到扩展类加载器
 ClassLoader extensionLader = systemLoader.getParent();
 System.out.println("扩展类加载器:" + extensionLader);
 System.out.println("扩展类加载器的加载路径:" + System.getProperty("java.ext.dirs"));
 System.out.println("扩展类加载器的parent: " + extensionLader.getParent());
 }
}

运行上面的程序,会看到如下运行结果

系统类加载器:sun.misc.Launcher$AppClassLoader@73d16e93
file:/F:/EclipseProjects/demo/bin/
扩展类加载器:sun.misc.Launcher$ExtClassLoader@15db9742
扩展类加载器的加载路径:C:\Program Files\Java\jre1.8.0_181\lib\ext;C:\Windows\Sun\Java\lib\ext
扩展类加载器的parent: null

从上面运行结果可以看出,系统类加载器的加载路径是程序运行的当前路径,扩展类加载器的加载路径是null(与 Java8 有区别),但此处看到扩展类加载器的父加载器是null,并不是根类加载器。这是因为根类加载器并没有继承 ClassLoader 抽象类,所以扩展类加载器的 getParent() 方法返回null。但实际上,扩展类加载器的父类加载器是根类加载器,只是根类加载器并不是 Java 实现的。

从运行结果可以看出,系统类加载器是 AppClassLoader 的实例,扩展类加载器 ExtClassLoader 的实例。实际上,这两个类都是 URLClassLoader 类的实例。

注意:JVM 的根类加载器并不是 Java 实现的,而且由于程序通常无须访问根类加载器,因此访问扩展类加载器的父类加载器时返回null。

类加载器加载 Class 大致要经过如下8个步骤。

  1. 检测此 Class 是否载入过(即在缓存区中是否有此Class),如果有则直接进入第8步,否则接着执行第2步。
  2. 如果父类加载器不存在(如果没有父类加载器,则要么 parent 一定是根类加载器,要么本身就是根类加载器),则跳到第4步执行;如果父类加载器存在,则接着执行第3步。
  3. 请求使用父类加载器去载入目标类,如果成功载入则跳到第8步,否则接着执行第5步。
  4. 请求使用根类加载器来载入目标类,如果成功载入则跳到第8步,否则跳到第7步。
  5. 当前类加载器尝试寻找 Class 文件(从与此 ClassLoader 相关的类路径中寻找),如果找到则执行第6步,如果找不到则跳到第7步。
  6. 从文件中载入 Class,成功载入后跳到第8步。
  7. 抛出 ClassNotFoundExcepuon 异常。
  8. 返回对应的 java.lang.Class 对象。

其中,第5、6步允许重写 ClassLoader的 findClass() 方法来实现自己的载入策略,甚至重写 loadClass() 方法来实现自己的载入过程。

创建并使用自定义的类加载器

JVM 中除根类加载器之外的所有类加载器都是 ClassLoader 子类的实例,开发者可以通过扩展 ClassLoader 的子类,并重写该 ClassLoader 所包含的方法来实现自定义的类加载器。查阅API文档中关于 ClassLoader 的方法不难发现,ClassLoader 中包含了大量的 protected 方法——这些方法都可被子类重写。

ClassLoader 类有如下两个关键方法。

  • loadClass(String name, boolean resolve):该方法为 ClassLoader 的入口点,根据指定名称来加载类,系统就是调用 ClassLoader 的该方法来获取指定类对应的 Class 对象。
  • findClass(String name):根据指定名称来查找类。

如果需要实现自定义的 ClassLoader,则可以通过重写以上两个方法来实现,通常推荐重写 findClass() 方法,而不是重写 loadClass() 方法。loadClass() 方法的执行步骤如下。

  1. 用 findLoadedClass(String) 来检查是否已经加载类,如果已经加载则直接返回。
  2. 在父类加载器上调用 loadClass() 方法。如果父类加载器为null,则使用根类加载器来加载。
  3. 调用 findClass(String) 方法查找类。

从上面步骤中可以看出,重写 findClass()方法可以避免覆盖默认类加载器的父类委托、缓冲机制两种策略:如果重写 loadClass() 方法,则实现逻辑更为复杂。

在 ClassLoader 里还有一个核心方法:Class defineClass(String name, byte[] b, int off,int len) 该方法负责将指定类的字节码文件(即 Class 文件,如 Hello.class)读入字节数组 byte[] b 内,并把它转换为 Class对象,该字节码文件可以来源于文件、网络等。

defineClass() 方法管理 JVM 的许多复杂的实现,它负责将字节码分析成运行时数据结构,并校验有效性等。不过不用担心,程序员无须重写该方法。实际上该方法是 final 的,即使想重写也没有机会。

除此之外,ClassLoader 里还包含如下一些普通方法。

  • findSystemClass(String name):从本地文件系统装入文件。它在本地文件系统中寻找类文件,如果存在,就使用 defineClass() 方法将原始字节转换成 Class 对象,以将该文件转换成类。
  • static getSystemClassLoader():这是一个静态方法,用于返回系统类加载器。
  • getParent():获取该类加载器的父类加载器。
  • resolveClass(Class<?> c):链接指定的类。类加载器可以使用此方法来链接类c。读者无须理会关于此方法的太多细节。
  • findLoadedClass(String name):如果此 Java 虚拟机已加载了名为 name 的类,则直接返回该类对应的 Class 实例,否则返回null,该方法是 Java 类加载缓存机制的体现。

下面程序开发了一个自定义的 ClassLoader,该 ClassLoader 通过重写 findClass() 方法来实现自定义的类加载机制。这个 ClassLoader 可以在加载类之前先编译该类的文件,从而实现运行 Java 之前先编译该程序的目标,这样即可通过该 ClassLoader 直接运行 Java 源文件。

public class CompileClassLoader extends ClassLoader {
 // 读取一个文件的内容
 private byte[] getBytes(String filename) throws IOException {
 File file = new File(filename);
 long len = file.length();
 byte[] raw = new byte[(int) len];
 try (FileInputStream fin = new FileInputStream(file)) {
  // 一次读取class文件的全部二进制数据
  int r = fin.read(raw);
  if (r != len)
  throw new IOException("无法读取全部文件:" + r + " != " + len);
  return raw;
 }
 }

 // 定义编译指定Java文件的方法
 private boolean compile(String javaFile) throws IOException {
 System.out.println("CompileClassLoader:正在编译 " + javaFile + "...");
 // 调用系统的javac命令
 Process p = Runtime.getRuntime().exec("javac " + javaFile);
 try {
  // 其他线程都等待这个线程完成
  p.waitFor();
 } catch (InterruptedException ie) {
  System.out.println(ie);
 }
 // 获取javac线程的退出值
 int ret = p.exitValue();
 // 返回编译是否成功
 return ret == 0;
 }

 // 重写ClassLoader的findClass方法
 protected Class<?> findClass(String name) throws ClassNotFoundException {
 Class clazz = null;
 // 将包路径中的点(.)替换成斜线(/)。
 String fileStub = name.replace(".", "/");
 String javaFilename = fileStub + ".java";
 String classFilename = fileStub + ".class";
 File javaFile = new File(javaFilename);
 File classFile = new File(classFilename);
 // 当指定Java源文件存在,且class文件不存在、或者Java源文件
 // 的修改时间比class文件修改时间更晚,重新编译
 if (javaFile.exists() && (!classFile.exists() || javaFile.lastModified() > classFile.lastModified())) {
  try {
  // 如果编译失败,或者该Class文件不存在
  if (!compile(javaFilename) || !classFile.exists()) {
   throw new ClassNotFoundException("ClassNotFoundExcetpion:" + javaFilename);
  }
  } catch (IOException ex) {
  ex.printStackTrace();
  }
 }
 // 如果class文件存在,系统负责将该文件转换成Class对象
 if (classFile.exists()) {
  try {
  // 将class文件的二进制数据读入数组
  byte[] raw = getBytes(classFilename);
  // 调用ClassLoader的defineClass方法将二进制数据转换成Class对象
  clazz = defineClass(name, raw, 0, raw.length);
  } catch (IOException ie) {
  ie.printStackTrace();
  }
 }
 // 如果clazz为null,表明加载失败,则抛出异常
 if (clazz == null) {
  throw new ClassNotFoundException(name);
 }
 return clazz;
 }

 // 定义一个主方法
 public static void main(String[] args) throws Exception {
 // 如果运行该程序时没有参数,即没有目标类
 if (args.length < 1) {
  System.out.println("缺少目标类,请按如下格式运行Java源文件:");
  System.out.println("java CompileClassLoader ClassName");
 }
 // 第一个参数是需要运行的类
 String progClass = args[0];
 // 剩下的参数将作为运行目标类时的参数,
 // 将这些参数复制到一个新数组中
 String[] progArgs = new String[args.length - 1];
 System.arraycopy(args, 1, progArgs, 0, progArgs.length);
 CompileClassLoader ccl = new CompileClassLoader();
 // 加载需要运行的类
 Class<?> clazz = ccl.loadClass(progClass);
 // 获取需要运行的类的主方法
  Method main = clazz.getMethod("main", (new String[0]).getClass());
 Object[] argsArray = { progArgs };
 main.invoke(null, argsArray);
 }
}

上面程序中的粗体字代码重写了 findClass() 方法,通过重写该方法就可以实现自定义的类加载机制。在本类的 findClass() 方法中先检查需要加载类的 Class 文件是否存在,如果不存在则先编译源文件,再调用 ClassLoader 的 defineClass() 方法来加载这个 Class 文件,并生成相应的 Class 对象。

接下来可以随意提供一个简单的主类,该主类无须编译就可以使用上面的 CompileClassLoader 来运行它。

public class Hello {
 public static void main(String[] args) {
 for (String arg : args) {
  System.out.println("运行Hello的参数:" + arg);
 }
 }
}

本示例程序提供的类加载器功能比较简单,仅仅提供了在运行之前先编译 Java 源文件的功能。实际上,使用自定义的类加载器,可以实现如下常见功能。

  • 执行代码前自动验证数字签名。
  • 根据用户提供的密码解密代码,从而可以实现代码混淆器来避免反编译 *.class 文件。
  • 根据用户需求来动态地加载类。
  • 根据应用需求把其他数据以字节码的形式加载到应用中。

URLClassLoader 类

Java 为 ClassLoader 提供了一个 URLClassLoader 实现类,该类也是系统类加载器和扩展类加载器的父类(此处的父类,就是指类与类之间的继承关系)。URLClassLoader 功能比较强大,它既可以从本地文件系统获取二进制文件来加载类,也可以从远程主机获取二进制文件来加载类。

在应用程序中可以直接使用 URLClassLoader 加载类,URLClassLoader 类提供了如下两个构造器。

  • URLClassLoader(URL[] urls):使用默认的父类加载器创建一个 ClassLoader 对象,该对象将从 urls 所指定的系列路径来查询并加载类。
  • URLClassLoader(URL[] urls, ClassLoader parent):使用指定的父类加载器创建一个 ClassLoader 对象,其他功能与前一个构造器相同。

一旦得到了 URLClassLoader 对象之后,就可以调用该对象的 loadClass() 方法来加载指定类。下面程序示范了如何直接从文件系统中加载 MySQL 驱动,并使用该驱动来获取数据库连接。通过这种方式来获取数据厍连接,可以无须将 MySQL 驱动添加到 CLASSPATH 环境变量中。

public class URLClassLoaderTest {
 private static Connection conn;

 // 定义一个获取数据库连接方法
 public static Connection getConn(String url, String user, String pass) throws Exception {
 if (conn == null) {
  // 创建一个URL数组
   URL[] urls = { new URL("file:mysql-connector-java-5.1.30-bin.jar") };
  // 以默认的ClassLoader作为父ClassLoader,创建URLClassLoader
  URLClassLoader myClassLoader = new URLClassLoader(urls);
  // 加载MySQL的JDBC驱动,并创建默认实例
  Driver driver = (Driver) myClassLoader.loadClass("com.mysql.jdbc.Driver").getConstructor().newInstance();
  // 创建一个设置JDBC连接属性的Properties对象
  Properties props = new Properties();
  // 至少需要为该对象传入user和password两个属性
  props.setProperty("user", user);
  props.setProperty("password", pass);
  // 调用Driver对象的connect方法来取得数据库连接
  conn = driver.connect(url, props);
 }
 return conn;
 }

 public static void main(String[] args) throws Exception {
 System.out.println(getConn("jdbc:mysql://localhost:3306/mysql", "root", "32147"));
 }
}

上面程序中的前两行粗体字代码创建了一个 URLClassLoader 对象,该对象使用默认的父类加载器,该类加载器的类加载路径是当前路径下的 mysql-connector-java-5.1.30-bin.jar 文件,将 MySQL 驱动复制到该路径下,这样保证该 ClassLoader 可以正常加载到 com.mysql.jdbc.Driver 类。

程序的第三行粗体字代码使用 ClassLoader 的 loadClass() 加载指定类,并调用 Class 对象的 newInstance() 方法创建了一个该类的默认实例——也就是得到 com.mysql.jdbc.Driver 类的对象,当然该对象的实现类实现了 java.sql.Driver 接口,所以程序将其强制类型转换为 Driver,程序的最后一行粗体字代码通过 Driver 而不是 DriverManager 来获取数据库连接,关于 Driver 接口的用法读者可以自行查阅API文档。

正如前面所看到的,创建 URLClassLoader 时传入了一个 URL 数组参数,该 ClassLoader 就可以从这系列 URL 指定的资源中加载指定类,这里的 URL 可以以 file: 为前缀,表明从本地文件系统加载;可以以 http: 为前缀,表明从互联网通过 HTTP 访问来加载;也可以以 ftp: 为前缀,表明从互联网通过 FTP访问来加载......功能非常强大。

以上就是浅谈JAVA 类加载器的详细内容,更多关于JAVA 类加载器的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java基于自定义类加载器实现热部署过程解析

    热部署: 热部署就是在不重启应用的情况下,当类的定义即字节码文件修改后,能够替换该Class创建的对象.一般情况下,类的加载都是由系统自带的类加载器完成,且对于同一个全限定名的java类,只能被加载一次,而且无法被卸载.可以使用自定义的 ClassLoader 替换系统的加载器,创建一个新的 ClassLoader,再用它加载 Class,得到的 Class 对象就是新的(因为不是同一个类加载器),再用该 Class 对象创建一个实例,从而实现动态更新.如:修改 JSP 文件即生效,就是利用自定

  • Java语言中的自定义类加载器实例解析

    本文研究的主要是Java语言中的自定义类加载器实例解析的相关内容,具体如下. 自己写的类加载器 需要注意的是:如果想要对这个实例进行测试的话,首先需要在c盘建立一个c://myjava的目录.然后将相应的java文件放在这个目录中.并将产生的.clas文件放在c://myjava/com/lg.test目录下,否则是找不到的.这是要注意的.. class FileClassLoader : package com.lg.test; import java.io.ByteArrayOutputSt

  • java自定义类加载器代码示例

    如果要使用自定义类加载器加载class文件,就需要继承java.lang.ClassLoader类. ClassLoader有几个重要的方法: protectedClassLoader(ClassLoaderparent):使用指定的.用于委托操作的父类加载器创建新的类加载器. protectedfinalClass<?>defineClass(Stringname,byte[]b,intoff,intlen):将一个byte数组转换为Class类的实例. protectedClass<

  • Java类加载器和类加载机制实例分析

    本文实例讲述了Java类加载器和类加载机制.分享给大家供大家参考,具体如下: 一 点睛 1 类加载器负责将.class文件(可能在磁盘上,也可能在网络上)加载到内存中,并为之生成对应的java.lang.Class对象. 2 当JVM启动时,会形成由三个类加载器组成的初始类加载器层次结构: Bootstrap ClassLoader:根类加载器. Extension ClassLoader:扩展类加载器. System ClassLoader:系统类加载器. 3 JVM的类加载机制主要有如下三种

  • Java类加载器ClassLoader用法解析

    这篇文章主要介绍了Java类加载器ClassLoader用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 正文 当程序主动使用某个类时,如果该类还未被加载到内存中,则JVM会通过加载.连接.初始化3个步骤来对该类进行初始化.如果没有意外,JVM将会连续完成3个步骤,所以有时也把这个3个步骤统称为类加载或类初始化. 一.类加载过程 1.加载 加载指的是将类的class文件读入到内存,并为之创建一个java.lang.Class对象,也就是说

  • Java的ThreadContext类加载器的实现

    疑惑 以前在看源码的时候,总是会遇到框架里的代码使用Thread.currentThread.getContextClassLoader()获取当前线程的Context类加载器,通过这个Context类加载器去加载类. 我们平时在程序中写代码的时候,遇到要动态加载类的时候,一般使用Class.forName()的方式加载我们需要的类.比如最常见的,当我们进行JDBC编程的时候,我们通过Class.forName()去加载JDBC的驱动. try { return Class.forName("o

  • java用类加载器的5种方式读取.properties文件

    用类加载器的5中形式读取.properties文件(这个.properties文件一般放在src的下面) 用类加载器进行读取:这里采取先向大家讲读取类加载器的几种方法:然后写一个例子把几种方法融进去,让大家直观感受.最后分析原理.(主要是结合所牵涉的方法的源代码的角度进行分析) 这里先介绍用类加载器读取的几种方法: 1.任意类名.class.getResourceAsStream("/文件所在的位置");[文件所在的位置从包名开始写] 2.和.properties文件在同一个目录下的类

  • Java实现的自定义类加载器示例

    本文实例讲述了Java实现的自定义类加载器.分享给大家供大家参考,具体如下: 一 点睛 1 ClassLoader类有如下两个关键方法: loadClass(String name, boolean resolve):该方法为ClassLoader的入口点,根据指定的二进制名称来加载类,系统就是调用ClassLoader的该方法来获取指定类对应的Class对象. findClass(String name):根据二进制名称来查找类. 如果需要实现自定义的ClassLoader,可以通过重写以上两

  • java类的加载过程以及类加载器的分析

    我们知道,我们写的java代码保存的格式是 .java, java文件被编译后会转换为字节码,字节码可以在任何平台通过java虚拟机来运行,这也是java能够跨平台的原因. 那JVM是如何来让我们写的java文件运行的呢? 这个问题通常的问法好像是:类是如何被加载的. 记得第一次遇见这个问题的时候,同学给我的回答是: 1.虚拟机会加载JDK里类的核心包 2.虚拟机会加载JDK里类的扩展包 3.虚拟机会加载JDK里类的系统包 4.虚拟机再会加载我们写好的java类. 初学的时候,大家都这么说,好像

  • 浅谈JAVA 类加载器

    类加载机制 类加载器负责加载所有的类,系统为所有被载入内存中的类生成一个 java.lang.Class 实例.一旦一个类被载入 JVM 中,同个类就不会被再次载入了.现在的问题是,怎么样才算"同一个类"? 正如一个对象有一个唯一的标识一样,一个载入 JVM 中的类也有一个唯一的标识.在 Java 中,一个类用其全限定类名(包括包名和类名)作为标识:但在 JVM 中,一个类用其全限定类名和其类加载器作为唯一标识.例如,如果在 pg 的包中有一个名为 Person 的类,被类加载器 Cl

  • 浅谈Java内存区域与对象创建过程

    一.java内存区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有的区域则依赖用户线程的启动和结束而建立和销毁.根据<Java虚拟机规范(JavaSE7版)>的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域. 1.程序计数器(线程私有) 程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码

  • 浅谈Java反射与代理

    Java反射机制与动态代理,使得Java更加强大,Spring核心概念IoC.AOP就是通过反射机制与动态代理实现的. 1 Java反射 示例: User user = new User(); user.setTime5Flag("test"); Class<?> cls = Class.forName("com.test.User"); //接口必须public,无论是否在本类内部使用!或者使用cls.getDeclaredMethod(),或者遍历修

  • 浅谈java项目与javaweb项目导入jar包的区别

    现在的项目基本上都是java web项目,所以导入jar包会出现问题,主要介绍一下java项目与javaweb项目的区别: java项目: 在classLoader加载jar和class的时候,是分开加载的,一般jar导入分两种: 1.在web-inf下的lib中直接引入 2.在user library上引入 无论以上哪种引入,jar包都能加载并且运行,classLoader会智能加载(本地JRE运行) javaweb项目: 不是通过本地的JRE运行的,而是部署到web服务器(比如tomcat,

  • 浅谈Java中的class类

    Class 类是在Java语言中定义一个特定类的实现.一个类的定义包含成员变量,成员方法,还有这个类实现的接口,以及这个类的父类.Class类的对象用于表示当前运行的 Java 应用程序中的类和接口. 比如:每个数组均属于一个 Class 类对象,所有具有相同元素类型和维数的数组共享一个Class 对象.基本的 Java 类型(boolean, byte, char, short,int, long, float 和 double) 和 void 类型也可表示为 Class 对象. 以下示例使用

  • 浅谈java 中文件的读取File、以及相对路径的问题

    一.对于java项目中文件的读取 1.使用System 或是 系统的Properties对象 ①直接是使用 String relativelyPath=System.getProperty("user.dir"); ②使用Properties对象 我们先来遍历一下系统的属性: Properties properties = System.getProperties(); Enumeration pnames = properties.propertyNames(); while (pn

  • 浅谈Java内存区域划分和内存分配策略

    如果不知道,类的静态变量存储在那? 方法的局部变量存储在那? 赶快收藏 Java内存区域主要可以分为共享内存,堆.方法区和线程私有内存,虚拟机栈.本地方法栈和程序计数器.如下图所示,本文将详细讲述各个区域,同时也会讲述创建对象过程,内存分配策略, 和对象访问定位原理.觉得写得好的,可以点个收藏,绝对不亏. Java内存区域 程序计数器 程序计数器,可以看作程序当前线程所执行的字节码行号指示器.字节码解释器工作时就是通过改变计数器的值来选取下一条需要执行的字节码指令,分支.循环.跳转.异常处理都需

  • 浅谈Java垃圾回收机制

    一.什么是垃圾 java中,什么样的对象是垃圾?有人说:没有被引用的对象就是垃圾对象.我一开始对此也是深信不疑的,但是当年我这么回答面试官的时候,得到的是一个大大的白眼. 判断一个对象是否是垃圾,有两种算法,一种是引用计数法,但是,这种方法解决不了循环引用的问题. /**循环问题*/ public class Demo{ public Demo instance; public static void main(String[] args) { Demo a=new Demo(); Demo b

  • 浅谈Java类的加载,链接及初始化

    一 类生命周期 Loading Linking(Verification.Preparation.Resolution) Initializing 二 类加载器 1 图解 2 代码 package jvm; public class T002_ClassLoadLevel { public static void main(String[] args) { System.out.println(String.class.getClassLoader()); System.out.println(

  • 浅谈Java 代理机制

    目录 一.常规编码方式 二.代理模式概述 三.静态代理 3.1.什么是静态代理 3.2.代码示例 四.Java 字节码生成框架 五.什么是动态代理 六.JDK 动态代理机制 6.1.使用步骤 6.2.代码示例 七.CGLIB 动态代理机制 7.1.使用步骤 7.2.代码示例 八.什么情况下使用动态代理 九.静态代理和动态代理对比 十.总结 一.常规编码方式 在学习代理之前,先回顾以下我们的常规编码方式:所有 interface 类型的变量总是通过向上转型并指向某个实例的. 1)首先,定义一个接口

随机推荐