Java反射——框架设计的灵魂

java反射概念:

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

获取Class类的实例对象的三种方式

在面向对象的世界里,万事万物皆对象。(java语言中,静态的成员、普通数据类型除外)

类是不是对象呢?类是(哪个类的对象呢?)谁的对象呢?

类是对象,类是java.lang.Class类的实例对象

//第一种、已知User的实例化对象,通过user.getClass()获得Class对象       
User user = new User();
Class c1 = user.getClass();
//第二种、知道User类名通过 .class获得Class对象,需要导包       
Class c2 = User.class;
Class c3 = null;
try {
/第三种、知道User的完全名(包名.类名) 通过Class.forName("java_reflex.User")获得Class对象 
    c3 = Class.forName("java_reflex.User");
    } catch (ClassNotFoundException e) {
    // TODO Auto-generated catch block
        e.printStackTrace();
    }
System.out.println(c1==c2);//true
System.out.println(c1==c3);//true
System.out.println(c3==c2);//true
//都为true,这就说明一个类只可能是Class类的一个实例对象,只是方式不同。

静态加载和动态加载

静态加载(需要重新编译):

//获取User的实例对象
User user = new User();

动态加载(不需要重新编译):

//newInstance() 获取User的实例化对象
try {
        Class c3 = Class.forName("java_reflex.User");
        c3.newInstance();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

举个栗子:

我们手机里的app要更新升级的时候,即使你不更新,也可以正常使用,但是不能使用更新的功能。我们的杀毒软件,需要下载一个功能插件时,不需要卸载掉重新安装完整版,只需要下载插件即可。这里面就用到了反射的知识哦。不需要重新编译,扩展所需功能即可。

代码示意:

假如有个Office,具有word、excel功能。

静态方式:

public interface OfficeAble {
    public void print();
}
public class Word implements OfficeAble {

    @Override
    public void print() {
        System.out.println("word-----start");
    }
}
public class Excel implements OfficeAble {

    @Override
    public void print() {
        System.out.println("excel-------statrt");
    }
}

Office.java

public class Office {
    public static OfficeAble fun(String name) {
        OfficeAble officeAble = null;
        if("word".equals(name)) {
            officeAble = new Word();
        }
        if("excel".equals(name)) {
            officeAble = new Excel();
        }
        return officeAble;
    }

    public static void main(String[] args) {
        Office.fun("word").print();
        Office.fun("excel").print();
    }

}

静态的方式必须要保证Word、Excel这两个类必须存在。假如现在Office需要ppt的功能,就要新建一个Ppt类实现OfficeAble并且改写Office业务代码,并且还要重新编译运行。

动态方式:

public class OfficeBetter {
    public static OfficeAble fun(String className) {
        OfficeAble officeAble=null;
        try {
            //动态加载类
            Class clazz = Class.forName(className);
            officeAble =  (OfficeAble) clazz.newInstance();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return officeAble;
    }

    public static void main(String[] args) {
        OfficeBetter clazz = new OfficeBetter();
        OfficeAble officeAble = clazz.fun("demo.Ppt");
        officeAble.print();
    }

}

动态方式,要ppt功能时只需要实现OfficeAble 即可,不需要改写Office的业务代码,不需要重新编译。

注:在编辑器里不好演示是否重新编译,可通过 javac命令在cmd中测试。

通过反射对构造器、方法、属性的操作:

User.java

package java_reflex;

public class User {
    private Integer id;
    private String username;
    private String password;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
    public User(Integer id, String username, String password) {
        super();
        this.id = id;
        this.username = username;
        this.password = password;
    }
    public User() {
        super();
        // TODO Auto-generated constructor stub
    }
    public User(String username, String password) {
        super();
        this.username = username;
        this.password = password;
    }
    @Override
    public String toString() {
        return "User [username=" + username + ", password=" + password + "]";
    }
    
    private void hello() {
        System.out.println("hello world");
    }
}

test.java

package java_reflex;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class test {
    public static void main(String[] args) {
        try {
            Class clazzUser = Class.forName("java_reflex.User");
            Constructor[] constructors = clazzUser.getDeclaredConstructors();
            System.out.println("----------获取所有公有的构造器----------");
            for(Constructor c:constructors) {
                System.out.println(c);
            }
            
        Constructor  con = clazzUser.getConstructor(String.class,String.class);
        //con.setAccessible(true);
        Object obj  = con.newInstance("stefan","password");
        System.out.println("----------实例化----------");
        System.out.println(obj.toString());
        Method[] methods = clazzUser.getMethods();
        System.out.println("----------获取所有公有方法----------");
        for(Method m :methods) {
            System.out.println(m);
        }
        
        System.out.println("----------获取所有私有方法----------");
        Method[] declaredmethods = clazzUser.getDeclaredMethods();
        for(Method m :declaredmethods) {
            System.out.println(m);
        }
        
        System.out.println("----------setUsername的调用----------");
        Method setUsername = clazzUser.getMethod("setUsername", String.class);
        Object obj1 = clazzUser.newInstance();
        setUsername.invoke(obj1, "tom");
        System.out.println(obj1.toString());
        
        System.out.println("----------私有方法hello的调用----------");
        Method hello = clazzUser.getDeclaredMethod("hello", null);
        hello.setAccessible(true);
        hello.invoke(obj1, null);
        
        System.out.println("----------获得私有私有属性----------");
        Field[] fields = clazzUser.getDeclaredFields();
        for(Field f:fields) {
            System.out.println(f);
        }
        
        System.out.println("----------username的set和get----------");
        Field username = clazzUser.getDeclaredField("username");
        username.setAccessible(true);
        Object obj2 = clazzUser.newInstance();
        username.set(obj2, "xufanyun");
        System.out.println(username.get(obj2));
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

}

运行结果:
----------获取所有公有的构造器----------
public java_reflex.User(java.lang.String,java.lang.String)
public java_reflex.User()
public java_reflex.User(java.lang.Integer,java.lang.String,java.lang.String)
----------实例化----------
User [username=stefan, password=password]
----------获取所有公有方法----------
public java.lang.String java_reflex.User.toString()
public java.lang.Integer java_reflex.User.getId()
public void java_reflex.User.setId(java.lang.Integer)
public void java_reflex.User.setUsername(java.lang.String)
public java.lang.String java_reflex.User.getPassword()
public void java_reflex.User.setPassword(java.lang.String)
public java.lang.String java_reflex.User.getUsername()
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()
----------获取所有私有方法----------
public java.lang.String java_reflex.User.toString()
public java.lang.Integer java_reflex.User.getId()
public void java_reflex.User.setId(java.lang.Integer)
public void java_reflex.User.setUsername(java.lang.String)
private void java_reflex.User.hello()
public java.lang.String java_reflex.User.getPassword()
public void java_reflex.User.setPassword(java.lang.String)
public java.lang.String java_reflex.User.getUsername()
----------setUsername的调用----------
User [username=tom, password=null]
----------私有方法hello的调用----------
hello world
----------获得私有私有属性----------
private java.lang.Integer java_reflex.User.id
private java.lang.String java_reflex.User.username
private java.lang.String java_reflex.User.password
----------username的set和get----------
xufanyun

getMethods()和getDeclaredMethods()的区别:

1、getMethods(),该方法是获取本类以及父类或者父接口中所有的公共方法(public修饰符修饰的)


image

2、getDeclaredMethods(),该方法是获取本类中的所有方法,包括私有的(private、protected、默认以及public)的方法。


image

同理获取构造器和属性有无Declared的区别。

通过反射对集合泛型的深入理解

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

public class test3 {

    public static void main(String[] args) {
        List list = new ArrayList();
        list.add("aa");
        list.add(100);
        
        List<String> list1 = new ArrayList<>();
        
        list1.add("bb");
        //list1.add(100);会报错
        Class c1 = list.getClass();
        Class c2 = list1.getClass();
        System.out.println(c1==c2);
        /**
         * c1==c2 true  说明编译之后集合的泛型是去泛型化的,
         * java中集合的泛型,是防止错误输入的,只在编译阶段有效
         * 跳过编译就无效了
         */
        try {
            Object o1 = c2.newInstance();
            Method add = c2.getMethod("add", Object.class);
            add.invoke(o1,"bb");
            add.invoke(o1,200);
            add.invoke(list1, 30);
            System.out.println(o1);
            System.out.println(list1);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }

}

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

推荐阅读更多精彩内容

  • 一:java概述: 1,JDK:Java Development Kit,java的开发和运行环境,java的开发...
    慕容小伟阅读 1,774评论 0 10
  • 整理来自互联网 1,JDK:Java Development Kit,java的开发和运行环境,java的开发工具...
    Ncompass阅读 1,537评论 0 6
  • ##永乐皇帝,明成祖,朱棣,明太祖朱元璋的第四个儿子,他就像唐代的太宗李世民一样,靠着不正当的手段当上了皇帝,而同...
    飘荡的果壳阅读 397评论 0 1
  • 祿记餐厅品牌设计 转载自:左右设计 祿记是一家港式餐厅主打正宗现代的中式点心,让原来熟悉的人群与对中式点心不了解的...
    懿木智库阅读 515评论 0 0
  • 想要美好的生活,就得不断的学习学习在学习,提高自己,就会接触不同层次的人,对吗?
    沈艳雯阅读 119评论 0 1