Class对象可以获取类里的方法,由Method对象表示,调用Method的invoke可以执行对应的方法;可以获取构造器,由Constructor对象表示,调用Constructor对象的newInstance方法可以执行类对应的构造方法;可以获取成员变量,由Field对象表示,通过Field对象可以直接修改类的成员变量的访问权限和值。

创建对象

通过反射有两种方式创建对象

使用Class对象的newInstance(),这是最常用的方式,根据配置文件信息创建对象。

使用Class对象获取指定的Constructor对象,再由Constructor对象的newInstance创建类的对象。

调用方法

首先要获取类在JVM中对应的Class对象,通过类的Class对象的getMethod方法来获取类的方法,返回Method类型的对象,最后通过Method对象的invoke来执行类的方法。

下面的例子将演示从配置文件中读取数据来动态地创建对象,还通过配置文件信息来执行对象的setter来初始化对象。

 package jvmTest;

 import java.io.FileInputStream;
import java.io.IOException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties; public class ExtendedObjectPoolFactory {
private Map<String, Object> objectPool = new HashMap<>();
private Properties config = new Properties();
public void init(String fileName) {
try {
FileInputStream fis = new FileInputStream(fileName);
config.load(fis);
} catch (IOException ex) {
System.out.println("读取 " + fileName + " 异常");
ex.printStackTrace();
}
}
private Object createObject(String clazzName) throws Exception {
Class<?> clazz = Class.forName(clazzName);
//使用clazz默认构造器创建对象
return clazz.newInstance();
}
//根据配置文件创建对象
public void initPool() throws Exception {
for (String name : config.stringPropertyNames()) {
if (!name.contains("%")) {
objectPool.put(name, createObject(config.getProperty(name)));
}
}
} //根据配置文件来执行对象对应的set方法
public void initProperty() throws Exception {
for (String name : config.stringPropertyNames()) {
if (name.contains("%")) {
String[] objAndPro = name.split("%");
//获取对象
Object target = getObject(objAndPro[0]);
//构造setter方法名
String mtdName = "set"+ objAndPro[1].substring(0,1).toUpperCase() + objAndPro[1].substring(1);
Class<?> targetClass = target.getClass();
Method mtd = targetClass.getMethod(mtdName, String.class);
mtd.invoke(target, config.getProperty(name));
}
}
} public Object getObject(String name) {
return objectPool.get(name);
} public static void main(String[] args) throws Exception {
ExtendedObjectPoolFactory epf = new ExtendedObjectPoolFactory();
epf.init("extObj.txt");
epf.initPool();
epf.initProperty();
System.out.println(epf.getObject("a"));
}
}

上面程序第23行获取Class对象后在第25行使用默认的构造器动态创建出了对象,

如果要使用指定构造器,需要先获取Constructor对象,再使用Cronstructor对象的newInstance创建对象,像下面这样,

 Constructor ctor = clazz.getConstructor(String.class);
return ctor.newInstance("here is parameter for specific constructor");

第45行获取Class对象后,在46行通过Class对象获取指定要执行的方法(带一个参数),用Method对象表示,最后在47行调用Method对象的invoke可以执行类的指定方法

上面的例子使用下面的测试文件进行测试,

extObj.txt

 a=javax.swing.JFrame
b=javax.swing.JLabel
a%title=Test title

输入结果,

javax.swing.JFrame[frame0,0,0,0x0,invalid,hidden,layout=java.awt.BorderLayout,title=Test title,resizable,normal,defaultCloseOperation=HIDE_ON_CLOSE,rootPane=javax.swing.JRootPane[,0,0,0x0,invalid,layout=javax.swing.JRootPane$RootLayout,alignmentX=0.0,alignmentY=0.0,border=,flags=16777673,maximumSize=,minimumSize=,preferredSize=],rootPaneCheckingEnabled=true]

访问成员变量

通过Class对象的getField方法可以访问指定的成员变量,调用继承自AccessableObject的setAccessable方法可以获得private成员变量的访问权限。

以下方法可以访问成员变量,

get(Object obj), 获取成员变量值。  如果是基本类型,直接使用类似 getInt(Object obj),  getChar(Object obj)的方式

set(Object obj), 设置成员变量值。 如果是基本类型,直接使用类似 setInt(Object, int val), setChar(Object obj, char c)的方式

下面演示访问成员变量,

 package jvmTest;

 import java.lang.reflect.Field;

 class Person {
private String name;
private int age;
public String toString() {
return "Person[name: "+name+", age: "+age+"]";
}
}
public class FieldTest {
public static void main(String[] args) throws Exception {
Person p = new Person();
Class<Person> clazz = Person.class;
//getDeclaredFiled可以获取所有访问权限的成员变量
Field nameField = clazz.getDeclaredField("name");
//true表示取消权限限制
nameField.setAccessible(true);
nameField.set(p, "Tom");
Field ageField = clazz.getDeclaredField("age");
ageField.setAccessible(true);
ageField.setInt(p, 18);
System.out.println(p);
}
}

上面程序第20行使用的是set(), 而23行使用的是setInt(), 输出结果,

Person[name: Tom, age: 18]

操作数组

java.lang.reflect还包含一个Array类,可以动态地创建数组,设置元素的值,比较强大,主要方法有,

newInstance(..),创建数组,可以指定数组元素类型, 数组维度, 数组长度

get(...), 获取索引为index的元素,对于基本数据类型的数组,方法为getInt(...),  getChar(...) ...

set(...), 设置索引为index的元素,对于基本数据类型的数组,方法为setInt(...),  setChar(...) ...

下面演示用法,

 package jvmTest;

 import java.lang.reflect.Array;

 public class ArrayTest2 {
public static void main(String[] args) {
// 创建三维数组
Object arr = Array.newInstance(String.class, 3, 4, 10);
// 获取index为2的元素,该元素是一个二维数组
Object arrObj = Array.get(arr, 2);
// 给二维数组index为2的元素赋值
// 二维数组的元素是一维数组,所以赋值也要用数组来赋值
Array.set(arrObj, 2, new String[] { "天王盖地虎", "宝塔镇河妖" });
//获取二维数组指定index的元素(结果是一维数组)
Object anArr = Array.get(arrObj, 3);
Array.set(anArr, 8, "野鸡闷头钻,哪能上天王山");
//将arr强制转换为三维数组
String[][][] cast = (String[][][])arr; System.out.println(cast[2][2][0]);
System.out.println(cast[2][2][1]);
System.out.println(cast[2][3][8]);
}
}

输出如下,

 天王盖地虎
宝塔镇河妖
野鸡闷头钻,哪能上天王山

JAVA基础知识之JVM-——使用反射生成并操作对象的更多相关文章

  1. JAVA基础知识总结18(反射)

    反射技术: 其实就是动态加载一个指定的类,并获取该类中的所有的内容.而且将字节码文件封装成对象,并将字节码文件中的内容都封装成对象,这样便于操作这些成员.简单说:反射技术可以对一个类进行解剖. 反射的 ...

  2. java 基础知识学习 JVM虚拟机参数配置

    1) 设置-Xms.-Xmx相等: 2) 设置NewSize.MaxNewSize相等: 3) 设置Heap size, PermGen space: Tomcat 的配置示例:修改%TOMCAT_H ...

  3. Java基础知识笔记第四章:类和对象

      编程语言的几个发展阶段 面向机器语言 面向过程语言 面向对象语言:封装.继承.多态 类 类声明 class Person{ ....... } class 植物{ ....... } 类体 类使用 ...

  4. 学习Spring必学的Java基础知识(1)----反射(转)

    引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓"登高必自卑,涉远必自迩".以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系 ...

  5. 学习Spring必学的Java基础知识(1)----反射

    引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓"登高必自卑,涉远必自迩".以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系 ...

  6. JAVA基础知识|java虚拟机(JVM)

    一.JVM简介 java语言是跨平台的,兼容各种操作系统.实现跨平台的基石就是虚拟机(JVM),虚拟机不是跨平台的,所以不同的操作系统需要安装不同的jdk版本(jre=jvm+类库:jdk=jre+开 ...

  7. Java基础知识总结(超级经典)

    Java基础知识总结(超级经典) 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部分用到哪些语句,方法,和对象. 4,代码实现.用具体的java ...

  8. 毕向东—Java基础知识总结(超级经典)

    Java基础知识总结(超级经典) 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部分用到哪些语句,方法,和对象. 4,代码实现.用具体的java ...

  9. 沉淀,再出发:Java基础知识汇总

    沉淀,再出发:Java基础知识汇总 一.前言 不管走得多远,基础知识是最重要的,这些知识就是建造一座座高楼大厦的基石和钢筋水泥.对于Java这门包含了编程方方面面的语言,有着太多的基础知识了,从最初的 ...

随机推荐

  1. C++向量(Vector)

    向量Vector: 向量 vector 是一种对象实体, 能够容纳许多其他类型相同的元素, 因此又被称为容器. 与string相同, vector 同属于STL(Standard Template L ...

  2. iOS 检查版本号的代码

    - (void)checkNewVersion{ if ([@"appStore" isEqualToString:CHANNEL]) { AFHTTPRequestOperati ...

  3. [原创] 分享一些linux教程

    书<鸟哥的linux私房菜第三版>,链接:http://pan.baidu.com/s/1i3femnr 配套视频,链接:http://pan.baidu.com/s/1v72xw --- ...

  4. SQL封装、多态与重载

    面向对象1.类:众多对象抽象出来的2.对象:类实例化出来的 3.类的定义关键字 class 4.类里面包含成员变量成员属性 成员方法 5.面向对象三大特性(1)封装目的:保护类,让类更加安全.做法:让 ...

  5. c++中的传参问题

    从概念上讲.指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变. 而引用是一个别名,它在逻辑上不是独立的,它的存在具有 ...

  6. PHP内核探索:哈希碰撞攻击是什么?

    最近哈希表碰撞攻击(Hashtable collisions as DOS attack)的话题不断被提起,各种语言纷纷中招.本文结合PHP内核源码,聊一聊这种攻击的原理及实现. 哈希表碰撞攻击的基本 ...

  7. PHP漏洞全解(详细介绍)

    转载  http://www.jb51.net/article/31898.htm 针对PHP的网站主要存在下面几种攻击方式: 1.命令注入(Command Injection) 2.eval注入(E ...

  8. redis数据库使用测试

    注:java spring data redis内置了几种redis序列化机制.JdkSerializationRedisSerializer最高效.有关序列化更详细的介绍-http://www.my ...

  9. mongo聚合和mapreduce例子

    聚合语句-比较集合内两字段大小 db.test.aggregate([ {$match:{"offlineTime":{$gt:ISODate("2016-09-13T0 ...

  10. Java IO总结之缓冲读入文件

    package com.io; import java.io.BufferedReader; import java.io.FileReader; import java.io.IOException ...