黑马程序员----反射机制

------- android培训java培训、期待与您交流! ----------

JAVA反射(放射)机制:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。

java中各个java类属于同一事物,而描述着同一事物的类名就是Class。比如ClassDemo这个类就是实例对象,有不同的属性等等。

获取字节码方式:(返回Class对象)
 1、类名.class 例如:ClassDemo.class
 2、对象名.getClass()  eg:new Date().getClass();
 3、class.forName("java.util.Date");
       class.forName("java.util.Date"); 作用:1、获取date类已经编译的字节码文件  2、如果未编译,就先编译在获取字节码文件

 

九个预定义Class对象 

基本的Java类型(boolean、byte、char、short、int、long、float和double)和关键字void,每一个都对应一个Class对象

方法:isPrimitive() 判断是不是基本类型(字符串不是基本类型)   判断是否是一个数组类.isArray()

  String  s1="abc";

               Class cls1=s1.getClass();

               Class cls2=String.class;

		Class  cls3=Class.forName("java.lang.String");
		System.out.println(cls1=cls2);//true
		System.out.println(cls1=cls3);//true
		//因为以上都是获取的同一份字节码文件
		//isPrimitive() 判断是不是基本类型(字符串不是基本类型)  
		System.out.println(cls1.isPrimitive());
		System.out.println(int.class.isPrimitive());
		判断int和Integer两个类的字节码是否相同false  
		System.out.println(int.class==Integer.class);//false
		System.out.println(int.class==Integer.TYPE);//true TYPE表示基本类型int的实例
		//判断数组  false
		System.out.println(int[].class.isPrimitive());
		//判断是否是一个数组类.isArray();true  
		System.out.println(int[].class.isArray());  
什么是反射?
 反射就是把java类中各种成分映射成相应的java类(也就是说Class类中getFiled,getMethod,getConstructor三个方法分别映射成java类中成员变量、方法、构造方法)
 也就是说:对任意一个类(class文件),都能够知道这个类的属性、方法,并且都能够任意的调用。
JAVA反射机制是在 运行状态 中,对于任意一个 ,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

下面是对对reflect类的Contructor、Field、Method三个方法使用以及功能讲解:

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;


public class ClassDemo {

	/**
	 * @param 本类对reflect类的Contructor、Field、Method三个方法使用以及功能讲解
	 */
	public static void main(String[] args) throws Exception{
		//Constructor代表某个类的构造方法 
		//得到某个类所有的构造方法:
		 Constructor[] constructor1=Class.forName("java.lang.String").getConstructors();
		 //得到某一个构造方法
	     Constructor constructor=Class.forName("java.lang.String").getConstructor(StringBuffer.class);
		 System.out.println(constructor1+"-------"+constructor);
		 //创建实例对象
		 String  str=new String(new StringBuffer("abc"));//普通创建对象方式
		 //反射创建对象  注意:调用获得的方法时要用到上面相同类型的实例对象
		 String  str1=(String) constructor.newInstance(new StringBuffer("abc")); 
		 System.out.println(str1.charAt(2));
		 //得到默认无参构造函数,直接创建对象。利用缓存机制来保存默认构造函数的实例对象
		 String  str2=String.class.newInstance();
		
		 
		 //Field类
		 
		 FieldDemo  fd1=new FieldDemo(3, 5);
		 Field  fieldy=fd1.getClass().getField("y");//获得类的成员变量,主要此处获取的fieldy值不是5
		 //通过对象获取对应的值
		 System.out.println(fieldy.get(fd1));
		 Field  fieldx=fd1.getClass().getDeclaredField("x");
		 //因为x变量是private的,所以这里用到暴力反射,设置可访问。getDeclaredField  
		 fieldx.setAccessible(true);
		 System.out.println(fieldx.get(fd1));
		 
	
       changValue(fd1);
       System.out.println(fd1);
       
       
       //Method类,代表一个某个类的成员方法
       //1、得到某个类中的一个成员方法
       Method method=String.class.getMethod("charAt", int.class);
       //调用方法
       System.out.println(method.invoke(str1, 1));
       //如果说把method.invoke(str1, 1)改成method.invoke(null, 1)那么就说明Method对应的方法就是静态的
       
       /**
        *   jdk1.4和jdk1.5的invoke方法的区别:

           jdk1.5:public Object invoke(Object obj,Object... args)(可变参数)

           jdk1.4:public Object invoke(Object obj,Object[] args),
	                   即按jdk1.4的语法,需要将一个数组作为参数传递给invoke方法时,
	                   数组中的每个元素分别对应被调用方法中的一个参数,
	                   所以,调用charAt方法的代码也可以用Jdk1.4改写为 charAt.invoke(“str”, new Object[]{1})形式。
        */
       
	}
	 //写一个方法,通过反射改变对象string类型 成员变量的值。把a改成b
	public static void changValue(Object obj) throws Exception{
		//得到所有的成员变量,包括private
		Field[] field=obj.getClass().getDeclaredFields();
		//System.out.println("changdu:"+field.length);
		for (Field  field2: field) {
			//这个使用==不使用equals,因为只有一份字节码文件
			if(field2.getType()==String.class){
				String  oldValue=(String) field2.get(obj);//得到String类型的成员变量
				String newValue=oldValue.replace('a', 'b');
				field2.set(obj, newValue);
				
			}
		}
	}
}
反射的作用----实现框架功能

    框架

    我做房子卖给用户住,由用户自己安装门窗和空调,我做的房子就是框架,用户需要使用我的框架,把门窗插入进我提供的框架中。框架与工具类有区别,工具类被用户的类调用,而框架则是调用用户提供的类。

    框架要解决的核心问题

    我在写框架(房子)时,你这个用户可能还在上小学,还不会写程序呢?我写的框架程序怎样能调用到你以后写的类(门窗)呢?因为在写才程序时无法知道要被调用的类名,所以,在程序中无法直接new 某个类的实例对象了,而要用反射方式来做。

例子:

import java.io.FileInputStream;
import java.io.InputStream;
import java.util.Collection;
import java.util.HashSet;
import java.util.Properties;
import java.util.Set;

public class CollectionsDemo {

	public static void main(String[] args) throws Exception{
		
        InputStream  is=new FileInputStream("prop.properties");
        Properties  prop=new Properties();
        prop.load(is);
        String classname=prop.getProperty("className");
        Collection collection=(Collection) Class.forName(classname).newInstance();
		//Collection collection=new HashSet();
		FieldDemo   fd1=new FieldDemo(1, 1);
		FieldDemo   fd2=new FieldDemo(2, 2);
		FieldDemo   fd3=new FieldDemo(1, 1);
	    collection.add(fd1);
	    collection.add(fd2);
	    collection.add(fd3);
	    collection.add(fd1);
	   // fd1.y=4;
	  //  collection.remove(fd1);
	    System.out.println(collection.size());    
	    String  str1=new String("zhangde");
	    String  str2=new String("zhangde");
	    System.out.println(str1==str2);//比较的是地址
	    System.out.println(str1.equals(str2));//比较对象的值
	    System.out.println(str1.hashCode());
	    System.out.println(str2.hashCode());   
	    Set   hashSet=new HashSet();
	    hashSet.add(str1);
	    hashSet.add(str2);
	    System.out.println(hashSet.size());
	    
		
	}

}

-------  android培训 java培训 、期待与您交流! ----------


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值