Java类反射(能力更上一台阶,Java跳跃)

如果你属于初级学习者,当你看类反射之后,说明你的编程能力又升级了!

什么是类反射:

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制。

反射(Reflection)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查, 也称自审,并能直接操作程序的内部属性。例如,使用它能获得Java类中各成员的名称并显示出来。

Java的这一能力在实际应用中应用得很多,在其它的程序语言中根本就不存在这一牲。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。
JavaBean是类反射的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过类反射动态的载入并取得Java组件(类)的属性。后面学习的各种框架,基本上都会有反射的使用。

首先举个例子:

api

<span style="font-size:18px;">/**
 * 
 */
package cn.hncu.reflect.test1.api;

/**
 * @author xinxin
 *
 */
public interface Intfaceebo {
	public void work();
}</span>
UsbFactory

<span style="font-size:18px;">/**
 * 
 */
package cn.hncu.reflect.test1.Factory;

import java.io.File;
import java.io.FileInputStream;
import java.util.Properties;

import cn.hncu.reflect.test1.api.Intfaceebo;
import cn.hncu.reflect.test1.imp.Imp;

/**
 * @author xinxin
 *
 */
public class USbFatory {
	public static  Intfaceebo getUsb(){
//		S<span style="color:#000099;">tring name1 ="cn.hncu.reflect.test1.imp.Imp";
//		String name2 ="cn.hncu.reflect.test1.imp.Imp2";</span>
		try {
			// 通过修改 name  就可以调用不同的类,实现了 解耦
			//name 不需要依靠类了,只需要路径名就可以获得该类的所有信息
			<span style="color:#ff0000;">//此处手动改麻烦,还可以 过修改属性文件(这样写就比较大了,当然修改属性可以用界面,让用户自己进行修噶</span>)
			 Properties p =new Properties();
			 File file =new File("properties.cog");
			 if(!file.exists()){
				 file.createNewFile();
			 }
			 FileInputStream files =new FileInputStream(file);//配置文件需要 IO流
			 p.load(files);//读文件中的数据
			<span style="color:#ff0000;">String  name =  p.getProperty("name").trim();//从配置文件中取出来
			Class clazz=Class.forName(name);//字符串传过来的 进行类反射
			Intfaceebo c=(Intfaceebo) clazz.newInstance();//获得类反射之后的对象</span>
			return c;
		} catch (Exception e) {
			
			e.printStackTrace();
		}
		
		return  null;
	}
	public static void main(String[] args) {
		//1,
		//String name1 ="cn.hncu.reflect.test1.imp.Imp";
		//(当Class clazz=Class.forName(name1);)work1  is Working
//		USbFatory.getUsb().work();//运行结果 work1  is Working
		
		// 2,String name2 ="cn.hncu.reflect.test1.imp.Imp2";
		//Class clazz=Class.forName(name2);
//		USbFatory.getUsb().work();// work2  is Working
//		采用配置文件写之后
		USbFatory.getUsb().work();
		//name =cn.hncu.reflect.test1.imp.Imp  //work1  is Working

//		name =cn.hncu.reflect.test1.imp.Imp2 //work2  is Working

		
		
	}

}
</span>

imp包

<span style="font-size:18px;">/**
 * 
 */
package cn.hncu.reflect.test1.imp;

import cn.hncu.reflect.test1.api.Intfaceebo;

/**
 * @author xinxin
 *
 */
public class Imp implements Intfaceebo {

	
	public void work() {
		System.out.println("work1  is Working");

	}

}
</span>
imp2

<span style="font-size:18px;">/**
 * 
 */
package cn.hncu.reflect.test1.imp;

import cn.hncu.reflect.test1.api.Intfaceebo;

/**
 * @author xinxin
 *
 */
public class Imp2 implements Intfaceebo {

	
	public void work() {
		System.out.println("work2  is Working");//仅仅只是测试使用

	}

}</span>

这几个类中,factory中用到了类反射,可以根据Class.forName("name")来获得相对应的Class 对象,相关的测试以及测试的结果都在后面的注释里,main函数为了方便写到了factory里面。


反射使用的三个步骤 :

如:method

第一步:获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。

第二步:调用诸如getDeclaredMethods的方法,取得该类中定义的所有方法的列表。
第三步:使用反射的API来操作这些信息。
 

自己写的Instance 测试类

1,首先需要一个对象

/**
 * 
 */
package cn.hncu.reflect.test;

/**
 * @author xinxin
 *
 */
public class A {

}
,2,进行判断(也就是调用Class. isInstance(),运行是都匹配

/**
 * 
 */
package cn.hncu.reflect.test;

/**
 * @author xinxin
 *
 */
public class TestInstanceof {

	/**
	 * @param args
	 */
	
	/*
	 * 验证 instanceof 功能 通过这个函数isInstanceof
	 * 通过类反射Class 中的isInstance 判断下
	 */
	public static void main(String[] args) {
		<span style="color:#ff0000;">System.out.println( isInstanceof(new A()));//结果 true
		System.out.println( isInstanceof(new Integer(10)));// false</span>

	}

	private static boolean  isInstanceof( Object obj) {
		boolean isT=false;
		try {
			<span style="color:#ff9900;">Class c =Class.forName("cn.hncu.reflect.test.A");//通过传过来的字符串,获得Class类型
			isT=c.isInstance(obj);</span>
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		return isT;
	}

}

获取Class对象的三种方式

        1,通过对象的getClass方法进行获取。这种方式需要具体的类和该类的对象,以及调用getClass方法。

      2,  任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。

  3,通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。

实例演示:

/**
 * 
 */
package cn.hncu.reflect.Three;

import java.lang.reflect.Method;

/**
 * @author xinxin
 *
 */
public class ReflectThree {

	/**
	 * @param args
	 */
	//类反射必须先有类,导入Person
	//这仅仅是获得Class 对象
	public static void main(String[] args) {
//		getObject1();//通过对象的getClass方法进行获取。
		//、这种方式需要具体的类和该类的对象,以及调用getClass方法。
		
		//2,任何数据类型(包括基本数据类型)都具备着一个静态的属性class,
//		通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,
//		然后调用类中的静态属性class完成,无需调用方法,性能更好。
//		getObject2();
		
		//3,通过Class.forName()方法获取。这种方式仅需使用类名,
//		就可以获取该类的Class对象,更有利于扩展。、
	//	getObject3();
		
		//4 小小的先实验一下 获得类反射出的方法
		getMethod();

	}

	

	<span style="color:#ff0000;">//1,已知具体的类 </span>
	private static void getObject1() {
		Person p =new Person("jack", 11);
		Class c =<span style="color:#ff0000;">p.getClass()</span>;//获得Class 对象
		System.out.println(c);//class cn.hncu.reflect.Three.Person
		
	}
//任何对象包括基本数据类型都有.class 获得Class 对象
	private static void getObject2() {
		Class c =<span style="color:#ff0000;">String.class</span>;
		System.out.println(c);//class java.lang.String
		
	}

	
	/*
	 <span style="color:#ff0000;">* 通过Class.forName(str)的方式获取Class对象,
	 * 该方式依赖的是字符串(类的名字),可以实现解耦</span>
	 */
	private static void getObject3() {
		String name ="cn.hncu.reflect.Three.Person";
		try {
			<span style="color:#ff0000;">Class  c =Class.forName(name)</span>;
			System.out.println(c);//class cn.hncu.reflect.Three.Person
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
	}
	
	private static void getMethod() {
		String name ="cn.hncu.reflect.Three.Person";
		try {
			Class c =Class.forName(name);
//			Object obj=c.newInstance();// 创建此 Class 对象所表示的类的一个新实例。
			Method method[] =c.getMethods();//获得包括本身,父类的所有public方法
//			Method method[] =c.getDeclaredMethods();//获得当前类的所有方法包括 private 等
			for(int i=0;i<method.length;i++){
				System.out.println(method[i]);
			}
			
			
		} catch (Exception e) {
			
			e.printStackTrace();
		}
		
		
	}
	

}

前面的基础基本介绍完了,现在开始解刨得到类对象之后:

获取类的方法:找出一个类中定义了些什么方法,这是一个非常有价值也非常基础的反射用法。

代码演示:首先需要一个未知的类(USerModel*(在做书店管理系统时用到的,直接拷贝过来)*)

/**
 * 
 */
package cn.hncu.reflect.reflectDecompose;

import java.io.Serializable;

public class UserModel{
	private String uuid;
	private String name;
	private static int type;
	private String pwd;
	
	
	public UserModel(String uuid, String name, int type, String pwd) {
		super();
		this.uuid = uuid;
		this.name = name;
		this.type = type;
		this.pwd = pwd;
	}

	public UserModel(String uuid, int type){
		this.uuid = uuid;
		this.type = type;
	}
	public UserModel(){
	}
	private UserModel(String uuid){
		this.uuid = uuid;
	}
	public String getUuid() {
		return uuid;
	}

	public void setUuid(String uuid) {
		this.uuid = uuid;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getType() {
		return type;
	}

	public void setType(int type) {
		this.type = type;
	}

	public String getPwd() {
		return pwd;
	}

	public void setPwd(String pwd) {
		this.pwd = pwd;
	}

	public String toString() {
		return "{"+uuid+","+name+","+type+","+pwd+"}";
	}
	
	private long privateMethod( double a){
		return (long)a;
	}
	
	void defaultMethod(int i){
		System.out.println("aa");
	}
	
}







然后就是我们的测试类:

获取类的方法

	private static void fetchMehtod(String name) throws Exception{
		
			Class c=Class.forName(name);
			Method method[] =c.getMethods();
			for(int i=0;i<method.length;i++){
				//方法名
				Method m =method[i];
				System.out.println("name:"+m.getName());
				System.out.println(m.getDeclaringClass());
				//getDeclaringClass(),,如果此 Class 对象所表示的类或接口是另一个类的成员,则返回的 Class 对象表示该对象的声明类。
				
				// 获得方法出现的异常
				Class[] ex=	m.getExceptionTypes();
				for(int j=0;j<ex.length;j++){
					System.out.println("exception:"+ex[j]);
				}
				
				//参数类型
				Class par[]=m.getParameterTypes();
				for(int j=0;j<par.length;j++){
					System.out.println("Parameter:"+par[j]);
				}
				System.out.println("type:"+m.getReturnType());
	}
}

获取类的构造器:找出一个类中定义的构造方法,构造器没有返回类型

/**
	 * 获取类中构造函数的信息
	 * getConstructors() : 获取当前类的public构造方法
	 * getDeclaredConstructors() :获取当前类声明的所有构造方法,包括private等其它非public方法
	 */
		private static void fetchConstror(String name2) throws Exception {
			Class c=Class.forName(name);
			
//			Constructor[] con=	c.getConstructors();
			Constructor[] con=	c.getDeclaredConstructors();
			for(int i=0;i<con.length;i++){
				//方法名
				Constructor cons=con[i];
				System.out.println("name:"+cons.getName());
				System.out.println("DeclaringClass"+cons.getDeclaringClass());
				//getDeclaringClass(),,如果此 Class 对象所表示的类或接口是另一个类的成员,则返回的 Class 对象表示该对象的声明类。
				
				// 获得方法出现的异常
				Class[] ex=	cons.getExceptionTypes();
				for(int j=0;j<ex.length;j++){
					System.out.println("exception:"+ex[j]);
				}
				
				//参数类型
				Class par[]=cons.getParameterTypes();
				for(int j=0;j<par.length;j++){
					System.out.println("Parameter:"+par[j]);
				}
			
	}
		
	}
获取类的属性字段:找出一个类中定义了哪些属性字段。

private static void fetField(String name) throws Exception{
		Class c=Class.forName(name);
//		Field field[]=c.getFields();//只能访问 public权限
		Field field[]=c.getDeclaredFields();//获取当前类声明的所有构造方法,包括private等其它非public变量
		for(int i=0;i<field.length;i++){
			 Field f=field[i];
			 System.out.println("name:"+f.getName());//变量名
			 System.out.println("Declaring:"+f.getDeclaringClass());//该类的明细路径
			 System.out.println("type:"+f.getType());// 变量的类型
			 int mod =f.getModifiers();
			 System.out.println("Modifiers::"+f);//变量的修饰符
			 System.out.println("Modifiers::"+Modifier.toString(mod));//输出类型 如  public  private
}

一起到这里我们已经会从获得的类中得到未知类的信息了,后面会有持续更新。



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值