声明:    本篇博客绝大多数内容为《Spring3.x企业开发应用实战》一书原内容,所有版权归原书作者所有!,仅供学习参考,勿作他用!

3.2 相关Java基础知识

Java语言允许通过程序化的方式间接对Class对象实例操作,Class文件由类装载器装在后,在JVM(Java虚拟机)中将形成一份描述Class结构的元信息对象,通过该元信息对象可以获知Class的结构信息: 如构造函数、属性和方法等。Java允许用户借由这个Class相关的元信息对象间接调用Class对象的功能,这就为使用程序化方式操作CLass对象开辟了途径。

3.2.1 简单实例

我们将从一个简单例子开始探访Java反射机制。

该类是测试所需要用到的主体类。

package com.baobaotao.reflect;

public class Car {
 private String brand;

private String color;

private int maxSpeed;

public Car(){System.out.println("init car!!");}
 public Car(String brand,String color,int maxSpeed){
  this.brand = brand;
  this.color = color;
  this.maxSpeed = maxSpeed;
 }
 public void introduce() {
       System.out.println("brand:"+brand+";color:"+color+";maxSpeed:"+maxSpeed);
 }

public String getBrand() {
  return brand;
 }

public void setBrand(String brand) {
  this.brand = brand;
 }

public String getColor() {
  return color;
 }

public void setColor(String color) {
  this.color = color;
 }

public int getMaxSpeed() {
  return maxSpeed;
 }

public void setMaxSpeed(int maxSpeed) {
  this.maxSpeed = maxSpeed;
 }
}

一般情况下,我们会使用如下的代码创建Car的实例:

Car car = new Car( );

car.setBrand("some brand");

或者

Car car = new Car("some brand","some color",);

以上两种方式都是采用传统方式直接地调用目标类的方法创建对象,下面我们通过Java反射机制以一种更加通用的方法间接地操作目标类,从而创建对象。

package com.baobaotao.reflect;

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;

public class ReflectTest {
 
 /*
  * 需求:
  *  以反射的方式获取com.baobaotao.reflect.Car类的对象。并设置其实例域的属性,最后输出。   
  */
 public static Car  initByDefaultConst() throws Throwable {
  //通过当前线程对象获取类装载器
  ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
  
  //通过类装载器加载com.baobaotao.reflect.Car类的字节码文件,从而获取到该类的Class对象。
  //类可能找不到,所以需要抛出异常。
  Class clazz = classLoader.loadClass("com.baobaotao.reflect.Car");
  
  //通过该类的默认(无参)构造方法创建对象
  Constructor cons = clazz.getDeclaredConstructor();
  Car car = (Car) cons.newInstance();
  //当然也可以通过该类的Class对象创建该类的实例
  //Car car = (Car) clazz.newInstance();
  
  
  //通过反射方法设置属性
  Method setBrand = clazz.getMethod("setBrand", String.class);
  setBrand.invoke(car, "红旗CA72");
  Method setColor = clazz.getMethod("setColor", String.class);
  setColor.invoke(car,"黑色");
  Method setMaxSpeed = clazz.getMethod("setMaxSpeed", int.class);
  setMaxSpeed.invoke(car, 200);
  
  return car;
 }
 
 public static void main(String[] args) throws Throwable {
  Car car = initByDefaultConst();
  car.introduce();
 }

}

结果:

init car!!
brand:红旗CA72;color:黑色;maxSpeed:200

这说明我们完全可以通过编程的方式调用Class的各项功能。这和直接通过构造函数和方法调用类功能的效果是一致的,不过前者是间接调用,后者是直接调用罢了。

在ReflectTest中,我们使用了几个重要的反射类: ClassLoader、Class、Constructor、Method。

ClassLoader:    用于将指定路径下的类的字节码文件装载到JVM中。注意,其方法ClassLoader#loadClass的参数一定要是类的全限定名称 —— 包名.类名

创建目标类的对象:

1 可以通过Class类的newInstance方法创建对象。这样创建的对象是通过默认构造器创建的,所以目标类Car中必须相应地给出默认构造器。

2 通过Constructor类创建。通过目标类对应的Class类的对象的getDeclaredConstructor方法创建Construcor类对象,该类描述了目标类的构造方法的信息。可以通过其newInstance()方法和newInstance(paramTypes)来分别创建不带参的对象和带参的对象。

如:

//已经获取目标类对应的Class类的对象clazz

//得到的未设置实例域的对象(不带参的)

Car car  = (Car)clazz.getDeclaredConstructor( ).newInstance( );

//得到的已设置实例域的对象(带参的)

(Car)clazz.getDeclaredConstructor(String.class,String.class,int.class).newInstance("红旗CA72","黑色",200);

3.2.2 类装载器ClassLoader

    类装载器工作原理

类装载器就是寻找类的字节码文件并构造出类再JVM内部的表示的对象组件。在Java中,将一个类装载到JVM中需要经过以下步骤:

1 装载:    查找和导入Class文件;

2 链接:    执行校验’准备和解析步骤,其中解析步骤是可以选择的:

a)   校验:    检查载入的Class文件的正确性。

b)   准备:    给类的静态变量分配内存空间。

c)    解析:    将符号引用转成直接引用。

类装载器工作由ClassLoader及其子类负责,ClassLoader是一个重要的Java运行时组件,它负责在运行时查找和装入Class字节码文件。JVM在运行时会产生三个类装载器:  根装载器、ExtClassLoader(扩展类装载器)、和AppClassLoader(系统装载器)。其中,根装载器不是ClassLoader的子类,它由C++编写,因此我们在Java中看不到它。

默认情况下,使用AppClassLoader装载应用程序的类。我们可以做个试验。

public static void main(String[] args) throws Throwable {
  ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
  System.out.println("current:"+classLoader);
  System.out.println("parrent:"+classLoader.getParent());
  System.out.println("grandparrent:"+classLoader.getParent().getParent());
 }

结果;

current:sun.misc.Launcher$AppClassLoader@73d16e93
    parrent:sun.misc.Launcher$ExtClassLoader@6d06d69c
    grandparrent:null

通过以上信息,可知装载当前类的装载器为AppClassLoader,父装载器是ExtClassLoader,祖父装载器是根装载器,因为在Java中找不到其句柄,所以直接返回null。

JVM装载类装载类时使用”全盘负责委托机制“,”全盘负责“指的是当一个类装载器装载类时,默认会选择它该类和其依赖和引用的类,除非显式地声明了使用别的装载器。”委托机制“指的是首先委托父装载器装载目标类的字节码文件,只有在找不到的情况下才从自己的类路径中查找并装载目标类。这是出于安全的考虑。试想,如果某人编写了一个恶意的基础类(如java.lang.String),并装载到了JVM中,这会造成多么可怕的后果!而”全盘委托负责机制“中,基础类永远是由根装载器装载,这便可以杜绝这种现象的发生。

实战经验:

许多Java开发者都会遇到这样的异常: java.lang.NoSuchMethodError。这便是JVM的“全盘委托机制”造成的。出现该错误说明存在不同版本的类包。

如commons-lang 2.x.jar和commons-lang3.x.jar都位于类路径中。代码中用到了后者的某个方法,而该方法在前者中并不存在,但是JVM又碰巧从前者中装载类。这便出现了该错误。

 菊子曰微博群发没烦恼!

Spring3.x企业开发应用实战读书笔记 —— 第三章IoC容器概述的更多相关文章

  1. 《Linux内核设计与分析》第六周读书笔记——第三章

    <Linux内核设计与实现>第六周读书笔记——第三章 20135301张忻估算学习时间:共2.5小时读书:2.0代码:0作业:0博客:0.5实际学习时间:共3.0小时读书:2.0代码:0作 ...

  2. Spring AOP (Spring 3.x 企业应用开发实战读书笔记第六章)

    从面相对象编程到面相切面编程,是一种代码组织方式的进化. 每一代的代码组织方式,其实是为了解决当时面对的问题.比如写编译器和写操作系统的时候的年代当然要pop,比如写界面的时候当然要oop,因为界面这 ...

  3. 《CSS3实战》读书笔记 第三章:选择器:样式实现的标记

    第三章:选择器:样式实现的标记 选择器的魔力在于,让你完全实现对网页样式的掌控.不同的选择器可以用在不同的情况下使用.总之把握的原则是:规范的编码,根据合理地使用选择器,比去背选择器的定义有价值的多. ...

  4. maven实战读书笔记(三)

    maven将一系列的步骤都封装为一系列的插件,运行命令后一系列的插件运行

  5. jQuery 实战读书笔记之第二章:选择元素

    基本选择器 html 代码如下,后面的 js 使用的 html 基本大同小异. <!doctype html> <html> <head> <title> ...

  6. 《R语言实战》读书笔记--第三章 图形初阶(二)

    3.4添加文本.自定义坐标轴和图例 很多作图函数可以设置坐标轴和文本标注.比如标题.副标题.坐标轴标签.坐标轴范围等.需要注意的是并不是所有的绘图函数都有上述的参数,需要进行验证.可以将一些默认的参数 ...

  7. 《R语言实战》读书笔记--第三章 图形初阶(一)

    3.1使用图形 可以使用pdf等函数将图形直接保存在文件中.在运用attach和detach函数的使用中经常出现错误,比如命名重复的问题,所以,应该尽量避免使用这两个函数. plot是一般的画图函数, ...

  8. Maven实战读书笔记(三):Maven依赖

    3.1 依赖的配置 一个依赖声明可以包含下面元素: <dependencies> <dependency> <groupId></groupId> &l ...

  9. Maven实战读书笔记(一):Maven概述

    1.1 Maven是什么,能做什么 Maven是一个跨平台的项目管理工具,主要服务于Java平台的项目构建.依赖管理和项目信息管理. Maven的用途之一是项目构建,能够自动化构建过程,从清理.编译. ...

随机推荐

  1. 拆分字符and读取properties文件

    在方法里面建立properties对象 Properties pps = new Properties(); 调用.load()方法 pps.load(new FileInputStream(&quo ...

  2. Linux 定期自动备份文件夹脚本

    根据网上代码改编而成,这个脚本分为两个文件,一个是执行文件backup.sh,另一个是备份项目文件backup.txt.backup.txt这个文件比较好理解,就是一个text文件,它的每一行表示一个 ...

  3. Linux磁盘分区(二):删除

    ***********************************************声明************************************************ 原创 ...

  4. 自定义spring mvc的json视图

    场景 前端(安卓,Ios,web前端)和后端进行了数据的格式规范的讨论,确定了json的数据格式: { "code":"200", "data&quo ...

  5. Coursera课程 Programming Languages, Part A 总结

    Coursera CSE341: Programming Languages 感谢华盛顿大学 Dan Grossman 老师 以及 Coursera . 碎言碎语 这只是 Programming La ...

  6. 过渡与动画 - steps调速函数&CSS值与单位之ch

    写在前面 上一篇中我们熟悉五种内置的缓动曲线和(三次)贝塞尔曲线,并且基于此完成了缓动效果. 但是如果我们想要实现逐帧动画,基于贝塞尔曲线的调速函数就显得有些无能为力了,因为我们并不需要帧与帧之间的过 ...

  7. Windows中根据端口定位持有者程序

    之前被一个问题所困扰, 电脑过一段时间就无法上网, 查询netstat, 会有大量的端口被占用, 但是看不出来是哪个程序开启的 ( 最终定位到是ADSafe搞得鬼 ) . 后来发现几个命令可以根据端口 ...

  8. Win10系统下的Tomcat7.0配置

    为什么不用更高版本的Tomcat呢?好几个老师都说7.0的版本最好用,所以就推荐用这个.安Tomcat之前,我电脑上装的是jdk 9,安装了好几次Apache都不能启动,日志显示的是"启动报 ...

  9. C# 把Div变为滚动条

    <div runat="server" style="overflow:auto;width:350px;height:200px" > <a ...

  10. 阿里云轻量应用服务器Lamp部署php工程踩过的坑

    第一次写博客,也不知道写什么,但是想坚持写博客来提升自己,不喜勿喷. 切回正题,使用阿里云的轻量应用服务器Lamp其实非常方便,价格也很便宜,一键购买需要的环境都帮你搭配好了,剩下的就是自己修改一下数 ...