java反射机制

在java的很多技术框架中,我们时常能看到函数的反射,回调等等。一开始感觉很高大上的东西,有木有?但是慢慢了解了之后,也就明白了其实就是一种很简单的方式,同时也是java设计中必不可少的机制。

一、反射机制及其作用

反射可以帮助我们在类动态运行的时候,对于任意一个类,可以获得其所有的方法(public private protected等),所有的变量(同上),然后对其进行操作。了解了反射机制的定义,我们其实可以很容易看出它的作用,获取某些类的一些私有变量。

二、代码实现

设定一个pojo类,里面含有一些私有成员变量,如下:

public class User {
    private String name;
    private String sex;


    public User() {
    }

    public String getName() {

        return name;
    }

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

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    @Override
    public String toString() {
        final StringBuilder sb = new StringBuilder("{");
        sb.append("\"name\":\"")
                .append(name).append('\"');
        sb.append(",\"sex\":\"")
                .append(sex).append('\"');
        sb.append('}');
        return sb.toString();
    }
}

java反射工具包(java.lang.reflect),利用反射来获取类的信息

public class Main {
    public static void main(String[] args) throws Exception {

        example5();
    }

    /**
     * 通过java反射机制得到类的信息
     */
    public static void example1() {
        User user = new User();
        System.out.println("例1:包名:" + user.getClass().getPackage().getName() + ","
                + "类名:" + user.getClass().getName());
    }

    /**
     * 验证所有类都是Class类的实例对象
     * @throws ClassNotFoundException
     */
    public static void example2() throws ClassNotFoundException {
        Class<?> class1 = null;
        Class<?> class2 = null;
        //动态加载类,这一步已经执行了该类中的静态代码段
        class1 = Class.forName("com.hust.reflect.User");
        System.out.println("例2.1:包名:" + class1.getClass().getPackage().getName() + ","
                + "类名:" + class1.getClass().getName());
        class2 = User.class;
        System.out.println("例2.2:包名:" + class2.getClass().getPackage().getName() + ","
                + "类名:" + class2.getClass().getName());
    }

    /**
     *通过java反射机制用Class创建类对象
     * @throws Exception
     */
    public static void example3() throws Exception{
        Class<?> class1 = null;
        class1 = Class.forName("com.hust.reflect.User");//动态加载类
        User user = (User) class1.newInstance(); //这两句相当于User user = new User();
        user.setName("钢铁侠");
        user.setSex("未知");
        System.out.println("例3:"+user.getName()+"-"+user.getSex());
    }

    /**
     * 通过java反射机制得到一个类的构造函数,并实例化对象
     * @throws Exception
     */
    public static  void example4() throws Exception{
        Class<?> class1 = null;
        User user1 = null;
        User user2 = null;
        class1 = Class.forName("com.hust.reflect.User");//动态加载类
        //得到一系列构造函数集合
        Constructor<?>[] constructors = class1.getConstructors();
        user1 = (User) constructors[0].newInstance();
        user2 = (User) constructors[1].newInstance();
        user1.setName("black widow");
        user2.setName("doctor strange");
        System.out.println("例4:"+user1.getName()+"-"+user2.getName());

    }

    /**
     * 通过反射操作成员变量
     * @throws Exception
     */
    public static  void example5() throws Exception {
        Class<?> class1 = null;
        class1 = Class.forName("com.hust.reflect.User");//动态加载类
        Object obj = class1.newInstance();
        Field userNameField = class1.getDeclaredField("name");
        userNameField.setAccessible(true);
        userNameField.set(obj,"groot");
        System.out.println("例5:获取成员变量并进行设置"+userNameField.get(obj));
    }

    /**
     * 通过反射得到类的一些属性:成员信息、函数信息;当然还可已获得父类、继承的接口等等
     * @throws Exception
     */
    public static void example6() throws Exception{
        Class<?> class1 = null;
        class1 = Class.forName("com.hust.reflect.User");//动态加载类
        Field[] fields = class1.getDeclaredFields();
        for (int i = 0; i <fields.length ; i++) {
            System.out.println("成员变量:"+fields[i]);
        }
        Method[] methods = class1.getDeclaredMethods();
        for (int i = 0; i < methods.length ; i++) {
            System.out.println("取得User类的方法:"+"\n"
                                +"函数名:"+methods[i].getName()+"\n"
                                +"函数返回的类型:"+methods[i]+"\n"
                                +"函数代码写法:"+methods[i]);
        }

    }

    /**
     * 通过反射机制调用类方法
     * @throws Exception
     */
    public static void example7() throws Exception{
        Class<?> class1 = null;
        class1 = Class.forName("com.hust.reflect.User");//动态加载类

        System.out.println("调用无参方法toString()");
        Method method = class1.getMethod("toString");
        method.invoke(class1.newInstance());

    }

    /**
     * 通过反射机制得到类加载信息
     * @throws Exception
     */
    public static void example8() throws Exception{
        Class<?> class1 = null;
        class1 = Class.forName("com.hust.reflect.zcl");//动态加载类
        String loaderName = class1.getClassLoader().getClass().getName();
        System.out.println("例8:类加载类名:"+loaderName);
    }
}

new 与 newInstance的区别

用newInstance与用new是区别的,区别在于创建对象的方式不一样,前者是使用类加载机制,那么为什么会有两种创建对象方式?这个就要从可伸缩、可扩展,可重用等软件思想上解释了。
Java中工厂模式经常使用newInstance来创建对象,因此从为什么要使用工厂模式上也可以找到具体答案。

案例:
Class c = Class.forName(“A”);factory = (AInterface)c.newInstance();
其中AInterface是A的接口,如果下面这样写,你可能会理解:
String className = “A”;Class c = Class.forName(className);factory = (AInterface)c.newInstance();
进一步,如果下面写,你可能会理解:
String className = readfromXMlConfig;//从xml 配置文件中获得字符串Class c = Class.forName(className);factory = (AInterface)c.newInstance();
上面代码就消灭了A类名称,优点:无论A类怎么变化,上述代码不变,甚至可以更换A的兄弟类B , C , D….等,只要他们继承Ainterface就可以。
从jvm的角度看,我们使用new的时候,这个要new的类可以没有加载;
但是使用newInstance时候,就必须保证:1、这个类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是class的静态方法forName()方法,这个静态方法调用了启动类加载器(就是加载javaAPI的那个加载器)。
有了上面jvm上的理解,那么我们可以这样说,newInstance实际上是把new这个方式分解为两步,即,首先调用class的加载方法加载某个类,然后实例化。
这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了我们降耦的手段。

[补充:]
newInstance: 弱类型。低效率。只能调用无参构造。
new: 强类型。相对高效。能调用任何public构造。
newInstance()是实现IOC、反射、面对接口编程 和 依赖倒置 等技术方法的必然选择,new 只能实现具体类的实例化,不适合于接口编程

关于上述代码中Class.forName()的具体机制与用法,可以查看这篇博文

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • 1. 前言   Java的反射功能平时已使用了多次,但从来没有仔细的梳理过,趁着最近在梳理Java基础,再来系统的...
    骑着乌龟去看海阅读 1,354评论 0 28
  • 问题: 在运行时,对一个JAVA类,能否知道属性和方法;能否调用它的任意方法? 答案是可以的,JAVA提供一种反射...
    糖宝_阅读 759评论 0 1
  • 我们之中的好多人向别人吐露自己的掏心话,不过是因为寂寞罢了。少年少女们啊,你们可要擦亮眼睛。 孤独给我一种不一样的...
    SHERRY177阅读 128评论 0 0
  • 妞: 这个开学季,对你来说都有点特别,大学生了,一个新的身份,也代表有一个全新的开始。祝贺你迎来了你人生最好的年龄...
    微语素心阅读 400评论 2 3
  • 相信很多女性朋友和我一样,想要护肤,但是面对琳琅满目的护肤品,价格层次不齐,也会出现不知道如何选择的情况,而且...
    彩虹色的猪阅读 1,805评论 3 25