在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";
2 Class cls1 = str1.getClass();
3 Class cls2 =String.class;
4 Class cls3 =Class.forName("java.lang.String");
5 System.out.println(str1.getClass());
6 System.out.println(cls1 == cls2);
7 System.out.println(cls3 == cls2);
8 /**
9 输出结果
10 class java.lang.String
11 true
12 true
13 */
 
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());
2 System.out.println(int[].class.isArray());
 
 
Construtor
        一个类首先要有一个无参的构造方法,其次有可能有有参的构造方法。同时我们知道方法的overload,其中最主要去区别就是参数的个数和类型的不同。那么Construtors也可能有多个构造方法,于是Java就提供了拿到具体的构造方法,或者拿到所有的构造方法。
//得到全部的构造方法
//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
*/
 
 
由此可以知道其实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
其实拿到一个类中方法,和构造方法其实是一样的,第一个参数就是方法名,第二个参数就是这个方法的传入参数的类型,如果还有更多的参数,也可以继续写进去的。
//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化。
String obj =(String)Class.forName("java.lang.String").newInstance();
 
 
 
 
 
 
 
 
 
来源:https://www.cnblogs.com/kgrdomore/p/7b6ed10cf90a0e30776696e3709b0604.html

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. .net时间格式与彻夜未眠的我

    夜已经很深了,外面的狂风还在呜呜的叫着,我的脚已经冰凉冰凉...从11点半到现在我一直在测试为什么正确的Json格式字符串传到服务器后还在报400错误... 尼玛啊,以前测试是没有问题的啊 事情是这样 ...

  2. FW: Dockerfile RUN, CMD & ENTRYPOINT

    Dockerfile RUN, CMD & ENTRYPOINT     在使用Dockerfile创建image时, 有几条指令比较容易混淆, RUN, CMD, ENTRYPOINT. R ...

  3. Java程序员面试题集(1-50

    下面的内容是对网上原有的Java面试题集及答案进行了全面修订之后给出的负责任的题目和答案,原来的题目中有很多重复题目和无价值的题目,还有不少的参考答案也是错误的,修改后的Java面试题集参照了JDK最 ...

  4. vue事件修饰器

    事件修饰器 Vue.js 为 v-on 提供了 事件修饰符.通过由点(.)表示的指令后缀来调用修饰符.· .stop .prevent .capture .self <div id=" ...

  5. node.js基本工作原理及流程

    概述 Node.js是什么 Node 是一个服务器端 JavaScript 解释器,用于方便地搭建响应速度快.易于扩展的网络应用.Node.js 使用事件驱动, 非阻塞I/O 模型而得以轻量和高效,非 ...

  6. R中的data.table 快速上手入门

    data.table包提供了一个非常简洁的通用格式:DT[i,j,by]. 可以理解为:对于数据集DT,选取子集行i,通过by分组计算j. 对比与dplyr等包,data.table的运行速度更快. ...

  7. Jquery 实现跨域处理

    JS部分代码: $.ajax({ url:url, dataType:'jsonp', data:{title:title}, jsonp:'callback', success:function(l ...

  8. JS连等赋值的坑

    cnblogs标题: JS连等赋值的坑 关于JS连等赋值有个经典的笔试题: var a = {n: 1}; var b = a; a.x = a = {n: 2}; console.log(a.x); ...

  9. Library Cache优化与SQL游标

    Library Cache主要用于存放SQL游标,而SQL游标最大化共享是Library Cache优化的重要途径,可以使SQL运行开销最低.性能最优. 1 SQL语句与父游标及子游标 在PL/SQL ...

  10. Centos学习笔记2-网络部分

    一:修改IP地址:vi /etc/sysconfig/network-scripts/ifcfg-eth0 IPADDR=192.168.80.100 NETMASK=255.255.255.0 GA ...