Java---类反射(1)---类反射入门和基础
什么是类反射
☆什么是反射
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制。
反射(Reflection)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查, 也称自审,并能直接操作程序的内部属性。例如,使用它能获得Java类中各成员的名称并显示出来。
Java的这一能力在实际应用中应用得很多,在其它的程序语言中根本就不存在这一特性。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。
JavaBean是类反射的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过类反射动态的载入并取得Java组件(类)的属性。后面学习的各种框架,基本上都会有反射的使用。
☆反射引例(HelloWorld、USB)
最简单的类反射:(相当于HelloWorld)
package cn.hncu.hello;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* 类反射的Helloworld版---入门
* 只演示了类方法的类反射---Method[]
* @author 陈浩翔
* @version 1.0 2016-5-1
*/
public class ReflectionHelloWold {
public static void main(String[] args) {
try {
Class c = Class.forName("cn.hncu.introduce.Person");
System.out.println(c);
Method mc[] = c.getDeclaredMethods();//获得
for(Method m:mc){
//获得整个方法
//包括修饰,返回类型,方法名字,方法参数
System.out.println(m);
//解剖Method对象
System.out.println(Modifier.toString(m.getModifiers()));//修饰符
System.out.println(m.getReturnType());//返回类型
System.out.println(m.getName());//方法名字
Class pt[] = m.getParameterTypes();
for(Class p :pt){
System.out.println("parameter = "+p);//参数
}
System.out.println();
System.out.println(m.getDeclaringClass());//这个方法所在的类
System.out.println();
System.out.println("------------------------------");
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
Person类:
package cn.hncu.introduce;
public class Person {
private String name;
int age;
public final String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
private static final int sum(int age){
return 0;
}
}
输出结果:
class cn.hncu.introduce.Person
private static final int cn.hncu.introduce.Person.sum(int)
private static final
int
sum
parameter = int
class cn.hncu.introduce.Person
------------------------------
public int cn.hncu.introduce.Person.getAge()
public
int
getAge
class cn.hncu.introduce.Person
------------------------------
public void cn.hncu.introduce.Person.setAge(int)
public
void
setAge
parameter = int
class cn.hncu.introduce.Person
------------------------------
public final java.lang.String cn.hncu.introduce.Person.getName()
public final
class java.lang.String
getName
class cn.hncu.introduce.Person
------------------------------
public void cn.hncu.introduce.Person.setName(java.lang.String)
public
void
setName
parameter = class java.lang.String
class cn.hncu.introduce.Person
------------------------------
USB实例:反射最大的好处是解耦
USB接口:
package cn.hncu.usb.api;
public interface USB {
public abstract void work();
}
实现类:
package cn.hncu.usb.impl;
import cn.hncu.usb.api.USB;
public class UsbMouse implements USB{
@Override
public void work() {
System.out.println("UsbMouse....");
}
}
package cn.hncu.usb.impl;
import cn.hncu.usb.api.USB;
public class UsbOne implements USB{
@Override
public void work() {
System.out.println("UsbOne...");
}
}
package cn.hncu.usb.impl;
import cn.hncu.usb.api.USB;
public class UsbTwo implements USB{
@Override
public void work() {
System.out.println("UsbTwo....");
}
}
工厂方法:!!!读取配置文件的在这里
package cn.hncu.usb.factory;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Properties;
import cn.hncu.usb.api.USB;
public class USBFactory {
public static USB getUSB(){
USB usb = null;
Properties p = new Properties();//配置文件
FileInputStream in;
try {
in = new FileInputStream("usb.config");//读取配置文件
p.load(in);
String className = p.getProperty("name");
//通过name获得name后面=号后面的字符串,这样就可以通过修改配置文件来new不同的类
Class c = Class.forName(className);
usb = (USB) c.newInstance();//new 一个对象
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
return usb;
}
}
main方法:
package cn.hncu.usb;
import cn.hncu.usb.api.USB;
import cn.hncu.usb.factory.USBFactory;
public class Client {
public static void main(String[] args) {
USB usb = USBFactory.getUSB();
usb.work();
}
}
配置文件:usb.config
你配置哪个,就是new哪个实现类的!!
‘#’号是配置文件的注释
#name = cn.hncu.usb.impl.UsbOne
#name = cn.hncu.usb.impl.UsbTwo
name = cn.hncu.usb.impl.UsbMouse
反射使用的三个步骤
用于反射的类,如Method,可以在java.lang.reflect包中找到。使用这些类的时候必须要遵循三个步骤:
第一步:获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。
第二步:调用诸如getDeclaredMethods的方法,取得该类中定义的所有方法的列表。
第三步:使用反射的API来操作这些信息。
如下面这段代码:
Class c = Class.forName(“java.lang.String”);
Method ms[] = c.getDeclaredMethods();
System.out.println(ms[0].toString());
它将以文本方式打印出String中定义的第一个方法的原型。
☆反射示例(模拟instanceof的功能)
获取Class对象的三种方式
★ 方式一
通过对象的getClass方法进行获取。这种方式需要具体的类和该类的对象,以及调用getClass方法。
★ 方式二
任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。
★ 方式三
通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。
下面看怎么获取Class对象的代码:
package cn.hncu.reflect;
/**
* //获取Class对象的三种方式
* @author 陈浩翔
*
* @version 1.0 2016-5-1
*/
public class ReflectGetClass {
public static void main(String[] args) {
//getClassObj1();
//getClassObj2();
//getClassObj3();
test();
}
private static void test() {
Person p = new Person();
Object obj = (Object)p;
Class c = obj.getClass();//运行时多态,new谁调谁
System.out.println(c);//输出为:class cn.hncu.reflect.Person
System.out.println(int.class);//int
System.out.println(Integer.class);//class java.lang.Integer
System.out.println(Integer.TYPE);//int
System.out.println((int.class)==(Integer.class));//false
System.out.println((int.class)==(Integer.TYPE));//true
}
private static void getClassObj3() {
try {
String className = "cn.hncu.reflect.Person";
Class c = Class.forName(className);//3333用字符串(类名)---依赖于String类
System.out.println(c);//class cn.hncu.reflect.Person
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
private static void getClassObj2() {
Class c = Person.class;//2222通过类名----依赖于Person类
System.out.println(c);//class cn.hncu.reflect.Person
}
private static void getClassObj1() {
Person p1 = new Person();
Class c1 = p1.getClass();//1111通过对象
System.out.println(c1);//class cn.hncu.reflect.Person
Person p2 = new Person();
Class c2 = p2.getClass();
System.out.println(p1==p2);//false-不同的对象
System.out.println(c1==c2);//true-同一个类模板,所有对象是共享同一个类模板的
}
}
运行结果在输出后面都注释了。。。
Class.isInstance(Object obj)判定指定的 Object 是否与此 Class 所表示的对象赋值兼容。
代码演示:
package cn.hncu.reflect;
/**
* @author 陈浩翔
* @version 1.0 2016-5-1
*/
public class SimulateInstanceof {
/**
* isInstance(Object obj)判定指定的 Object 是否与此 Class 所表示的对象赋值兼容。
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
Class c = Class.forName("cn.hncu.reflect.A");
boolean boo = c.isInstance(new Integer(100));
System.out.println(boo);//false
boolean boo2 = c.isInstance( new A() );
System.out.println(boo2);//true
boolean boo3 = c.isInstance(new B());//B和A兼容
//也就是说子类可以当父类用
System.out.println(boo3);//true
}
}
class A{
private int age;
}
class B extends A{
private String name ;
}
类的解剖(获取类的定义信息)
Person类:
package cn.hncu.reflect;
import java.io.IOException;
public class Person extends Parent{
String name;
private int age;
public int num;
public static String schoolName;
public Person(){
}
private Person(String name, int age){
}
protected Person(int age) throws IOException,NumberFormatException{
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
protected void setAge(int age) {
this.age = age;
}
private int sum( int a){
return 0;
}
private int sum(){
return 0;
}
int aa(String a,int b) throws IOException,NumberFormatException{
return 0;
}
}
class Parent{
public static final int N=100;
}
★ 获取类的方法
找出一个类中定义了些什么方法,这是一个非常有价值也非常基础的反射用法。
代码演示:
package cn.hncu.reflect;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* 类的解剖(获取类的定义信息)
* @author 陈浩翔
*
* @version 1.0 2016-5-1
*/
public class ReflectDecompose {
private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";
public static void main(String[] args) {
try {
fetchMethods(CLASS_FILE_NAME);//方法演示
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
/**
* 方法演示
* @param classFileName
* @throws ClassNotFoundException
*/
private static void fetchMethods(String classFileName) throws ClassNotFoundException {
Class c = Class.forName(CLASS_FILE_NAME);
//1: 获取该类的能访问的所有public方法(包含当前类和父类的)
Method methods[] = c.getMethods();
//2: 获取Person类中声明的所有方法,包括私有和其它权限的
//Method methods[] = c.getDeclaredMethods();
for(int i=0;i<methods.length;i++){
Method m = methods[i];
//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符
System.out.println("修饰符:"+ Modifier.toString( m.getModifiers() ) );//转换成字符串修饰符了。
System.out.println("方法返回类型:"+ m.getReturnType());
System.out.println("方法名称:"+ m.getName());
Class paramTypes[] = m.getParameterTypes();//方法需要传入的参数类型-为空就是没有
System.out.print("方法需要传入的参数类型: ");
for(Class p:paramTypes){
System.out.print(p+" ");
}
System.out.println();
Class exceptionTypes[] = m.getExceptionTypes();//方法抛出的异常
System.out.print("方法抛出的异常: ");
for(Class e:exceptionTypes){
System.out.print(e+" ");
}
System.out.println();
System.out.println("此方法属于的类: "+m.getDeclaringClass());
System.out.println();
System.out.println("--------------------------------");
}
}
}
运行结果太长了,自己可以运行看看!
★ 获取类的构造器
找出一个类中定义的构造方法,构造器没有返回类型。
代码演示:
package cn.hncu.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* 类的解剖(获取类的定义信息)
* @author 陈浩翔
*
* @version 1.0 2016-5-1
*/
public class ReflectDecompose {
private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";
public static void main(String[] args) {
try {
//fetchMethods(CLASS_FILE_NAME);//方法演示
fetchConstructors(CLASS_FILE_NAME);//类的构造方法演示
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
/**
* 类的构造方法演示
* @param classFileName
* @throws ClassNotFoundException
*/
private static void fetchConstructors(String classFileName) throws ClassNotFoundException {
Class c = Class.forName(CLASS_FILE_NAME);
//构造方法不能继承父类的!!!!!!!
//1: 该类中所有public的构造方法。
Constructor cons[] = c.getConstructors();
//2: 获取该类中声明的所有构造方法,包括私有和其它权限的
//Constructor cons[] = c.getDeclaredConstructors();
for(int i=0;i<cons.length;i++){
Constructor cs = cons[i];
//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符
System.out.println("修饰符:"+ Modifier.toString( cs.getModifiers() ) );//转换成字符串修饰符了。
System.out.println("构造方法名称:"+ cs.getName());
Class paramTypes[] = cs.getParameterTypes();//构造方法需要传入的参数类型-为空就是没有
System.out.print("构造方法需要传入的参数类型: ");
for(Class p:paramTypes){
System.out.print(p+" ");
}
System.out.println();
Class exceptionTypes[] = cs.getExceptionTypes();//构造方法抛出的异常
System.out.print("构造方法抛出的异常: ");
for(Class e:exceptionTypes){
System.out.print(e+" ");
}
System.out.println();
System.out.println("此构造方法属于的类: "+cs.getDeclaringClass());//全部是当前类!因为构造方法不能被继承
System.out.println();
System.out.println("--------------------------------");
}
}
}
★ 获取类的属性字段
找出一个类中定义了哪些属性字段。
代码演示:
package cn.hncu.reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* 类的解剖(获取类的定义信息)
* @author 陈浩翔
*
* @version 1.0 2016-5-1
*/
public class ReflectDecompose {
private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";
public static void main(String[] args) {
try {
//fetchMethods(CLASS_FILE_NAME);//方法演示
//fetchConstructors(CLASS_FILE_NAME);//类的构造方法演示
fetchFields(CLASS_FILE_NAME);//类的成员变量演示
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
/**
* 类的成员变量演示
* @param classFileName
* @throws ClassNotFoundException
*/
private static void fetchFields(String classFileName) throws ClassNotFoundException {
Class c = Class.forName(CLASS_FILE_NAME);
//获取该类及其父类中的所有public成员变量(属性)
Field fields[] = c.getFields();
//获取该类声明的所有成员变量,包括私有及其它权限的
//Field fields[] = c.getDeclaredFields();
for(int i=0;i<fields.length;i++){
Field f = fields[i];
//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符
System.out.println("修饰符:"+ Modifier.toString( f.getModifiers() ) );//转换成字符串修饰符了。
System.out.println("方法名称:"+ f.getName());
System.out.println("此构造方法属于的类: "+f.getDeclaringClass());//全部是当前类!因为构造方法不能被继承
System.out.println();
System.out.println("--------------------------------");
}
}
}
Class类真的很强大!
Java---类反射(1)---类反射入门和基础的更多相关文章
- java 深入技术七(类的结构和反射)
1.java反射(reflect) java 类的结构 java反射机制就是把java类按结构分解,每一部分对应特定的反射类 java反射机制允许运行时加载,探知和使用在编译期间完全未知的classe ...
- java反射工具类
package com.yingchao.kgou.core; import java.lang.reflect.Field; import java.lang.reflect.InvocationT ...
- Java反射机制(获取Class对象的三种方式+获取Class中的构造函数进行对象的初始化+获取反射类的字段+获取反射类的一般方法)
反射技术其实就是动态加载一个指定的类,并获取该类中的所有内容.而且将字节码文件封装成对象,并将字节码文件中的内容都封装成对象,这样便于操作这些成员,简单来说:反射技术可以对一个类进行解剖,反射大大增强 ...
- java 反射提取类信息, 动态代理 和过滤某些方法演示
java 反射提取类信息, 动态代理 和过滤某些方法演示 package org.rui.classts.reflects; import java.lang.reflect.Constructor; ...
- Java 反射 分析类和对象
Java 反射 分析类和对象 @author ixenos 摘要:优化程序启动策略.在运行时使用反射分析类的结构和对象 优化程序启动策略 在启动时,包含main方法的类被加载.它会加载所有它需要的类. ...
- Java 反射 Class类
Java 反射 Class类 @author ixenos 摘要:Class类在反射中的地位.构造Class对象的三种方式.Class对象构造对应类型对象的三种方式 Class类在反射中的地位 位于j ...
- java利用反射获取类的属性及类型
java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...
- Java利用反射取得类的所有信息
Java中可以利用反射获取类的名称.构造函数.属性.方法.也就是说可以通过反射可以取得类的所有信息(不管该成员是否封装为private). 如有下面的Dept类定义: package org.lyk. ...
- 反射工具类.提供调用getter/setter方法, 访问私有变量, 调用私有方法, 获取泛型类型Class,被AOP过的真实类等工具函数.java
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.Validate; import org.ap ...
随机推荐
- producer怎样发送消息到指定的partitions
http://www.aboutyun.com/thread-9906-1-1.html http://my.oschina.net/u/591402/blog/152837 https://gith ...
- C#中Dictionary、ArrayList、Hashtable和Array的区别
IDictionary接口是所有字典类集合的基本接口,该接口与ICollection,IEnumerable接口是所有非泛型类集合的最基本的接口 IEnumerable接口用于公开枚举数,该枚举数支持 ...
- MySQL命令行下查看运行状态
查看MySQL的运行状态用命令行的show语句show status是查看MySQL运行情况,和上面那种通过pma查看到的信息基本类似. show variables是查看MySQL的配置参数,还可以 ...
- CI 笔记4 (easyui 手风琴)
添加父div标签,和子div标签 <div class="easyui-accordion" data-options="fit:true,border:false ...
- Zend Server安装后首次运行就出现Internal Server Error的解决
无论是使用哪个版本的Zend Server来搭建PHP服务器,首次运行都会出现Internal Server Error的错误,对很多新手而言,每当看到这种错误时,那一刻内心绝对都是崩溃的.然而,这个 ...
- Xcode4.4中,代码无法高亮、无法自动补全
1.代码无法高亮显示:2.代码不能自动补全,或者给出提示建议:(当然这个补全的功能我在设置当中是打开的状态)3.新建一个项目,代码还是依然没有高亮显示,但是有补全功能:4.然后我就在网络上搜索了相关的 ...
- Java操作hbase总结
用过以后,总得写个总结,不然,就忘喽. 一.寻找操作的jar包. java操作hbase,首先要考虑到使用hbase的jar包. 因为咱装的是CDH5,比较方便,使用SecureCRT工具,远程连接到 ...
- Node.js的长连接
之前写的js服务器脚本,在服务器上运行的挺好的.也经过了压力测试,单次接受4000次的连接不成问题.在5s里,可以应答1W多次的连接.对于这个连接次数,我们还是挺满意的,但是Boss说:客户端每2分钟 ...
- lmsw - 加载机器状态字
将源操作数加载到机器状态字,即寄存器 CR0 的位 0 到 15.源操作数可以是 16 位通用寄存器或内存位置.只有源操作数的低 4 位(也就是 PE.MP.EM 及 TS 标志)会加载到 CR0.C ...
- ThinkPHP 自动验证与自动填充无效可能的原因
原文链接:http://www.5idev.com/p-thinkphp_validate_auto_Invalid.shtml 自动验证与自动填充是在使用ThinkPHP时经常用到的功能,但偶尔会遇 ...