Java基础巩固——反射

简介: 什么是反射    反射机制就是指程序运行时能够获取自身的信息。在Java中,只要给出类的名字,就可以通过反射机制来获取类的信息 哪里用的到反射机制    在jdbc中就是使用的反射来实例化对象,比如:Class.forName("com.mysql.jdbc.Driver.class").newInstance();    框架都用到反射机制,spring,hibernate、struts都是用反射机制实现的。

什么是反射


   反射机制就是指程序运行时能够获取自身的信息。在Java中,只要给出类的名字,就可以通过反射机制来获取类的信息

哪里用的到反射机制


   在jdbc中就是使用的反射来实例化对象,比如:Class.forName("com.mysql.jdbc.Driver.class").newInstance();

   框架都用到反射机制,spring,hibernate、struts都是用反射机制实现的。

反射机制的优点和缺点


  为什么要用反射机制?直接创建对象不就可以了吗,这就涉及到了动态与静态的概念,
    静态编译:在编译时确定类型,绑定对象,即通过。
    动态编译:运行时确定类型,绑定对象。动态编译最大限度发挥了java的灵活性,体现了多
    态的应用,有以降低类之间的藕合性。
    一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中.

  它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功
    能。
       它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它
    满足我们的要求。这类操作总是慢于只直接执行相同的操作。

利用反射机制能干什么


Class c=Class.forName("className");注明:className必须为全名,也就是得包含包名,比如,cn.netjava.pojo.UserInfo;
Object obj=c.newInstance();//创建对象的实例

获取构造器

  Constructor getConstructor(Class[] params)//根据指定参数获得public构造器

  Constructor[] getConstructors()//获得public的所有构造器

  Constructor getDeclaredConstructor(Class[] params)//根据指定参数获得public和非public的构造器

  Constructor[] getDeclaredConstructors()//获得public的所有构造器

获取类的方法

  Method getMethod(String name, Class[] params),根据方法名,参数类型获得方法

    Method[] getMethods()//获得所有的public方法

    Method getDeclaredMethod(String name, Class[] params)//根据方法名和参数类型,获得public和非public的方法

    Method[] getDeclaredMethods()//获得所以的public和非public方法

获取类的属性

  Field getField(String name)//根据变量名得到相应的public变量

      Field[] getFields()//获得类中所以public的方法

      Field getDeclaredField(String name)//根据方法名获得public和非public变量

      Field[] getDeclaredFields()//获得类中所有的public和非public方法

使用实例


 获取类属性:

public class TestGetField extends Object {
    private static final long serialVersionUID = -2862585049955236662L;

    public static void main(String args[]) throws Exception {
        Class<?> clazz = Class.forName("reflect.TestGetField");
        System.out.println("===============本类属性===============");
        // 取得本类属性
        Field[] fields = clazz.getDeclaredFields();
        getField(fields);


        System.out.println("==========实现的接口或者父类的属性==========");
        // 取得实现的接口或者父类的属性
        Field[] fatherField = clazz.getFields();
        getField(fatherField);
    }

    public static void getField(Field[] fields) {
        for (Field field : fields) {
            // 权限修饰符
            int mo = field.getModifiers();
            String priv = Modifier.toString(mo);
            Class<?> type = field.getType();
            System.out.println(priv + " " + type.getName() + " " + field.getName() + ";");
        }
    }
}

 

 获取类的方法:

public class TestGetMethod implements Serializable{
    private static final String  testString= "hello";
    public static void main(String args[]) throws Exception{
        Class<?> clazz = Class.forName("reflect.TestGetMethod");
        Method[] methods = clazz.getMethods();
        for (Method method :methods){
            Class<?> returnType = method.getReturnType();
            Class<?> para[] = method.getParameterTypes();
            int temp = method.getModifiers();
            System.out.print(Modifier.toString(temp));
            System.out.print(returnType.getName());
            System.out.print(method.getName());

            for (Class par:para){
                System.out.println(par.getName());
            }
            
        }
    }
}

 

 实例化类:

public class TestNewInstance {
    public static void main(String[] args) throws Exception{
        Class<?> class1 = null;
        class1 = Class.forName("reflect.User");
        // 第一种方法,实例化默认构造方法,调用set赋值
        User user = (User)class1.newInstance();
        user.setAge(20);
        user.setName("adf");
        System.out.println(user);
        // 第二种 取得全部的构造函数 使用构造函数赋值

    }
}

class User {
    private int age;
    private String name;
    public User() {
        super();
    }
    public User(String name) {
        super();
        this.name = name;
    }
    public User(int age, String name) {
        super();
        this.age = age;
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "User [age=" + age + ", name=" + name + "]";
    }
}

 

使用类的方法:

public class TestUseMethod {
    public static void main(String[] args)throws Exception{
        Class<?> clazz = Class.forName("reflect.TestUseMethod");
        // 调用reflect1方法
        Method method = clazz.getMethod("reflect1");
        method.invoke(clazz.newInstance());

        // 调用reflect2方法
        method = clazz.getMethod("reflect2", int.class, String.class);
        method.invoke(clazz.newInstance(),20,"test");
    }

    public void reflect1() {
        System.out.println("Java 反射机制 - 调用某个类的方法1.");
    }
    public void reflect2(int age, String name) {
        System.out.println("Java 反射机制 - 调用某个类的方法2.");
        System.out.println("age -> " + age + ". name -> " + name);
    }
}

 

动态代理:

public class TestProxy {
    public static void main(String args[]) throws Exception{
        MyInvocationHandler demo = new MyInvocationHandler();
        Subject subject = (Subject) demo.bind(new RealSubject());
        System.out.println(subject.say("janti",20));
    }
}

interface Subject{
    public String say(String name, int age);
}

// 定义真实项目
class RealSubject implements Subject {
    public String say(String name, int age) {
        return name + "  " + age;
    }
}

//如果想要完成动态代理,首先需要定义一个InvocationHandler接口的子类,已完成代理的具体操作。

class MyInvocationHandler implements InvocationHandler{

    private Object object = null;
    public Object bind(Object obj){
        this.object = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object temp = method.invoke(this.object,args);
        return temp;
    }
}

 

个人博客网站 http://www.janti.cn
相关文章
|
27天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
2月前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
3月前
|
存储 缓存 Java
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
这篇文章详细介绍了Java中的IO流,包括字符与字节的概念、编码格式、File类的使用、IO流的分类和原理,以及通过代码示例展示了各种流的应用,如节点流、处理流、缓存流、转换流、对象流和随机访问文件流。同时,还探讨了IDEA中设置项目编码格式的方法,以及如何处理序列化和反序列化问题。
110 1
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
|
2月前
|
Java
Java的反射
Java的反射。
43 2
|
3月前
|
存储 Java
[Java]反射
本文详细介绍了Java反射机制的基本概念、使用方法及其注意事项。首先解释了反射的定义和类加载过程,接着通过具体示例展示了如何使用反射获取和操作类的构造方法、方法和变量。文章还讨论了反射在类加载、内部类、父类成员访问等方面的特殊行为,并提供了通过反射跳过泛型检查的示例。最后,简要介绍了字面量和符号引用的概念。全文旨在帮助读者深入理解反射机制及其应用场景。
53 0
[Java]反射
|
4月前
|
安全 Java 索引
Java——反射&枚举
本文介绍了Java反射机制及其应用,包括获取Class对象、构造方法、成员变量和成员方法。反射允许在运行时动态操作类和对象,例如创建对象、调用方法和访问字段。文章详细解释了不同方法的使用方式及其注意事项,并展示了如何通过反射获取类的各种信息。此外,还介绍了枚举类型的特点和使用方法,包括枚举的构造方法及其在反射中的特殊处理。
97 9
Java——反射&枚举
|
3月前
|
安全 Java 测试技术
🌟Java零基础-反射:从入门到精通
【10月更文挑战第4天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
38 2
|
4月前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
|
3月前
|
IDE Java 编译器
java的反射与注解
java的反射与注解
32 0
|
4月前
|
Java 程序员 编译器
Java的反射技术reflect
Java的反射技术允许程序在运行时动态加载和操作类,基于字节码文件构建中间语言代码,进而生成机器码在JVM上执行,实现了“一次编译,到处运行”。此技术虽需更多运行时间,但广泛应用于Spring框架的持续集成、动态配置及三大特性(IOC、DI、AOP)中,支持企业级应用的迭代升级和灵活配置管理,适用于集群部署与数据同步场景。