最近在研读《深入理解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. ideal中把项目打成war包,并放在tomcat运行,遇见的问题。。。

    先说下我遇见的问题吧:最近做项目要把项目放在tomcat上运行,用的springboot框架, 在建项目时选择的是  jar包,项目写完要部署打包是,在pom中虽然把包改成了war ,可是每次放入to ...

  2. Cocos2d-x游戏开发之lua编辑器 Sublime 搭建,集成cocos2dLuaApi和自有类

    版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/wisdom605768292/article/details/34085969 Sublime Te ...

  3. nodeJS之crypto模块md5和Hmac加密

    nodeJS之crypto模块md5和Hmac加密 原文地址:https://www.cnblogs.com/tugenhua0707/p/9128690.html 在nodejs中,可以使用cryp ...

  4. [转]C#通过委托更新UI(异步加载)

    我们在使用 windowform 编程的时候,我们或许可能会越到,各种在窗体加载的时候,会进行其他的操作: 1.如果是在加载之前进行其它操作,则整个界面出来的很慢,而且若是时间长的话,页面很久才能出来 ...

  5. 使用maven时出现Failure to transfer 错误的解决方法

    在eclipse里使用maven,连接nexus私服. 添加依赖之后,总是报添加的依赖jar文件找不到,但是在nexus的库里面能找到这个依赖的jar文件,但是在本地的maven库里面找不到,于是我将 ...

  6. MySQL的运算符及其优先级

    +++++++++++++++++++++++++++++++++++++++++++标题:MySQL的常见运算符时间:2019年2月23日内容:MySQL的常见运算符重点:主要讲述MySQL常见运算 ...

  7. 小小知识点(六)——算法中的P问题、NP问题、NP完全问题和NP难问题

    转自CSDN默一鸣 https://blog.csdn.net/yimingsilence/article/details/80004032 在讨论算法的时候,常常会说到这个问题的求解是个P类问题,或 ...

  8. 解决Jenkins邮件配置问题

    最近要配置Jenkins邮箱,由于一直报如下错误,又没办法解决,所以想到了另外的办法发邮件. 我采用shell脚本执行python命令的方式来发邮件: #!/bin/bash cd /software ...

  9. java valueOf()函数

    valueOf() 方法用于返回给定参数的原生 Number 对象值,参数可以是原生数据类型, String等. 该方法是静态方法.该方法可以接收两个参数一个是字符串,一个是基数. 语法 该方法有以下 ...

  10. kettle查询-2

    模糊匹配: 1.主数据/查询数据 2.模糊匹配 3.输出:jaro/jaro winkler/pair letters similarity(各自算法的匹配度measure value) http c ...