黑马程序员--Java基础加强学习笔记之Class类、反射(Reflect)

时间:2022-01-26 00:42:19
--------------------- ASP.Net+Android+IOS开发.Net培训、期待与您交流! ------------------- 一、反射的基础Class类

1、如何得到各个字节码对应的实例对象

(1)类名.class,例如 System.class

(2)对象.getClass(),例如 new Data().getClass()

(3)Class.forName("包.类名"),例如 Class.forName("java.util.Data")  查询或加载,即有了就用,没有就加载进来

2、九个预定义的Class实例对象

8个基本类型+1个void 

二、反射

1、定义:反射就是把java类中的各种成分映射成相应的java类。

2、构造方法的反射应用,class ->constructor -> object

(1)

Constructor[] constructors = Class.getName("java.lang.String").getConstructors();
(2)

//new String(new StringBuffer("abc"));
Constructor constructor = String.class.getConstructor(StringBuffer.class);
String str = (String)constructor.newInstance(new StringBuffer("abc"));// *.newInstance("abc")运行时错误
//String.class获得String类的字节码
//String.class.getConstructor()通过字节码获得该类的相对应的构造函数
//constructor.newInstance()通过构造函数生成该类的对象
//(String) :newInstance()返回类型为Object,多态,因此必须强制转化为String
//Person p = new Chinese();
//Chinese c = (Chinese)p;

(3)Class.newInstance()。先得到默认构造方法,即无参数的构造方法,然后利用无参数构造方法来创建对象。

利用缓存机制来保存默认构造方法的实例。

3、Field类  成员变量的反射应用

(1)使用
/*
class Student{
private String name;
public int age;
public Student(String name,int age){
this.name = name;
this.age = age;
}
}
*/

Student zs = new Student("zhangsan",23);
Field fieldAge = zs.getClass().getField("age");
sop(fieldZ.get(zs));

Field fieldName = zs.getClass().getDeclaredField("name");//可以得到私有的变量
fieldName.setAccessible(true);//设置为可以访问
sop(fieldName.get(zs));//才可以访问

//fieldZ不是一个值,而是该类的所有对象都有的一个字节码属性,通过fieldZ.get(对象)来获取某个对象的该属性的具体值

//getDeclaredField()

//setAccessible(true

(2)实例:改变对象中所有String类型的值中的a改为b

public void changFieldValue(Object obj) throws Exception{
Field[] fields = obj.getClass().getFields();
for(Field field : fields){
if(field.getType() == String.class){//比较字节码,唯一,故此处使用 == 而不推荐用equals()
String oldValue = (String)field.get(obj);
String newValue = oldValue.replace("a","b");
field.set(obj,newValue);
}
}
}

4、Method类  方法的反射应用

(1)

/**传统方法
String str = "abc";
str.charAt(2);
*/
//反射方法

Method mycharAt = Class.getName("java.lang.String").getMethod("charAt",int.class);//

sop(mycharAt.invoke(str,2));//invoke(null,2);第一个参数如果为null,则表示为静态方法

(2)用反射方式执行某个类中的main方法

/*

传统方式

//Test.main(new String[]{"as","234"});

*/

//反射方式

Method method = Class.forName(className).getMethod("main",String[].class);//className类的string表示
method.invoke(null,new Object[]{new String[]{"as","234"}});

//由于jdk1.5为了兼容1.4之前的版本,故此处jvm会将参数数组拆开,作为两个参数传入.因此用object数组封装起来

//或者(Object)new String[]{"as","234"},告诉编译器将其作为一个对象传入,而不拆包

5、数组的反射

6、HashSet如何保证元素唯一性?

是通过元素的两个方法,hashCode()和equals()来完成。

*如果元素的HashCode值相同,才会判断equals是否为true。

*注意,对于判断元素是否存在、及删除都是依赖于以上两个方法。而ArrayList只依赖于equals()

*一般,对象存入HashSet之后,就不要修改那些hashCode()方法依赖的变量,否则会改变hashCode()的计算一致性,

此时再执行remove()等方法时无法正确操作,从而产生内存泄露。

7、框架的实现(集合)  反射应用





--------------------- ASP.Net+Android+IOS开发.Net培训、期待与您交流! -------------------