>  기사  >  Java  >  Java 반사 메커니즘

Java 반사 메커니즘

伊谢尔伦
伊谢尔伦원래의
2017-01-16 11:58:121475검색

프로그램이 실행될 때 프로그램 구조나 변수 유형을 변경할 수 있는 언어를 동적 언어라고 합니다. 우리는 Java가 동적 언어는 아니지만 일반적으로 리플렉션이라고 알려진 매우 중요한 동적 관련 메커니즘을 가지고 있다고 믿습니다.

IT 업계에서는 기존 프레임워크는 모두 Reflection을 기반으로 구축됐다고 합니다. 실제 프로젝트 개발에서는 프레임워크가 가장 많이 사용되며, 클래스가 가장 많이 채워집니다. 리플렉션의 개념은 프레임워크와 클래스를 결합하는 매개체입니다. 그러므로 성찰은 프로젝트 개발을 위한 디딤돌입니다!

반사 메커니즘은 무엇인가요?

반사 메커니즘은 모든 클래스에 대해 모든 속성과 상태를 알 수 있습니다. 클래스의 속성 메소드, 모든 객체의 메소드 및 속성을 호출할 수 있으며, 객체의 메소드를 동적으로 호출하는 기능을 Java 언어의 반사 메커니즘이라고 합니다.


반사 메커니즘으로 수행할 수 있는 작업

반사 메커니즘은 주로 다음 기능을 제공합니다.

  • 런타임에 객체가 속한 클래스를 결정합니다.

  • 런타임에 모든 클래스의 객체를 구성합니다. >

    런타임에 모든 클래스의 멤버 변수와 메소드를 판단합니다.
  • 런타임에 모든 객체의 메소드를 호출합니다.
  • 생성합니다. 동적 프록시.
  • 특정 함수 구현

1. Reflection 메커니즘을 통해 클래스를 얻는 방법에는 세 가지가 있습니다.

//第一种方式:  
Classc1 = Class.forName("Employee");  
//第二种方式:  
//java中每个类型都有class 属性.  
Classc2 = Employee.class;  
   
//第三种方式:  
//java语言中任何一个java对象都有getClass 方法  
Employeee = new Employee();  
Classc3 = e.getClass(); //c3是运行时类 (e的运行时类是Employee)

2 , 객체 생성: 클래스를 가져온 후 newInstance를 사용하여 해당 객체를 생성합니다:

Class c =Class.forName("Employee");  
  
//创建此Class 对象所表示的类的一个新实例  
Objecto = c.newInstance(); //调用了Employee的无参数构造方法.

3. 속성 가져오기: 모든 속성과 지정된 속성으로 구분:

먼저 모두 가져오기를 살펴보세요. 속성 작성 방법:

//获取整个类  
            Class c = Class.forName("java.lang.Integer");  
              //获取所有的属性?  
            Field[] fs = c.getDeclaredFields();  
       
                   //定义可变长的字符串,用来存储属性  
            StringBuffer sb = new StringBuffer();  
            //通过追加的方法,将每个属性拼接到此字符串中  
            //最外边的public定义  
            sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() +"{\n");  
            //里边的每一个属性  
            for(Field field:fs){  
                sb.append("\t");//空格  
                sb.append(Modifier.toString(field.getModifiers())+" ");//获得属性的修饰符,例如public,static等等  
                sb.append(field.getType().getSimpleName() + " ");//属性的类型的名字  
                sb.append(field.getName()+";\n");//属性的名字+回车  
            }  
      
            sb.append("}");  
      
            System.out.println(sb);

특정 속성을 가져오고 기존 방법과 비교하여 학습:

public static void main(String[] args) throws Exception{  
              
<span style="white-space:pre">  </span>//以前的方式:  
    /* 
    User u = new User(); 
    u.age = 12; //set 
    System.out.println(u.age); //get 
    */  
              
    //获取类  
    Class c = Class.forName("User");  
    //获取id属性  
    Field idF = c.getDeclaredField("id");  
    //实例化这个类赋给o  
    Object o = c.newInstance();  
    //打破封装  
    idF.setAccessible(true); //使用反射机制可以打破封装性,导致了java对象的属性不安全。  
    //给o对象的id属性赋值"110"  
    idF.set(o, "110"); //set  
    //get  
    System.out.println(idF.get(o));  
}

방법 반영

Class 클래스에는 가장 간단한 메소드인 getName()이 있습니다.

public class Demo2 {
public static void main(String[] args) {
Class c1 = int.class;//int 的类类型
Class c2 = String.class;//String类的类类型
Class c3 = void.class;
System.out.println(c1.getName());
System.out.println(c2.getName());
System.out.println(c2.getSimpleName());
System.out.println(c3.getName());
}
}

getName 메소드는 클래스 유형의 클래스 이름을 인쇄할 수 있습니다. getSimpleName() 메소드를 사용하여 인쇄할 수도 있습니다. 클래스 이름 없이 패키지 이름을 출력합니다. 위의 코드에서 볼 수 있듯이 클래스 유형에는 기본 데이터 유형과 void 키워드가 모두 존재합니다.

사례:

public class ClassUtil {
public static void printClassMethodMessage(Object obj){
//要获取类的信息》》首先我们要获取类的类类型
Class c = obj.getClass();
//我们知道Object类是一切类的父类,所以我们传递的是哪个子类的对象,c就是该子类的类类型。
//接下来我们要获取类的名称System.out.println("类的名称是:"+c.getName());
/*
*我们知道,万事万物都是对象,方法也是对象,是谁的对象呢?
* 在java里面,方法是Method类的对象
*一个成员方法就是一个Method的对象,那么Method就封装了对这个成员
*方法的操作
*///如果我们要获得所有的方法,可以用getMethods()方法,这个方法获取的是所有的Public的函数,包括父类继承而来的。如果我们要获取所有该类自己声明的方法,就可以用getDeclaredMethods()方法,这个方法是不问访问权限的。
Method[] ms = c.getMethods();//c.getDeclaredMethods()
//接下来我们拿到这些方法之后干什么?我们就可以获取这些方法的信息,比如方法的名字。
//首先我们要循环遍历这些方法
for(int i = 0; i < ms.length;i++){
//然后可以得到方法的返回值类型的类类型
Class returnType = ms[i].getReturnType();
//得到方法的返回值类型的名字
System.out.print(returnType.getName()+" ");
//得到方法的名称
System.out.print(ms[i].getName()+"(");
//获取参数类型--->得到的是参数列表的类型的类类型
Class[] paramTypes = ms[i].getParameterTypes();
for (Class class1 : paramTypes) {
System.out.print(class1.getName()+",");
}
System.out.println(")");
}
}
}

요약 아이디어:

메서드 반영을 통해 클래스 이름 가져오기 단계:

1. 클래스의 클래스 유형 가져오기

2. 클래스를 가져오는 클래스 Type 메서드(getMethods())
3. 획득한 메서드를 반복합니다
4. 이러한 메서드의 getReturnType()을 통해 반환 값 유형을 가져오고 반환 값을 가져옵니다. 클래스 유형 Type name
5.getName()을 통해 메소드 이름을 가져오고, getParameterTypes()는 이 메소드의 매개변수 유형의 클래스 유형을 가져옵니다.



멤버 변수 반영


먼저 멤버 변수도 객체이고 java.lang이라는 점을 알아야 합니다. .reflect .Field 클래스 객체, 즉 Field 클래스는 멤버 변수에 대한 작업을 캡슐화합니다. 멤버 변수를 캡슐화하므로 이러한 멤버 변수를 어떻게 얻을 수 있습니까?

public class ClassUtil {
public static void printFieldMessage(Object obj){
Class c = obj.getClass();
//Field[] fs = c.getFields();
}

여기서 getFields() 메서드는 모든 공용 멤버 변수의 정보를 가져옵니다. 그리고 리플렉션 메서드의 공용 멤버 변수에는 자체 선언된 모든 멤버 변수에 대한 정보도 있습니다.

Field[] fs = c.getDeclaredFields();

이를 얻은 후 순회할 수 있습니다(필드 정보가 캡슐화되어 있으므로 다음을 수행할 수 있습니다). 필드 유형 가져오기)

for (Field field : fs) {
//得到成员变量的类型的类类型
Class fieldType = field.getType();
String typeName = fieldType.getName();
//得到成员变量的名称
String fieldName = field.getName();
System.out.println(typeName+" "+fieldName);
}

생성자 리플렉션

메서드 리플렉션인지, 멤버 변수 리플렉션인지, 생성자 리플렉션인지만 알면 됩니다. 클래스 정보를 얻으려면 , 먼저 클래스의 클래스 유형을 얻어야 합니다.

아아앙

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.