最近在研读《深入理解java虚拟机:JVM高继特性与最佳实践》第二版,

今天想起来很久前,写数据库连接,使用Class.forName,当时没有深究,所以便简单的看了下源码,顺便做了以下记录:

以下为简单的数据库连接代码:        
     String driver = "com.mysql.jdbc.Driver";
String url = "jdbc:mysql://localhost:3306/db_iot";
String username = "root";
String password = "123456";
Connection conn = null; try {
Class.forName(driver);
conn = (Connection) DriverManager.getConnection(url,username,password);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (SQLException e) {
e.printStackTrace();
}
//Class.forName源码:
public static Class<?> forName(String className)
throws ClassNotFoundException {
return forName0(className, true,      //这里的true是是否初始化的标志,默认初始化
ClassLoader.getClassLoader(Reflection.getCallerClass()));
}

  

  

结合JVM的相关知识:类加载分为 加载,连接(验证,准备(类静态成员设置类型默认值),解析),初始化(类静态成员初始化,不包括构造器的初始化),使用,卸载

  //以下为com.mysql.jdbc.Driver的静态代码块
  static {
try {
DriverManager.registerDriver(new Driver());//registerDriver方法会将driver存入到类DriverManger的静态成员中去
} catch (SQLException var1) {
throw new RuntimeException("Can\'t register driver!");
}
}

  因此,使用Class.forName(driver);的结果是:driver被初始化,driver被加载的DriverManager中,可被查看及使用。

但如果使用classLoader.loadClass("className");则是另一回事了,下面是classLoader.loadClass的源码:

    public Class<?> loadClass(String name) throws ClassNotFoundException {
return loadClass(name, false);//看jdk的注解知道,参数false表示是否解析,即尚未初始化
}

  所以,若使用classLoader.loadClass,则driver并不会被加载道DriverManager中。

以下为补充的类加载步骤:

@.加载:

1.通过“类全名”定位到类文件,并获取该类文件的二进制字节流;

2.class文件二进制字节流中的静态数据结构转换为方法区中动态的运行时数据结构;

3.在java堆中生成一个代表这个类的java.lang.Class对象,作为方法区中该类相关数据的访问入口。

@.连接-验证:

1.验证文件的各式是否正确:是否以魔数0xCAFEBABE开头,紧接着魔数是不是正确的主次版本号,主版本号最小45等等验证;

2.验证元数据是否合法:该类是否有父类(除了java.lang.object外都当有父类),是否继承了final修饰的类等等;

3.验证字节码是否安全:确保被验证类的方法在运行时不会做出危害虚拟机安全的行为,如保证跳转命令不会跳转到方法体意外的字节码命令上;

4.符号应用验证:对当前类意外的信息进行匹配性验证,如通过“类全限定名”是否可找到对应的类,方法是否可访问等等。

需要重视的是:Java类在编译期会被检查验证,但是我们无法保证,class文件不会被篡改,所以,虚拟机的验证期是非常重要的。

@连接-准备:

1.为类成员变量分配内存设置类型默认值(比如int,则设为0),这些内存都将在方法区中进行分配,但需要注意的是,若静态成员变量被final修饰,则此时会直接赋值,而不是取类型默认值。

@连接-解析:

1.,将类的符号应用(单纯的符号字面量,不涉及jvm内存)转为直接应用(直接定位到目标的内存地址);

@初始化:

1.“初始化”是类加载过程的最后一步,在此阶段,类的静态成员将会被赋值(静态方法将会被执行);

上面反复提到了方法区,以下为方法区的解释:

方法区:方法区是jvm运行时数据结构中的一部分,为所有线程共享(线程安全)。

方法区中主要存放类信息:

1.类的全限定名,直接超类的全限定名(如果是Object,则没有超类),类的类型(类还是接口),类的访问修饰符(public,abstract,final等),所有的直接接口的全限定名,常量池;

2.常量池中主要存放该类的 字段,方法信息,类变量信息,装在该类的装载器的引用,类型引用等。

------------------------------------------写到后来,就成了背诵了,卡壳了还要翻书---------------------------------------------------------

Class.forname和ClassLoader.loadClass的源码分析的更多相关文章

  1. 深度分析 Java 的 ClassLoader 机制(源码级别)

    写在前面:Java中的所有类,必须被装载到jvm中才能运行,这个装载工作是由jvm中的类装载器完成的,类装载器所做的工作实质是把类文件从硬盘读取到内存中,JVM在加载类的时候,都是通过ClassLoa ...

  2. 深度分析Java的ClassLoader机制(源码级别)

    写在前面:Java中的所有类,必须被装载到jvm中才能运行,这个装载工作是由jvm中的类装载器完成的,类装载器所做的工作实质是把类文件从硬盘读取到内存中,JVM在加载类的时候,都是通过ClassLoa ...

  3. 深度分析 Java 的 ClassLoader 机制(源码级别)(转)

    写在前面:Java中的所有类,必须被装载到jvm中才能运行,这个装载工作是由jvm中的类装载器完成的,类装载器所做的工作实质是把类文件从硬盘读取到内存中,JVM在加载类的时候,都是通过ClassLoa ...

  4. 【JVM】深度分析Java的ClassLoader机制(源码级别)

    原文:深度分析Java的ClassLoader机制(源码级别) 为了更好的理解类的加载机制,我们来深入研究一下ClassLoader和他的loadClass()方法. 源码分析 public abst ...

  5. 反射中Class.forName()和ClassLoader.loadClass()的区别

    一 Java类装载过程 装载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象: 链接:执行下面的校验.准备和解析步骤,其 ...

  6. 【转】深度分析Java的ClassLoader机制(源码级别)

    原链接 Java中的所有类,必须被装载到jvm中才能运行,这个装载工作是由jvm中的类装载器完成的,类装载器所做的工作实质是把类文件从硬盘读取到内存中, JVM在加载类的时候,都是通过ClassLoa ...

  7. 别翻了,这篇文章绝对让你深刻理解java类的加载以及ClassLoader源码分析【JVM篇二】

    目录 1.什么是类的加载(类初始化) 2.类的生命周期 3.接口的加载过程 4.解开开篇的面试题 5.理解首次主动使用 6.类加载器 7.关于命名空间 8.JVM类加载机制 9.双亲委派模型 10.C ...

  8. Class.forName() 与 ClassLoader.loadClass()的区别

        看到一个面试题,说说Class.forName() 与 ClassLoader.loadClass()的区别,特意记录一下,方便后续查阅.     在我们写java代码时,通常用这两种方式来动 ...

  9. 015 反射中的 Class.forName() 与 ClassLoader.loadClass() 的区别

    作者:nnngu GitHub:https://github.com/nnngu 博客园:http://www.cnblogs.com/nnngu 简书:https://www.jianshu.com ...

随机推荐

  1. Building Lambda Architecture with Spark Streaming

    The versatility of Apache Spark’s API for both batch/ETL and streaming workloads brings the promise ...

  2. 【转】Android 增,删,改,查 通讯录中的联系人

    一.权限 操作通讯录必须在AndroidManifest.xml中先添加2个权限, <uses-permission android:name="android.permission. ...

  3. springBoot集成redisCluster

    本文主要内容:springBoot简介,在SpringBoot中如何集成Redis,可配置Redis集群. 关于SpringBoot 你想要的,这里都有:https://spring.io/proje ...

  4. WPF: 自动设置Owner的ShowDialog 适用于MVVM

    原文:WPF: 自动设置Owner的ShowDialog 适用于MVVM 原文地址:http://www.mgenware.com/blog/?p=339 WPF中的Windows的ShowDialo ...

  5. 分享Winform datagridview 动态生成中文HeaderText

    缘起 很久以前给datagridview绑定列的时候都是手动的,记得以前用Display自定义属性来动态给datagridview绑定列.后来发现不行,于是还在博问发了问题: 后来热心网友帮我回答了这 ...

  6. django 之 stark组件

    ----------------------------------------------------------------烦恼没完没了,内心动荡不安,呜呼哀哉. 一.有个特殊的需求,需要用sta ...

  7. 全局css控制<td>标签属性

    td { text-align: center; /*设置水平居中*/ vertical-align: middle; /*设置垂直居中*/    height:50px;             / ...

  8. rocketmq双主模式

    1.官网 https://rocketmq.apache.org/ 官方安装文档 https://rocketmq.apache.org/docs/quick-start/ 2.rocketmq多主配 ...

  9. 一些char和byte对应的字符

    1: 2: 3: 4: 5: 6: 7: 8: 9: 10: 11: 12: 13: 14: 15: 16: 17: 18: 19: 20: 21: 22: 23: 24: 25: 26: 27: 2 ...

  10. springboot整合redis(注解形式)

    springboot整合redis(注解形式) 准备工作 springboot通常整合redis,采用的是RedisTemplate的形式,除了这种形式以外,还有另外一种形式去整合,即采用spring ...