J2SE-反射
目录
- 1 什么是反射?主要提供了哪些功能?
- 2 如何通过反射获取类的信息?
- 3 数组的特殊处理。
- 4 反射的应用场景。
- 5 其他。
正文
1 什么是反射?主要提供了哪些功能?
1) Java的反射机制是指在程序运行期间,对于任意一个类,都可以知道这个类的属性和方法,并进行调用。
2) 提供以下功能:
① 在运行时可以判断任意一个对象所属的类。
② 在运行时可以构造任意一个类的对象。
③ 在运行时可以获取任意一个类的所有成员变量和方法。
④ 在运行时调用任意一个对象的方法。
⑤ 生成动态代理。
3) 优缺点
① 优点:增加灵活性。
② 缺点:使用不当会导致效率低、会破坏封装,且可以访问类的私有方法,不安全。
2 如何通过反射获取类的信息?
1) 首先获取类的Class对象:
① 每一个类、接口、数组、基本类型都有唯一的一个对应的Class类对象。通过Class类对象可以获取类的全部信息(包括成员变量、方法、父类、接口等)。
② 获得Class对象的三种方式。【参考TestLoadClass.java】
③ Class的常用方法:【参考TestClass.java】
package com.devway.j2se.reflect;
public class TestLoadClass {
public static void main(String[] args) {
try {
// 获取Class对象的三种方法
// 方法一,通过对象实例获取,
A a = new A();
Class<?> class1 = a.getClass();
System.out.println("1:getClass " + class1);
// 方法二,根据类名获取
Class<?> class2 = B.class;
System.out.println("2:.class " + class2);
// 方法三,根据类的全名获取
Class<?> class3 = Class.forName("com.devway.j2se.reflect.C");
System.out.println("3:Class.forName " + class3);
} catch (ClassNotFoundException e) {
System.err.println(e);
}
}
}
class A {
static {
System.out.println("init A static");
}
}
class B {
static {
System.out.println("init B static");
}
}
class C {
static {
System.out.println("init C static");
}
}
TestLoadClass.java
package com.devway.j2se.reflect;
public class TestClass {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child");
System.out.println("1:" + class1);// true
// Class类的常用方法
// 判断是否是接口
boolean isInterface = class1.isInterface();// false
// 判断是否是数组
boolean isArray = class1.isArray();// false
// 判断是否是8种基本类型
boolean isPrimitive = class1.isPrimitive();// false;
System.out.println("2:" + isInterface + "," + isArray + "," + isPrimitive);
// 获取父类的Class类对象
Class<?> superClass = class1.getSuperclass();
// 获取所有接口
Class<?>[] interfaces = class1.getInterfaces();
// 获取类型修饰符
int modifiers = class1.getModifiers();
System.out.println("3:" + superClass + "," + interfaces + "," + modifiers);
// 获取数组的组件类型
int[] ints = { 1, 2, 3 };
Class<?> class4 = ints.getClass();
Class<?> class5 = class4.getComponentType();// int
System.out.println("4:" + class4);
System.out.println("4:" + class5);
// 返回声明这个的Class对象,如果不是嵌套类型,返回null
Class<?> class6 = class1.getDeclaringClass();// null
System.out.println("5:" + class6);
// 获取包
Package package1 = class1.getPackage();
System.out.println("6:" + package1);
} catch (ClassNotFoundException e) {
System.err.println(e);
}
}
}
package com.devway.j2se.reflect;
public interface Interface1 {
String iv1 = "a";
String iv2 = "b";
void f1();
}
package com.devway.j2se.reflect;
public class Parent {
public String pv1;
protected String pv2;
public Parent() {
}
public String pf1(String s) {
return s;
}
protected void pf2() {
}
void pf3() {
}
private void pf4() {
}
}
package com.devway.j2se.reflect;
public class Child extends Parent implements Interface1 {
public String cv1;
protected String cv2;
private String cv3;
public static String scv1;
public Child() {
}
public Child(String s, int i) {
}
protected Child(int i) {
}
private Child(String s) {
}
public String cf1(String s, int i) throws NullPointerException {
return s + i;
}
protected void cf2() {
System.out.println("protected method cf4()");
}
void cf3() {
System.out.println("default method cf4()");
}
private void cf4() {
System.out.println("private method cf4()");
}
@Override
public void f1() {
System.out.println("public method f1()");
}
}
TestClass.java
2) 对类的构造函数、方法、属性的访问和调用:
① Constructor类:用于构造函数。【参考TestConstrcutor.java】
② Method类:用于类的方法。【参考TestMethod.java】
③ Field类:用于类的属性。【参考TestField.java】
④ Modifer类:用于判断修饰符类型。【参考TestModifier.java】
⑤ Field、Method、Constructor类都继承了AccessibleObject类(该类用于获取和修该访问权限)【参考TestAccessible.java】,同时也实现了Member接口。
package com.devway.j2se.reflect;
import java.lang.reflect.Constructor;
public class TestConstrcutor {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child");
// 获取所有的public构造函数,包括父类和接口的
Constructor<?>[] publicConstructors = class1.getConstructors();
for (Constructor<?> constructor : publicConstructors) {
System.out.println("1:" + constructor);
}
// 获取当前类的所有构造函数,不包括父类和接口的
Constructor<?>[] declaredConstructors = class1.getDeclaredConstructors();
for (Constructor<?> constructor : declaredConstructors) {
System.out.println("2:" + constructor);
}
// 根据名称和参数获取方法
Constructor<?> constructor1 = class1.getConstructor(String.class, int.class);
System.out.println("3:" + constructor1);
// 获取方法参数对应的Class对象
Class<?>[] paramTypesClass = constructor1.getParameterTypes();
for (Class<?> paramType : paramTypesClass) {
System.out.println("4:" + paramType);
}
// 获取方法抛出的异常对应的Class对象
Class<?>[] exceptionTypesClass = constructor1.getExceptionTypes();
for (Class<?> exceptionType : exceptionTypesClass) {
System.out.println("6:" + exceptionType);
}
// 创建实例并调用方法
Interface1 inf = (Interface1) class1.newInstance();
inf.f1();
} catch (ClassNotFoundException | NoSuchMethodException | SecurityException | InstantiationException | IllegalAccessException e) {
System.err.println(e);
}
}
}
TestConstrcutor.java
package com.devway.j2se.reflect; import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; public class TestMethod {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child"); // 获取所有的public方法,包括父类和接口的
Method[] publicMethods = class1.getMethods();
for (Method method : publicMethods) {
System.out.println("1:" + method);
} // 获取当前类的所有方法,不包括父类和接口的
Method[] declaredMethods = class1.getDeclaredMethods();
for (Method method : declaredMethods) {
System.out.println("2:" + method);
} // 根据名称和参数获取方法
Method method1 = class1.getMethod("cf1", String.class, int.class);
System.out.println("3:" + method1); // 获取方法返回值类型对应的Class对象
Class<?> returnTypeClass = method1.getReturnType();
System.out.println("4:" + returnTypeClass); // 获取方法参数对应的Class对象
Class<?>[] paramTypesClass = method1.getParameterTypes();
for (Class<?> paramType : paramTypesClass) {
System.out.println("5:" + paramType);
} // 获取方法抛出的异常对应的Class对象
Class<?>[] exceptionTypesClass = method1.getExceptionTypes();
for (Class<?> exceptionType : exceptionTypesClass) {
System.out.println("6:" + exceptionType);
} // 方法调用
Child child = (Child) class1.newInstance();
String value = (String) method1.invoke(child, "abc", 123);
System.out.println("7:" + value);
} catch (ClassNotFoundException | NoSuchMethodException | SecurityException | InstantiationException | IllegalAccessException
| IllegalArgumentException | InvocationTargetException e) {
System.err.println(e);
}
}
}
TestMethod.java
package com.devway.j2se.reflect;
import java.lang.reflect.Field;
public class TestField {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child");
// 获取所有的public域,包括父类和接口的
Field[] publicFields = class1.getFields();
for (Field field : publicFields) {
System.out.println("1:" + field);
}
// 获取当前类的所有域,不包括父类和接口的
Field[] declaredFields = class1.getDeclaredFields();
for (Field field : declaredFields) {
System.out.println("2:" + field);
}
// 根据名称获取域
Field field1 = class1.getField("cv1");
System.out.println("3:" + field1);
// 获取域类型的Class对象
Class<?> field1Class = field1.getType();
System.out.println("4:" + field1Class);
// 读取域的值
Child child = (Child) class1.newInstance();
child.cv1 = "abc";
String field1Value = (String) field1.get(child);// abc
System.out.println("5:" + field1Value);
// 修改域的值
field1.set(child, "def");
System.out.println("6:" + child.cv1);
} catch (ClassNotFoundException | NoSuchFieldException | SecurityException | InstantiationException | IllegalAccessException e) {
System.err.println(e);
}
}
}
TestField.java
package com.devway.j2se.reflect;
import java.lang.reflect.Modifier;
public class TestModifier {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child");
boolean isPrivate = Modifier.isPrivate(class1.getModifiers());// false;
System.out.println("1:" + isPrivate);
boolean isAbastract = Modifier.isAbstract(class1.getModifiers());// false
System.out.println("2:" + isAbastract);
} catch (ClassNotFoundException e) {
System.err.println(e);
}
}
}
TestModifier.java
package com.devway.j2se.reflect; import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; public class TestAccessible {
public static void main(String[] args) {
try {
Class<?> class1 = Class.forName("com.devway.j2se.reflect.Child");
Method method1 = class1.getDeclaredMethod("cf4"); Interface1 inf = (Interface1) class1.newInstance();
// ! method1.invoke(inf,null);//cf4是private方法,不能访问
System.out.println("1:" + method1.isAccessible());//false //取消访问限制
method1.setAccessible(true);
method1.invoke(inf,null);
System.out.println("2:" + method1.isAccessible());//true
} catch (ClassNotFoundException | NoSuchMethodException | SecurityException | InstantiationException | IllegalAccessException
| IllegalArgumentException | InvocationTargetException e) {
System.err.println(e);
} }
}
TestAccessible.java
3 数组的特殊处理。
1) 数组是没有方法和属性的对象,所以不能通过反射常用的的方法来获取或设置成员。
2) 可以通过反射创建数组,使用两种new Instance的方法来创建。使用Array.set和Array.get来设置或获取数组元素的值。【参考TestArray.java】
package com.devway.j2se.reflect; import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.Method; public class TestArray {
public static void main(String[] args) {
Integer[] arrays1 = { 1, 2, 3, 4 };
Class<?> class1 = arrays1.getClass(); // 数组是没有方法和域的对象,所以都返回null
Method[] delcaredMethods = class1.getDeclaredMethods();
for (Method method : delcaredMethods) {
System.out.println("1:" + method);// null
} Field[] delcaredFields = class1.getDeclaredFields();
for (Field field : delcaredFields) {
System.out.println("2:" + field);// null
} // 创建数组方式1
Integer[] arrays2 = (Integer[]) Array.newInstance(Integer.class, 5); // 创建数组方式2,返回多为数组
int[] intval = { 2, 2 };
Integer[][] arrays3 = (Integer[][]) Array.newInstance(Integer.class, intval); // 设置数组元素的值
for (int i = 0; i < arrays2.length; i++) {
Array.set(arrays2, i, i + 1);
} // 获取数组元素的值
for (int i = 0; i < arrays2.length; i++) {
System.out.println("3:" + Array.get(arrays2, i));
} }
}
TestArray.java
4 反射的应用场景。
1) 用于基础框架中,如Spring。
2) JDBC,数据库可能使用Oracle,也可能使用Mysql,不同的数据库有不同的驱动,在运行时,再通过Class.forName()运用反射动态加载。
3) 应用开发中,尽量避免使用反射。
5 其他。
1) class.getName方法返回结果
① 数组类型以[为前缀,后面跟着成员的类型编码,基础类型则简写如下。
(B:byte、C:char、D:double、F:float、I:int、J:long、S:short、Z:boolean)
② 嵌套类型使用$符号将类型名隔开:Outer$Inner
2) 注意这种写法是错误的:Class<Parent> c = Child.class; 因为Child.class不是Parent的子类,正确写法是:Class<? extends Parent> c= Child.class。
3) 向上转型和向下转型:
① 向上转型:子类转为父类,自动转换。
② 向下转型:父类转为子类,需要强制转换。

J2SE-反射的更多相关文章
- J2SE 8的反射
1.获得Class的四种方式 //(1) 利用对象调用getClass()方法获取该对象的Class实例 Class<? extends ReflectTest> class1 = new ...
- Java反射机制的学习
Java反射机制是Java语言被视为准动态语言的关键性质.Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调 ...
- 初识JAVA(【面向对象】:pub/fri/pro/pri、封装/继承/多态、接口/抽象类、静态方法和抽象方法;泛型、垃圾回收机制、反射和RTTI)
JAVA特点: 语法简单,学习容易 功能强大,适合各种应用开发:J2SE/J2ME/J2EE 面向对象,易扩展,易维护 容错机制好,在内存不够时仍能不崩溃.不死机 强大的网络应用功能 跨平台:JVM, ...
- Java 使用反射拷贝对象一般字段值
在<Java解惑>上面看到第八十三例--诵读困难者,要求使用非反射实现单例对象的拷贝.查阅了部分资料,先实现通过反射拷贝对象. 1. 编写需要被拷贝的对象Person package co ...
- Java反射机制(转载)
原文链接:http://www.blogjava.net/zh-weir/archive/2011/03/26/347063.html Java反射机制是Java语言被视为准动态语言的关键性质.Jav ...
- 认识J2SE
1. J2SE的定义 J2SE:全称为Java 2 Standard Edition.Java 2平台包括:标准版(J2SE).企业版(J2EE)和微缩版(J2ME)三个版本. J2SE主要包括UI. ...
- (转载)Java反射机制
Java反射机制是Java语言被视为准动态语言的关键性质.Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调 ...
- J2SE 8的注解
1. 注解概念 (1) 注解格式 modifiers @interface AnnotationName { type elementName(); type elementName() defaul ...
- J2SE语言--百度百科
Java2平台包括:标准版(J2SE).企业版(J2EE)和微缩版 (J2ME)三个版本.J2SE,J2ME和J2EE,这也就是SunONE(Open NetEnvironment)体系.J2SE就是 ...
- J2SE核心开发实战(一)——认识J2SE
认识J2SE 一.课程简单介绍 在本章学习開始前,你应该具备一些Java的基础知识. 我们将在本章来认识J2SE,并复习一下前面学过的面向对象的相关知识. 注:全部的蓝色文字都是带超链接的,这些链接是 ...
随机推荐
- LINUX 笔记-grep命令
grep [-acinv] [--color=auto] '查找字符串' filename 它的常用参数如下: -a :将binary文件以text文件的方式查找数据 -c :计算找到'查找字符串'的 ...
- python + selenium 自动化测试框架
分享一个网站自动化测试框架 结构如下: test_project|--logs|---pages |---register_page.py| |---base_page.py|---test ...
- 基于HTML5及WebGl下生成的json格式的工控SCADA风机叶轮旋转
突然有个想法,如果能把一些用到不同的知识点放到同一个界面上,并且放到一个盒子里,这样我如果要看什么东西就可以很直接显示出来,而且这个盒子一定要能打开.我用HT实现了我的想法,代码一百多行,这么少的代码 ...
- 【问题排查】fastjson线上排坑记
前言 版本上线时发现fastjson的toString方法的返回的字符串与与之前版本的toString方法返回的字符串不相同,这导致依赖toString进行md5计算所得到的结果不相同,更进一步导致其 ...
- 最近折腾老机器:还是 Xp 最好!
闲来无事,折腾机器. -------------------------------------------------------------- 硬件: CPU:amd x2450 Memory:6 ...
- cocos-Lua中的class与require机制
cocos-Lua中的class与require机制 local layer = require("PaiGow.src.GamePlayerListLayer")local Ga ...
- js实现前端下载文件
在前端下载文本格式的文件时,可采用下面的方式: (1)创建基于文件内容的Blob对象: (2)通过URL上的createObjectURL方法,将blob对象转换成一个能被浏览器解析的文件地址. (3 ...
- 机器学习数学|微积分梯度jensen不等式
机器学习中的数学 觉得有用的话,欢迎一起讨论相互学习~Follow Me 原创文章,如需转载请保留出处 本博客为七月在线邹博老师机器学习数学课程学习笔记 索引 微积分,梯度和Jensen不等式 Tay ...
- Ubuntu系统下静态DNS配置详解
1.DNS服务的简介: DNS(Domain Name Server,域名服务器)是进行域名(domain name)和与之相对应的IP地址 (IP address)转换的服务器.DNS中保存了一张域 ...
- 字符函数 php
strrchr( '123456789.xls' , '.' ); //程序从后面开始查找 '.' 的位置,并返回从 '.' 开始到字符串结尾的所有字符