java反射

什么是反射

简单的来说,一般情况下,都是知道一个类,然后由类去创建对象

而反射就算反过来,有一个对象去获得类,去获取类的这个过程就叫做反射

然后获取了类之后,在用这个类对象去获取属性方法,并使用

值得注意的是采用了反射的情况代码运行速度比未使用反射的代码运行速度慢很多

获取Class类对象的方式

先定义两个类

class Father{
}

class Son extends Father{
}

通过类获取

 Class c1 = Son.class;
 System.out.println(c1);

class com.kehao.Son

通过对象获取

Son son = new Son();
Class c2 = son.getClass();
System.out.println(c2);

class com.kehao.Son

通过全类名获取

Class c3 = Class.forName("com.kehao.Son");
System.out.println(c3);

class com.kehao.Son

基本数据类型的TYPE属性

Class c4 = Integer.TYPE;
System.out.println(c4);

int

使用Class对象获取父类的Class对象

Class c5 = c1.getSuperclass();
System.out.println(c5);

class com.kehao.Father

获取类对象相关属性

先定义一个类

public class User {
    public String id;
    private String name;
    private int age;

    public User() {
    }
    
    private User(String name){
        this.name = name;
    }


    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
    
    private void hello(){
        System.out.println("This is a private hello method!");
    }
}

获取Class对象

  Class cUser = Class.forName("com.kehao.User");

获取类的一些参数

比如类的名字等

String cUserName = cUser.getName();
String cUserSimpleName = cUser.getSimpleName();
System.out.println("getName:"+cUserName);
System.out.println("getSimpleName:"+cUserSimpleName);

getName:com.kehao.User
getSimpleName:User

获取成员变量

Field[] fields= cUser.getFields();//只会返回public成员变量
for(Field field:fields){
    System.out.println(field);
}

public java.lang.String com.kehao.User.id

Field[] declaredFields= cUser.getDeclaredFields();//返回所有的成员变量
for(Field field:declaredFields){
    System.out.println(field);
}

public java.lang.String com.kehao.User.id
private java.lang.String com.kehao.User.name
private int com.kehao.User.age

Field age = cUser.getDeclaredField("age");
System.out.println(age);

private int com.kehao.User.age

获取方法

返回所有的public的方法(包括父类的方法)

Method[] methods = cUser.getMethods();
for(Method method:methods){
    System.out.println(method);
}

public java.lang.String com.kehao.User.id
public java.lang.String com.kehao.User.id
private java.lang.String com.kehao.User.name
private int com.kehao.User.age
private int com.kehao.User.age
public java.lang.String com.kehao.User.toString()
public java.lang.String com.kehao.User.getName()
public void com.kehao.User.setName(java.lang.String)
public void com.kehao.User.setAge(int)
public int com.kehao.User.getAge()
public final void java.lang.Object.wait() throws java.lang.InterruptedException
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
public boolean java.lang.Object.equals(java.lang.Object)
public native int java.lang.Object.hashCode()
public final native java.lang.Class java.lang.Object.getClass()
public final native void java.lang.Object.notify()
public final native void java.lang.Object.notifyAll()

返回所有的方法(包括私有的方法,不包括父类的方法)

Method[] declaredMethods = cUser.getDeclaredMethods();
for(Method method :declaredMethods){
    System.out.println(method);
}

public java.lang.String com.kehao.User.toString()
public java.lang.String com.kehao.User.getName()
public void com.kehao.User.setName(java.lang.String)
private void com.kehao.User.hello()
public void com.kehao.User.setAge(int)
public int com.kehao.User.getAge()

返回具体的方法

Method getName = cUser.getMethod("getName");
System.out.println(getName);
Method setName = cUser.getMethod("setName", String.class);
System.out.println(setName);

public java.lang.String com.kehao.User.getName()
public void com.kehao.User.setName(java.lang.String)

获得构造器

Constructor[] constructors = cUser.getConstructors();//返回public构造器
for(Constructor constructor : constructors){
    System.out.println(constructor);
}

public com.kehao.User(java.lang.String,int)
public com.kehao.User()

Constructor[] declaredConstructors = cUser.getDeclaredConstructors();//返回所有构造器
for (Constructor constructor :declaredConstructors){
    System.out.println(constructor);
}

public com.kehao.User(java.lang.String,int)
private com.kehao.User(java.lang.String)
public com.kehao.User()

Constructor constructor = cUser.getConstructor(String.class,int.class);
System.out.println(constructor);

public com.kehao.User(java.lang.String,int)

创建对象并调用方法

创建对象

使用无参构造器创建对象

User user  = (User) cUser.newInstance();

使用有参构造器创建对象

Constructor constructor = cUser.getConstructor(String.class,int.class);
User kehao = (User)constructor.newInstance("kehao", 20);

调用方法

Constructor constructor = cUser.getConstructor(String.class,int.class);
User kehao = (User)constructor.newInstance("kehao", 20);
System.out.println(kehao);

Method setName = cUser.getMethod("setName", String.class);
setName.invoke(kehao,"Liming");
System.out.println(kehao);

User{name=‘kehao’, age=20}
User{name=‘Liming’, age=20}

操作属性

Constructor constructor = cUser.getConstructor(String.class,int.class);
User kehao = (User)constructor.newInstance("kehao", 20);
System.out.println(kehao);

Field name = cUser.getDeclaredField("name");
name.set(kehao,"XiaoMIng");
System.out.println(kehao);
User{name='kehao', age=20}
Exception in thread "main" java.lang.IllegalAccessException: Class com.kehao.Test04 can not access a member of class com.kehao.User with modifiers "private"
   at sun.reflect.Reflection.ensureMemberAccess(Reflection.java:102)
   at java.lang.reflect.AccessibleObject.slowCheckMemberAccess(AccessibleObject.java:296)
   at java.lang.reflect.AccessibleObject.checkAccess(AccessibleObject.java:288)
   at java.lang.reflect.Field.set(Field.java:761)
   at com.kehao.Test04.main(Test04.java:30)

发生了报错,原因在于name是私有变量权限不够,但是反射可以忽视掉这个问题,强行修改变量值,修改代码如下

Constructor constructor = cUser.getConstructor(String.class,int.class);
User kehao = (User)constructor.newInstance("kehao", 20);
System.out.println(kehao);

Field name = cUser.getDeclaredField("name");
name.setAccessible(true);
name.set(kehao,"XiaoMIng");
System.out.println(kehao);

User{name=‘kehao’, age=20}
User{name=‘XiaoMIng’, age=20}

name.setAccessible(true);这句代码是用于开关安全检查的,当关闭安全检查的时候,代码运行速度也会相应的加块,但是比未使用反射机制的情况相比依然慢许多

获取泛型信息

获取参数中的泛型

public class Test05 {
    //参数中有泛型的函数
    public void test01(Map<String,User> map, List<User> list,int i){
        System.out.println("Demo.test01()");
    }
    
    public static void main(String[] args) throws Exception {
        Method test01 = Test05.class.getMethod("test01", Map.class,List.class,int.class);
        Type[] t = test01.getGenericParameterTypes();//返回所有的参数(也有不是泛型的参数)

        for (Type paramType : t) {
            System.out.println("Type:"+paramType);//输出所有的参数
            if(paramType instanceof ParameterizedType){//如果是带泛型的参数
                //获取泛型中的具体信息
                Type[] genericTypes = ((ParameterizedType) paramType).getActualTypeArguments();
                for (Type genericType : genericTypes) {
                    System.out.println("泛型类型:"+genericType);
                }
            }else {
                System.out.println("不是泛型类型的参数");
            }
        }
    }
}

Type:java.util.Map<java.lang.String, com.kehao.User>
泛型类型:class java.lang.String
泛型类型:class com.kehao.User
Type:java.util.List<com.kehao.User>
泛型类型:class com.kehao.User
Type:int
不是泛型类型的参数

获取返回值中的泛型

public class Test05 {
    //返回值中有泛型
    public Map<Integer,User> test02(){
        System.out.println("Demo.test02()");
        return null;
    }

    public static void main(String[] args) throws Exception{
        Method m2 = Test05.class.getMethod("test02", null);
        Type returnType = m2.getGenericReturnType();//得到返回值类型
        if(returnType instanceof ParameterizedType){
            Type[] genericTypes = ((ParameterizedType) returnType).getActualTypeArguments();
            for (Type genericType : genericTypes) {
                System.out.println("返回值,泛型类型:"+genericType);
            }
        }
    }
}

返回值,泛型类型:class java.lang.Integer
返回值,泛型类型:class com.kehao.User

获取注解信息

public class Test06 {
    public static void main(String[] args) throws Exception {
        Class cPerson = Person.class;
        Annotation[] annotations = cPerson.getAnnotations();
        for (Annotation annotation : annotations){
            System.out.println("循环:"+annotation);
        }
        Annotation myTable = cPerson.getAnnotation(MyTable.class);
        System.out.println(myTable);
        System.out.println(((MyTable)myTable).value());

        Field name = cPerson.getDeclaredField("name");
        MyField nameAnnotation = name.getAnnotation(MyField.class);
        System.out.println(nameAnnotation);
        System.out.println(nameAnnotation.column());
        System.out.println(nameAnnotation.type());
        System.out.println(nameAnnotation.length());

    }
}

@MyTable("person_table")
class Person{
    @MyField(column = "col_name",type = "char",length = 20)
    private String name;
    @MyField(column = "col_age",type = "int",length = 20)
    private int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface MyTable{
    String value();
}

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface MyField{
    String column();
    String type();
    int length();
}

循环:@com.kehao.MyTable(value=person_table)
@com.kehao.MyTable(value=person_table)
person_table
@com.kehao.MyField(column=col_name, type=char, length=20)
col_name
char
20

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值