反射技术总结

反射技术;其实就是动态加载一个指定的类,并获取该类中的所有内容。

而且将字节码文件与内容都封装成了对象。

这样便于操作这些成员,简单说;反射技术可以对一个类进行解刨。

反射的好处;大大的增强了程序的扩展性。

反射的基本步骤;

获得Class对象。就是获取到指定的名称的字节码文件对象。

实例化对象,获得类的属性,方法或构造函数。

访问属性,调用方法,调用构造函数创建对象。

获取Class对象有三种方法。

每个类都具备getClass();方法,弊端;必须要创建该类对象,才可以调用该方法。

每一个数据类型(基本数据类型和引用数据类型)都有一个静态的属性class。

前两种方式不利于程序的扩展,因为都需要在程序中使用具体的类来完成。

使用Class类中的静态方法forName。

指定什么类名,就获取什么类的字节码文件对象,输入类名字符串即可,扩展性强。

有八个基本数据类型。每一个基本数据类型都对应着Class字节码文件对象,void.class等等--

import java.lang.reflect.*;
class most1{
	public static void main(String[] args)throws ClassNotFoundException{
		String s1="abc";
		Class c1=s1.getClass();
		Class c2=String.class;
		Class c3=Class.forName("java.lang.String");
		System.out.println(c1==c2);
		System.out.println(c1==c3);
		System.out.println(c1.isPrimitive());
		System.out.println(int.class.isPrimitive());
		System.out.println(int.class==Integer.class);
		System.out.println(int.class==Integer.TYPE);
		System.out.println(int[].class.isPrimitive());
		System.out.println(int[].class.isArray());
	}
}

反射;就是把java类中的各个成分映射成相应的类,反射比较消耗资源。

得到类的所有构造方法。

Constructor[] tts=Class.forName("java.lang.String").getConstructors();

得到类的一个构造方法。

Constructor tts=Class.forName("java.lang.String").getConstructor(StringBuffer.class);

import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args)throws NoSuchMethodException,InstantiationException,IllegalAccessException,InvocationTargetException{
		Constructor tts=String.class.getConstructor(StringBuffer.class);
		String s1=(String)tts.newInstance(new StringBuffer("abc"));
		System.out.println(s1.charAt(2));
	}
}

获取类的变量:ReflectPoint pt=new ReflectPoint(3,5);

Field fieldY=pt.getClass().getField("y");

System.out.println(field.get(pt));

获取类的私有变量:Field fieldX=pt.getClass().getDeclaredField("x");

fieldX.setAccessible(true);

System.out.println(fieldX.get(pt));

import java.lang.reflect.*;
class ReflectPoint{
	public int x;
	private int y;
	ReflectPoint(int x,int y){
		this.x=x;
		this.y=y;
	}
}
class ReflectDemo{
	public static void main(String[] args)throws NoSuchFieldException,IllegalAccessException{
		ReflectPoint pt=new ReflectPoint(3,3);
		Field fieldX=pt.getClass().getField("x");
		System.out.println(fieldX.get(pt));
		Field fieldY=pt.getClass().getDeclaredField("y");
		fieldY.setAccessible(true);
		System.out.println(fieldX.get(pt));
	}
}

获取类的方法;// Class c1=obj.getClass();

Method method=Object.class.getMethod("charAt",int.class);

System.out.println(method.invoke(obj,index));

import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args)throws NoSuchMethodException,IllegalAccessException,InvocationTargetException{
		String s="abc";
		Class c1=s.getClass();
		Method method=String.class.getMethod("charAt",int.class);
		System.out.println(method.invoke(s,1));
		System.out.println(method.invoke(s,new Object[]{2}));
	}
}

反射的用法;首先需要提供类的Class对象,获得Class对象的三种方式。

Class.forName() 用于做类加载。

obj.getClass()
用于获得对象的类型。

类名.class()
用于获得指定的类型,传参用。

获取了字节码文件对象后,最终都需要创建指定类的对象。

创建对象的两种方式(其实就是对象在进行实例化时的初始化方式)

调用空参数的构造函数;使用了Class类中的newInstance方法。

调用带参数的构造函数;先要获取指定参数列表的构造函数对象。

然后通过该构造函数对象的newInstance进行对象的初始化。

替换一个类的变量内容;

import java.lang.reflect.*;
class ReflectPoint{
	String s1="abll";
	String s2="base";
	String s3="itcast";
	public String toString(){
		return s1+"--"+s2+"--"+s3;
	}
}
class ReflectDemo{
	public static void main(String[] args)throws IllegalAccessException{
		ReflectPoint pt=new ReflectPoint();
		Field[] fields=pt.getClass().getDeclaredFields();
		for(Field field:fields){
			if(field.getType()==String.class){
				String str=(String)field.get(pt);
				String value=str.replace('b','a');
				field.set(pt,value);
			}
		}
		System.out.println(pt);
	}
}

写一个程序;这个程序能够根据用户提供的类名去执行该类中的main方法。

启动java程序的main方法是一个字符数组,public static void main(String[] args){}

通过反射方式来调用这个main方法,按照jdk1.5的语法给invoke传递参数。

整个数组就是一个参数,而jdk1.4的语法数组中的每一个元素都对应一个参数。

当把一个字符串数组作为参数传递给invoke方法时,1.5兼容1.4的语法。

会按照1.4的语法进行处理,就将数组打散成若干个元素。

所以在给main方法传递参数时,不能使用代码。

method.invoke(null,new String[]{"","",""});java只把他当做1.4的语法处理。

而不把他当做1.5的语法处理,因此会出现参数类型不对的问题。

method.invoke(null,new Object[]{new String[]{"","",""}});

method.invoke(null,(Object)new String{}{"","",""});

编译器会作特殊处理;编译时不把参数当做数组看待,也就不会将数组打散。

import java.lang.reflect.*;
class ReflectPoint{
	public static void main(String[] args)throws InvocationTargetException,ClassNotFoundException,NoSuchMethodException,IllegalAccessException{
		for(String str: args){
			System.out.println(str);
		}
	}
}
class ReflectDemo{
	public static void main(String[] args)throws InvocationTargetException,ClassNotFoundException,NoSuchMethodException,IllegalAccessException{
		String startName=args[0];
		Method mainMethod=Class.forName(startName).getMethod("main",String[].class);
		mainMethod.invoke(null,(Object)new String[]{"111","222","333"});
	}
}

具有相同维数和元素类型的数组属于同一类型,具有相同Class实例对象。

代表数组的Class实例化对象的getSuperclass方法返回的是父类对应的Class。

基本类型的一维数组可以被当做Object类型使用,不能当做Object[]类型使用。

非基本类型的一维数组,既可以当做Object类型使用,又可以当做Object[]类型使用。

Arrays.asList方法处理int[]和String[]时的差异。

Array工具类用于完成数组的反射操作。

思考题;怎么得到数组中的元素类型?

import java.lang.reflect.*;
class ReflectDemo{
	public static void main(String[] args){
		String[] arr={"x","y","z"};
		printObject(arr);
		printObject("xyz");
	}
	public static void printObject(Object obj){
		Class c=obj.getClass();
		if(c.isArray()){
			int len=Array.getLength(obj);
			for(int x=0; x<len; x++){
				System.out.println(Array.get(obj,x));
			}
		}else{
			System.out.println(obj);
		}
	}
}
package pack;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
class ReflectPoint{
<span style="white-space: pre;">	</span>private int x;
<span style="white-space: pre;">	</span>private int y;
<span style="white-space: pre;">	</span>ReflectPoint(int x,int y){
<span style="white-space: pre;">		</span>this.x=x;
<span style="white-space: pre;">		</span>this.y=y;
<span style="white-space: pre;">	</span>}
<span style="white-space: pre;">	</span>public boolean equals(Object obj){
<span style="white-space: pre;">		</span>if(!(obj instanceof ReflectPoint))
<span style="white-space: pre;">			</span>throw new RuntimeException("类型不匹配");
<span style="white-space: pre;">		</span>ReflectPoint r=(ReflectPoint)obj;
<span style="white-space: pre;">		</span>return this.x==(r.x)&&this.y==r.y;
<span style="white-space: pre;">	</span>}
}
class ReflectDemo{
<span style="white-space: pre;">	</span>public static void main(String[] args){
<span style="white-space: pre;">		</span>Collection coll=new ArrayList();
<span style="white-space: pre;">		</span>ReflectPoint pt1=new ReflectPoint(3,3);
<span style="white-space: pre;">		</span>ReflectPoint pt2=new ReflectPoint(5,5);
<span style="white-space: pre;">		</span>ReflectPoint pt3=new ReflectPoint(3,3);
<span style="white-space: pre;">		</span>coll.add(pt1);
<span style="white-space: pre;">		</span>coll.add(pt2);
<span style="white-space: pre;">		</span>coll.add(pt3);
<span style="white-space: pre;">		</span>coll.add(pt1);
<span style="white-space: pre;">		</span>System.out.println(coll.size());
<span style="white-space: pre;">	</span>}
}
时间: 2024-07-29 13:22:16

反射技术总结的相关文章

Java中类加载机制和反射技术

我们知道一个对象在运行时有两种类型,一个是编译类型,一个是运行时类型.在程序运行时,往往是需要发现类和对象的真实的信息的.那么如何获的这种信息呢? 其一,如果我们在编译和运行时都知道类型的具体信息,这时是可以手动将一个对象转换为运行时的类型. 其二,如果我们在编译时无法预知对象和类到底是属于哪些类,那么程序只有依靠运行时的信息来发现对象和类的真实的信息了,这时就必须要用到反射技术. 在谈具体的发射技术之前,我想先回顾下,有关类的加载的一些基本的性质和原理,以方便我们更好地理解,反射的作用和特点.

7. 反射技术:其实就是动态加载一个指定的类

反射技术:其实就是动态加载一个指定的类,并获取该类中的所有的内容.而且将字节码文件封装成对象,并将字节码文件中的内容都封装成对象,这样便于操作这些成员.简单说:反射技术可以对一个类进行解剖. 反射的好处:大大的增强了程序的扩展性. 反射的基本步骤: 1.获得Class对象,就是获取到指定的名称的字节码文件对象. 2.实例化对象,获得类的属性.方法或构造函数. 3.访问属性.调用方法.调用构造函数创建对象. 获取这个Class对象,有三种方式: 1:通过每个对象都具备的方法getClass来获取.

C#反射技术概念作用和要点

反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等.还可以获得每个成员的名称.限定符和参数等.有了反射,即可对每一个类型了如指掌.如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道. 1..NET可执行应用程序结构 程序代码在编译后生成可执行的应用,我们首先要了解这种可执行应用程序的结构. 应用程序结构分为应用程序域—程序集—模块—类型—成员

Java反射技术学习总结

-------<a href="http://www.itheima.com/"">android培训</a>.<a href="http://www.itheima.com/" ">java培训</a>期待与您交流!---------- Java反射技术就是将java中的类的各个组成部分反射出来变为java中相应的数据类型获得. Java同样将类中的各个组成部分进行封装,变为一个个Java类. C

反射技术

反射技术:其实就是动态加载一个指定的类,并获取该类中的所有的内容.而且将字节码文件封装成对象,并将字节码文件中的内容都封装成对象,这样便于操作这些成员.简单说:反射技术可以对一个类进行解剖. 反射的好处:大大的增强了程序的扩展性. 反射的基本步骤: 1.获得Class对象,就是获取到指定的名称的字节码文件对象. 2.实例化对象,获得类的属性.方法或构造函数. 3.访问属性.调用方法.调用构造函数创建对象. 获取这个Class对象,有三种方式: 1:通过每个对象都具备的方法getClass来获取.

Android中Java反射技术的使用示例

import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; import android.os.Bundle; import android.app.Activity; /** * Demo描述: * Android中Java反射技术的使用示例 * 在Java中描述字节码文件(xxx.class)的类叫Class * 反射的过程可视为剖析Class的过

c#中反射技术在Unity中的运用

反射技术给类赋值的好处就是可以简化代码,封装的好处就显而易见了.最直接的用途就是用在在显示配置文件的时候,个人习惯性做法是做一个VO来存储需要的数据,其代码如下: internal class BaseItemVO { public string name; public string lockA; } 运用反射来获取类中的字段: public static void setValue(Object tar, String name, Object value) { FieldInfo fInf

C# 反射技术应用

反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等.还可以获得每个成员的名称.限定符和参数等.有了反射,即可对每一个类型了如指掌.如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道. 反射是一种机制,通过这种机制我们可以知道一个未知类型的类型信息.比如,有一个对象a,这个对象不是我们定义的,也许是通过网络捕捉到的,也许是使用泛型定义的,但我们

C#反射技术的简单操作(读取和设置类的属性)

public class A { public int Property1 { get; set; } } static void Main(){ A aa = new A(); Type type = aa.GetType();//获取类型 System.Reflection.PropertyInfo propertyInfo = type.GetProperty("Property1"); propertyInfo.SetValue(aa, 5, null);//给对应属性赋值 i

小怪兽 Java反射技术 等你来打

Java反射技术,是java的难点,也是程序员进化过程中的必打小怪兽,这里就根据java api好好研究一下java的反射技术. Class Fields Methods 请先回忆一下[java类的定义]. 一.反射是什么 反射是什么?百度百科这样定义:JAVA反射机制是在运行状态下,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取对象信息以及动态调用对象方法的功能称为java语言的反射机制. 可以这样理解,大家小时候有木有玩过平