在JVM中对一个类实例的创建,有两种方式,一种是编译时,一种是运行时。两种方式在开发过程中都是十分重要的。在Java中无时无刻无处不在的Java对象,实例化的过程也就变得尤为引人瞩目。我们经常用new Object()方法来创建实例,而相反的,反射在这个时候就显得突兀,因为有人就会问,Java对象实例的时候为什么不直接new Object就好了呢?其实是情况所迫,有些时候,我们在编译的时候并不知道要什么对象,这个对象的字节码也不知道存在什么地方?是在硬盘,内存,还是遥远的世界另一端的一台电脑中,所以,我们这个时候就用另外一种婉转地方式去编写。

 
分析


       一个类中有不同的portion,而Field(属性),Constructor(构造器),Method(方法)三大重要的部分。我们对一个类的操作也无非就对这个三个部分进行invoke而已。所以通过反射,我们拿到这些attribute用来操作。
       同时我们也要了解,既然在Java中一切皆为对象,那么一个类中的组成部分也是可以单独被拿出来作为各自不同的对象来着。这样子有利于我们对象思维的理解和简单化编写代码。
 
  • 字节码的理解
  • Construtor
  • Field
  • Method
  • Array数组和Object的关系
  • 创建newInstance对象
 

 
 
 
实例讲解

字节码的理解
一切对象在JVM中,在计算机的内存中,在计算机的硬盘中,其实都是以二进制的形式存在。那么我们去理解Java中的字节码,仔细一想,也不就是二进制的内容形式存在于电脑中吗?对的,那么这个时候去了解这个部分,其实就简单得多了。无论是String,Integer,还是Char,说到底就是一段杂乱无序的0011而已。我们忽略掉这个貌似繁杂的部分去看真实的问题所在,说到底,就是一叠数据和符号而已。
 
  1.  String str1 ="abc";
    Class cls1 = str1.getClass();
    Class cls2 =String.class;
    Class cls3 =Class.forName("java.lang.String");
    System.out.println(str1.getClass());
    System.out.println(cls1 == cls2);
    System.out.println(cls3 == cls2);
    /**
    输出结果
    class java.lang.String
    true
    true
    */
 
String的一个实例是str1,所以str1的class就是java.lang.String 所以我们发现String.class和通过Class.forName("java.lang.String")去得到一个Class其实都是String在JVM中的Class。
除此之外,还有一些有关于Class这个类的方法(一切皆为对象,对象本身也是一个对象,有属于自己的特性)
  1.  System.out.println(int[].class.isPrimitive());
    System.out.println(int[].class.isArray());
 
Construtor
        一个类首先要有一个无参的构造方法,其次有可能有有参的构造方法。同时我们知道方法的overload,其中最主要去区别就是参数的个数和类型的不同。那么Construtors也可能有多个构造方法,于是Java就提供了拿到具体的构造方法,或者拿到所有的构造方法。
  1. //得到全部的构造方法
    //constructor
    Constructor[] constuctor =Class.forName("java.lang.String").getConstructors();
    //得到参数为StringBuffer的构造方法
    Constructor constructor =Class.forName("java.lang.String").getConstructor(StringBuffer.class);
Field
       类所拥有的属性,private或者public的。我们要得到所有的Field,可能在private属性的情况下,就需要通过getDeclaredField和setAccessible来获取这个属性了。
public class ReflectPoint{
//private和public的属性
private int x;
public int y;
public ReflectPoint(int x,int y){
super();
this.y = y;
this.x = x;
}
publicString toString(){
System.out.println(str1 + str2 + str3);
return str1 + str2 + str3;
}
}
//Field
ReflectPoint pt1 =newReflectPoint(3,5);
Field fieldY = pt1.getClass().getField("y");
System.out.println(fieldY.get(pt1));
Field fieldX = pt1.getClass().getDeclaredField("x");
fieldX.setAccessible(true);
System.out.println(fieldX.get(pt1));
/**
结果:
5
3
*/
  1.         
 
由此可以知道其实private和public的属性是需要通过不同的方法才可以拿到的。 
同时这个我们可以通过反射拿到Field的值,然后再将值改变。调用下面的方法,会将传入的Object中的属性中的值,将含有a都替换成b

private static void changeStringValue(Object obj) throws Exception{
// TODO Auto-generated method stub
Field[] fields = obj.getClass().getFields();
for(Field field : fields){
if(field.getType()==String.class){
String oldValue =(String) field.get(obj);
String newValue = oldValue.replace("a","b");
field.set(obj, newValue);
}
}
}
 
Method
其实拿到一个类中方法,和构造方法其实是一样的,第一个参数就是方法名,第二个参数就是这个方法的传入参数的类型,如果还有更多的参数,也可以继续写进去的。
  1. //Method方法
    String str1 = "a";
    Method methodCharAt =String.class.getMethod("charAt",int.class);
    System.out.println(methodCharAt.invoke(str1,1));
    //str1是一个String的实例对象

      

同时方法的启动是用invoke的第一参数是含有这个方法的类的一个实体。第二个参数则是这个方法的要传入的各种参数的真实数据啦。当然,有时候静态方法static实际是不需要实例对象的,因为在java运行的时候就将这些静态方法的实例都存在JVM中了,所以invoke可以是null,invoke(null, new Object[]{new String[]{"111","222","333"}});
 
 
Array数组和Object的关系
        一个Object可以是什么类型呢?我们知道所有的类都是以Object为父类,但是数组呢?String不是Java的8大数据类型,String的父类是Object,但是Java中8大数据类型(boolean,byte,char,short,int,float, long,double)其实对应的Java类分别为(Boolean, Byte , Character ,Short , Integer , Float , Long ,Double)这些封装类的父类才是Object,他们本身不是,所以,我们在数组中保存这些数据类型的时候就是数组就是一个Object,但是String的数组可以用Object[] 来表示。

       讲了那么多,我们才可以用数组的反射做事情。

/**
* 数组的反射
* @param o1
*/
private static void printObject(Object o1){
// TODO Auto-generated method stub
Class clazz = o1.getClass();
if(clazz.isArray()){
int length =Array.getLength(o1);
for(int i =0; i < length; i++){
System.out.println(Array.get(o1, i));
}
}else{
System.out.println(o1);
}
} int[] a1 =newint[]{1,2,3};
int[] a2 =newint[4];
int[][] a3 =newint[3][4];
String[] a4 =newString[]{"a","b","c"};
printObject(a1);
printObject(a4);
注意这里,String的数组即可以用Object来表示,也是可以用Object[]来表示的,但是int[]的数组却只能用Object来表示,不能用Object[]来表示。
 
创建newInstance对象
这个就比较简单啦,通过得到一个Class对象,再将其newInstance化。
  1. String obj =(String)Class.forName("java.lang.String").newInstance();

结束:我们知道真正地改变是在不断地练习中成长的。(文不对题哈哈)

Java反射深入浅出(一)的更多相关文章

  1. Java反射深入浅出

    在JVM中对一个类实例的创建,有两种方式,一种是编译时,一种是运行时.两种方式在开发过程中都是十分重要的.在Java中无时无刻无处不在的Java对象,实例化的过程也就变得尤为引人瞩目.我们经常用new ...

  2. 深入浅出Java反射

    反射,它就像是一种魔法,引入运行时自省能力,赋予了 Java 语言令人意外的活力,通过运行时操作元数据或对象,Java 可以灵活地操作运行时才能确定的信息 这里笔者就深入浅出总结下Java反射,若有不 ...

  3. 第28章 java反射机制

    java反射机制 1.类加载机制 1.1.jvm和类 运行Java程序:java 带有main方法的类名 之后java会启动jvm,并加载字节码(字节码就是一个类在内存空间的状态) 当调用java命令 ...

  4. Java反射机制

    Java反射机制 一:什么事反射机制 简单地说,就是程序运行时能够通过反射的到类的所有信息,只需要获得类名,方法名,属性名. 二:为什么要用反射:     静态编译:在编译时确定类型,绑定对象,即通过 ...

  5. java反射(基础了解)

    package cn.itcast_01; /** *Person类 */ public class Person {    /** 姓名 */    private String name;     ...

  6. java基础知识(十一)java反射机制(上)

    java.lang.Class类详解 java Class类详解 一.class类 Class类是java语言定义的特定类的实现,在java中每个类都有一个相应的Class对象,以便java程序运行时 ...

  7. java基础知识(十一)java反射机制(下)

    1.什么是反射机制? java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对于任意一个对象都能够调用他的属性和方法,这种动态获取属性和方法的功能称为java的反射机制. ...

  8. java反射学习之二万能EXCEL导出

    一.EXCEL导出的实现过程 假设有一个对象的集合,现在需要将此集合内的所有对象导出到EXCEL中,对象有N个属性:那么我们实现的方式是这样的: 循环这个集合,在循环集合中某个对象的所有属性,将这个对 ...

  9. java反射学习之一反射机制概述

    一.反射机制背景概述 1.反射(reflection)是java被视为动态语言的一个关键性质 2.反射机制指的是程序在运行时能获取任何类的内部所有信息 二.反射机制实现功能概述 1.只要给定类的全名, ...

随机推荐

  1. AJAX 跨域请求 - JSONP获取JSON数据

    Asynchronous JavaScript and XML (Ajax ) 是驱动新一代 Web 站点(流行术语为 Web 2.0 站点)的关键技术.Ajax 允许在不干扰 Web 应用程序的显示 ...

  2. [Cocos2D-x For WP8]Box2D物理引擎

    物理引擎通过为刚性物体赋予真实的物理属性的方式来计算运动.旋转和碰撞反映.为每个游戏使用物理引擎并不是完全必要的—简单的“牛顿”物理(比如加速和减速)也可以在一定程度上通过编程或编写脚本来实现.然而, ...

  3. 百度mobile UI组件GMU demo学习1-结构和初始化

    移动web现在已经是zepto的天下,但是一直找不到合适UI库,找了一段时间,终于找到了百度的ui库gum和inter 的 appframework UI库 相比之下,百度的UI库更接地气,配合百度强 ...

  4. 如何伪装成为一名前端(JS方向)

    作为一个菜鸟级别的.NET开发者,在连服务器都没搞定的情况下,要研究前端,这是在扯淡,不过,迫于工作的需要,时常需要去前端打杂,所以经常伪装成为一名前端,有时候竟产生错觉,去应聘Y一份前端work吧. ...

  5. sublime text 3 的在文件夹中查找的快捷键没有反应 的bug冲突

    11:19 2015/11/18 sublime text 3 的在文件夹中查找的快捷键没有反应 的bug冲突 在文件夹查找的快捷键:ctrl shift f没有反应,后来发现是百度输入法与它有问题, ...

  6. 【BZOJ】2924: [Poi1998]Flat broken lines

    题意 平面上有\(n\)个点,如果两个点的线段与\(x\)轴的角在\([-45^{\circ}, 45^{\circ}]\),则两个点可以连线.求最少的折线(折线由线段首尾相连)使得覆盖所有点. 分析 ...

  7. (转)KeyDown、KeyUp、KeyPress区别

    Windows窗体通过引发键盘事件来处理键盘输入以响应Windows消息,大多数Windows窗体应用程序都通过处理键盘事件来以独占方式处理键盘输入. 1.按键的类型 Windows窗体将键盘输入标 ...

  8. Node.js 手册查询-2-MongoDB数据库方法

    MongoDb 标签(空格分隔): 数据库 MongoDb 安装 当前版本 2.X 解压至任意目录,最好不要是c盘. 在根目录下建立一个文件夹用来存储工程 我的例子: 安装至: d:\mongodb ...

  9. IDictionary<TKey, TValue> vs. IDictionary

    Enumerating directly over an IDictionary<TKey,TValue>returns a sequence of  KeyValuePair struc ...

  10. 兼容性好的CSS字体投影

    <p>兼容性良好的css文字描边</p> <style><!-- h1, p { color: #fff; width: 100%; text-align: ...