搞清楚两者之间区别前,我们来了解下类加载过程。
  
  一、类加载过程
  
  1、加载
  
  通过一个类的全限定名来获取定义此类的二进制字节流。
  
  将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。
  
  在内存中生成一个代表这个类的Java.lang.class对象,作为方法区这个类的各种数据的访问入口。
  
  2、验证
  
  文件格式验证
  
  元数据验证
  
  字节码验证
  
  符号引用验证
  
  3、准备
  
  准备阶段是正式为类变量(仅被static修饰的变量)分配内存并设置类变量初始值的阶段,这些变量所使用的内存都将在方法区中进行分配。注意,这里所说的初始值“通常情况”下是数据类型的零值,“特殊情况”下是初始化真实的值,即常量值(static final)。
  
  4、解析
  
  解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。注意,这个阶段不一定要有。
  
  5、初始化
  
  类初始化阶段是类加载过程的最后一步,真正开始执行类中定义的Java程序代码。初始化类中的所有类变量的赋值和静态语句块。
  
  有空可以看看《【JVM基础知识】java类加载机制》
  
  二、源码分析
  
  1、JDK8中Class.forName()源码
  
  @CallerSensitive
  
  public static Class<?> forName(String className)
  
  throws ClassNotFoundException {
  
  Class<?> caller = Reflection.getCallerClass(www.yongshiyule178.com);
  
  return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
  
  }
  
  @CallerSensitive
  
  public static Class<?> forName(String name, boolean initialize, ClassLoader loader)
  
  throws ClassNotFoundException
  
  {
  
  Class<?> caller = null;
  
  SecurityManager sm = System.getSecurityManager();
  
  if (sm != null) {
  
  // Reflective call to get caller class is only needed if a security manager
  
  // is present. Avoid the overhead of making this call otherwise.
  
  caller = Reflection.getCallerClass();
  
  if (sun.misc.VM.isSystemDomainLoader(loader)) {
  
  ClassLoader ccl www.hengtongyoule.com= ClassLoader.getClassLoader(caller);
  
  if (!sun.misc.VM.isSystemDomainLoader(www.gaozhuoyiqi.com)) {
  
  sm.checkPermission(
  
  SecurityConstants.GET_CLASSLOADER_PERMISSION);
  
  }
  
  }
  
  }
  
  return forName0(name, initialize, loader, caller);
  
  }
  
  Class.forName(className)方法,内部实际调用的方法是forName0这个方法,在这个forName0方法中的第二个参数被默认设置为了true,这个参数代表是否对加载的类进行初始化,设置为true时会类进行初始化,代表会执行类中的静态代码块,以及对静态变量的赋值等操作。
  
  当然,可以通过Class.forName(String name, boolean initialize,ClassLoader loader)方法来手动选择在加载类的时候是否要对类进行初始化。
  
  2、JDK8中ClassLoader.loadClass()源码
  
  public Class<?> loadClass(String name) throws ClassNotFoundException {
  
  return loadClass(name,www.huarenyl.cn false);
  
  }
  
  ClassLoader.loadClass(className)方法,内部实际调用的方法是loadClass(name, false);
  
  第2个 boolean参数,表示目标对象是否进行链接,false表示不进行链接,即不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行。
  
  3、结论
  
  所以Class.forName加载类是将类进了初始化,而ClassLoader的loadClass并没有对类进行初始化,只是把类加载到了虚拟机中。
  
  三、应用场景
  
  1、JDBC时通常是使用Class.forName()方法来加载数据库连接驱动。因为在JDBC规范中明确要求Driver(数据库驱动)类必须向DriverManager注册自己,而注册的操作都在静态代码块中。
  
  2、Spring框架中的IOC的实现就是使用的ClassLoader。
  
  参考
  
  反射中Class.forName()和ClassLoader.loadClass()的区别
  
  在Java的反射中,Class.forName和ClassLoader的区别

反射中Class.forName()和classLoader的区别的更多相关文章

  1. Java反射中Class.forName和classloader的区别

    Java中Class.forName和classloader都可以用来对类进行加载. Class.forName除了将类的.class文件加载到jvm中之外,还会对类进行解释,执行类中的static块 ...

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

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

  3. 在Java的反射中,Class.forName和ClassLoader的区别

    前言 最近在面试过程中有被问到,在Java反射中Class.forName()加载类和使用ClassLoader加载类的区别.当时没有想出来后来自己研究了一下就写下来记录一下. 解释 在java中Cl ...

  4. 反射中,Class.forName 和 classloader 的区别

    https://blog.csdn.net/qq_27093465/article/details/52262340 java中class.forName()和classLoader都可用来对类进行加 ...

  5. 在 Java 的反射中,Class.forName 和 ClassLoader 的区别

    1. 解释 在java中Class.forName()和ClassLoader都可以对类进行加载.ClassLoader就是遵循双亲委派模型最终调用启动类加载器的类加载器,实现的功能是“通过一个类的全 ...

  6. Class.forName和ClassLoader的区别

    一 看名字就知道了,一个是类的创建,一个类加载器 二 再看下Class.forName源码,调用了ClassLoader @CallerSensitive public static Class< ...

  7. class.forName 和 classLoader的区别

    Java中的Class.forName()和ClassLoader都可以用来对类进行加载.Class.forName()除了将类的.class文件加载到JVM中 还会对类进行解释,执行类中的stati ...

  8. class.forName() 和 classLoader 的区别

    相同点:        java中class.forName() 和 classLoader 都可用来对类进行加载 不同店:        1.class.forName()除了将类的 .class ...

  9. java反射中,Class.forName和classloader的区别

    http://blog.csdn.net/qq_27093465/article/details/52262340

随机推荐

  1. 多线程系列之八:Thread-Per-Message模式

    一,Thread-Per-Message模式 翻译过来就是 每个消息一个线程.message可以理解为命令,请求.为每一个请求新分配一个线程,由这个线程来执行处理.Thread-Per-Message ...

  2. this is incompatible with sql_mode=only_full_group_by

    mysql命令gruop by报错this is incompatible with sql_mode=only_full_group_by - Jim_.NET - 博客园 http://www.c ...

  3. 基于redis实现的点赞功能设计思路详解

    点赞其实是一个很有意思的功能.基本的设计思路有大致两种, 一种自然是用mysql等 数据库直接落地存储, 另外一种就是利用点赞的业务特征来扔到redis(或memcache)中, 然后离线刷回mysq ...

  4. HashMap深度解析(转载)

    原文地址:http://blog.csdn.net/ghsau/article/details/16890151 实现原理:用一个数组来存储元素,但是这个数组存储的不是基本数据类型.HashMap实现 ...

  5. mybatis出现NoSuchMethodException异常

    今天在idea中调试项目(ssm搭建的项目)的时候,mybatis突然出现了NoSuchMethodException异常,具体的异常时: java.lang.NoSuchMethodExceptio ...

  6. [转帖]一键获取 所有连接过的wifi 密码

    cmd 一键获取 所有连接过的wifi 密码 转帖来源: http://www.cnblogs.com/hookjoy/p/5537623.html for /f "skip=9 token ...

  7. Plugin/Preset files are not allowed to export objects,webpack报错/babel报错的解决方法

    1.为什么会报错 ? 这里抱着错误是因为 babel 的版本冲突. 多是因为你的 babel 依赖包不兼容. 可以查看你的 package.json 的依赖列表 即有 babel 7.0 版本的( @ ...

  8. Javaweb小结之——JavaBean+持久层

    数据持久层学习了JDBC.连接池以及DBUtil 思考一下,在学会使用SSM框架之前,还是先多使用DBUtil吧 数据库持久层的JDBC.连接池和DBUtil,这个链接给我了一些参考https://b ...

  9. python 编码格式

    1. 字符编码简介 1.1. ASCII ASCII(American Standard Code for Information Interchange),是一种单字节的编码.计算机世界里一开始只有 ...

  10. delphi中如何实现DBGrid中的两列数据想减并存入另一列

    可参考下面的例子:   数据自动计算的实现:“金额”是由“单价”和“工程量”相乘直接得来的,勿需人工输入. 这可在“数据源构件”的onupdatedata例程添加如下代码实现: procedure T ...