声明:    本篇博客绝大多数内容为《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. Echarts后台封装option对象

    该方法返回的keyword指向了前台负责图表显示的jsp页面 public String keyword() { if(this.dateNum == null || this.dateNum.equ ...

  2. js屏蔽广告

    最近遇到有些广告的问题,首先是在手机端,可能是用户访问了一些小网站的,(你懂得),然后在访问我的网站时,会带小广告过来,通常是wifi被dns劫持的情况下导入到广告脚本, 1.处理这些要知道广告的根源 ...

  3. PHP 面试时常考的文件操作函数

    前两天面试的时候,笔试要求写一些文件操作类函数,什么遍历目录获取文件扩展名之类的,回来后大致总结了一下PHP中一些常见的文件操作函数分享一下. 1.使用递归删除目录及目录下所有文件的方法. funct ...

  4. SQL Server学习之路(七):Python3操作SQL Server数据库

    0.目录 1.前言 2.准备工作 3.简单测试语句 4.提交与回滚 5.封装成类的写法 1.前言 前面学完了SQL Server的基本语法,接下来学习如何在程序中使用sql,毕竟不能在程序中使用的话, ...

  5. 局域网下的html注入及DNS劫持

    之前研究Dsploit的部分功能实现原理,现在了解的差不多了,简要记录下Dsploit的断网攻击,html注入,图片替换,cookie劫持的原理.本篇blog需要有一定的网络知识基础在看. 假设现有三 ...

  6. 实现快餐配送页面jq

    <!DOCTYPE html><html> <head> <meta charset="utf-8"> <title>快 ...

  7. 浅copy 与 深copy

    import copy names = ["zhangyang", 'guyun', 'xiangpeng', ['jack','tom'], 'xuliangchen'] nam ...

  8. java学习笔记之System类

    System类常用总结 System类概述 java.lang.System类,系统属性信息工具类 常用静态方法: 1. public static long currentTimeMillis() ...

  9. LDA数学八卦笔记(一)Gamma函数

    Technorati Tags: LDA主题模型

  10. SQL-删除重复记录

    前几日工作的时候,有个小需求,是要求删除一个表table_A里的重复记录(保留一条),假设以字段COL_PK重复来判断记录重复,那么有几种写法: 在Oracle里,可以利用rowid来删除,这是非常高 ...