一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務(wù)器之家 - 編程語言 - JAVA教程 - Java 反射機(jī)制詳解及實(shí)例代碼

Java 反射機(jī)制詳解及實(shí)例代碼

2020-06-15 12:02Rellen Holt JAVA教程

本文主要介紹Java 反射機(jī)制的知識,這里提供示例代碼幫助大家學(xué)習(xí)理解此部分知識,有需要的小伙伴可以參考下

Java反射詳解

本篇文章依舊采用小例子來說明,因?yàn)槲沂冀K覺的,案例驅(qū)動是最好的,要不然只看理論的話,看了也不懂,不過建議大家在看完文章之后,在回過頭去看看理論,會有更好的理解。

下面開始正文。

【案例1】通過一個對象獲得完整的包名和類名

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package Reflect;
 
/**
 * 通過一個對象獲得完整的包名和類名
 * */
class Demo{
  //other codes...
}
 
class hello{
  public static void main(String[] args) {
    Demo demo=new Demo();
    System.out.println(demo.getClass().getName());
  }
}

【運(yùn)行結(jié)果】:Reflect.Demo

添加一句:所有類的對象其實(shí)都是Class的實(shí)例。

【案例2】實(shí)例化Class類對象

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package Reflect;
class Demo{
  //other codes...
}
 
class hello{
  public static void main(String[] args) {
    Class<?> demo1=null;
    Class<?> demo2=null;
    Class<?> demo3=null;
    try{
      //一般盡量采用這種形式
      demo1=Class.forName("Reflect.Demo");
    }catch(Exception e){
      e.printStackTrace();
    }
    demo2=new Demo().getClass();
    demo3=Demo.class;
     
    System.out.println("類名稱  "+demo1.getName());
    System.out.println("類名稱  "+demo2.getName());
    System.out.println("類名稱  "+demo3.getName());
     
  }
}

【運(yùn)行結(jié)果】:

類名稱   Reflect.Demo

類名稱   Reflect.Demo

類名稱   Reflect.Demo

【案例3】通過Class實(shí)例化其他類的對象

通過無參構(gòu)造實(shí)例化對象

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package Reflect;
 
class Person{
   
  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 "["+this.name+" "+this.age+"]";
  }
  private String name;
  private int age;
}
 
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    Person per=null;
    try {
      per=(Person)demo.newInstance();
    } catch (InstantiationException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    } catch (IllegalAccessException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    per.setName("Rollen");
    per.setAge(20);
    System.out.println(per);
  }
}

【運(yùn)行結(jié)果】:

[Rollen  20]

但是注意一下,當(dāng)我們把Person中的默認(rèn)的無參構(gòu)造函數(shù)取消的時候,比如自己定義只定義一個有參數(shù)的構(gòu)造函數(shù)之后,會出現(xiàn)錯誤:

比如我定義了一個構(gòu)造函數(shù):

?
1
2
3
4
public Person(String name, int age) {
    this.age=age;
    this.name=name;
  }

然后繼續(xù)運(yùn)行上面的程序,會出現(xiàn):

java.lang.InstantiationException: Reflect.Person

    at java.lang.Class.newInstance0(Class.java:340)

    at java.lang.Class.newInstance(Class.java:308)

    at Reflect.hello.main(hello.java:39)

Exception in thread "main" java.lang.NullPointerException

    at Reflect.hello.main(hello.java:47)

所以大家以后再編寫使用Class實(shí)例化其他類的對象的時候,一定要自己定義無參的構(gòu)造函數(shù)

【案例】通過Class調(diào)用其他類中的構(gòu)造函數(shù) (也可以通過這種方式通過Class創(chuàng)建其他類的對象)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
package Reflect;
 
import java.lang.reflect.Constructor;
 
class Person{
   
  public Person() {
     
  }
  public Person(String name){
    this.name=name;
  }
  public Person(int age){
    this.age=age;
  }
  public Person(String name, int age) {
    this.age=age;
    this.name=name;
  }
  public String getName() {
    return name;
  }
  public int getAge() {
    return age;
  }
  @Override
  public String toString(){
    return "["+this.name+" "+this.age+"]";
  }
  private String name;
  private int age;
}
 
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    Person per1=null;
    Person per2=null;
    Person per3=null;
    Person per4=null;
    //取得全部的構(gòu)造函數(shù)
    Constructor<?> cons[]=demo.getConstructors();
    try{
      per1=(Person)cons[0].newInstance();
      per2=(Person)cons[1].newInstance("Rollen");
      per3=(Person)cons[2].newInstance(20);
      per4=(Person)cons[3].newInstance("Rollen",20);
    }catch(Exception e){
      e.printStackTrace();
    }
    System.out.println(per1);
    System.out.println(per2);
    System.out.println(per3);
    System.out.println(per4);
  }
}

【運(yùn)行結(jié)果】:

[null  0]

[Rollen  0]

[null  20]

[Rollen  20]

【案例】

返回一個類實(shí)現(xiàn)的接口:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
package Reflect;
 
interface China{
  public static final String name="Rollen";
  public static int age=20;
  public void sayChina();
  public void sayHello(String name, int age);
}
 
class Person implements China{
  public Person() {
     
  }
  public Person(String sex){
    this.sex=sex;
  }
  public String getSex() {
    return sex;
  }
  public void setSex(String sex) {
    this.sex = sex;
  }
  @Override
  public void sayChina(){
    System.out.println("hello ,china");
  }
  @Override
  public void sayHello(String name, int age){
    System.out.println(name+" "+age);
  }
  private String sex;
}
 
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    //保存所有的接口
    Class<?> intes[]=demo.getInterfaces();
    for (int i = 0; i < intes.length; i++) {
      System.out.println("實(shí)現(xiàn)的接口  "+intes[i].getName());
    }
  }
}

【運(yùn)行結(jié)果】:

實(shí)現(xiàn)的接口   Reflect.China

(注意,以下幾個例子,都會用到這個例子的Person類,所以為節(jié)省篇幅,此處不再粘貼Person的代碼部分,只粘貼主類hello的代碼)

【案例】:取得其他類中的父類

?
1
2
3
4
5
6
7
8
9
10
11
12
13
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    //取得父類
    Class<?> temp=demo.getSuperclass();
    System.out.println("繼承的父類為:  "+temp.getName());
  }
}

【運(yùn)行結(jié)果】

繼承的父類為:   java.lang.Object

【案例】:獲得其他類中的全部構(gòu)造函數(shù)

這個例子需要在程序開頭添加import java.lang.reflect.*;

然后將主類編寫為:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    Constructor<?>cons[]=demo.getConstructors();
    for (int i = 0; i < cons.length; i++) {
      System.out.println("構(gòu)造方法: "+cons[i]);
    }
  }
}

【運(yùn)行結(jié)果】:

構(gòu)造方法:  public Reflect.Person()

構(gòu)造方法:  public Reflect.Person(java.lang.String)

但是細(xì)心的讀者會發(fā)現(xiàn),上面的構(gòu)造函數(shù)沒有public 或者private這一類的修飾符

下面這個例子我們就來獲取修飾符

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    Constructor<?>cons[]=demo.getConstructors();
    for (int i = 0; i < cons.length; i++) {
      Class<?> p[]=cons[i].getParameterTypes();
      System.out.print("構(gòu)造方法: ");
      int mo=cons[i].getModifiers();
      System.out.print(Modifier.toString(mo)+" ");
      System.out.print(cons[i].getName());
      System.out.print("(");
      for(int j=0;j<p.length;++j){
        System.out.print(p[j].getName()+" arg"+i);
        if(j<p.length-1){
          System.out.print(",");
        }
      }
      System.out.println("){}");
    }
  }
}

【運(yùn)行結(jié)果】:

構(gòu)造方法:  public Reflect.Person(){}

構(gòu)造方法:  public Reflect.Person(java.lang.String arg1){}

有時候一個方法可能還有異常,呵呵。下面看看:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
class hello{
  public static void main(String[] args) {
    Class<?> demo=null;
    try{
      demo=Class.forName("Reflect.Person");
    }catch (Exception e) {
      e.printStackTrace();
    }
    Method method[]=demo.getMethods();
    for(int i=0;i<method.length;++i){
      Class<?> returnType=method[i].getReturnType();
      Class<?> para[]=method[i].getParameterTypes();
      int temp=method[i].getModifiers();
      System.out.print(Modifier.toString(temp)+" ");
      System.out.print(returnType.getName()+" ");
      System.out.print(method[i].getName()+" ");
      System.out.print("(");
      for(int j=0;j<para.length;++j){
        System.out.print(para[j].getName()+" "+"arg"+j);
        if(j<para.length-1){
          System.out.print(",");
        }
      }
      Class<?> exce[]=method[i].getExceptionTypes();
      if(exce.length>0){
        System.out.print(") throws ");
        for(int k=0;k<exce.length;++k){
          System.out.print(exce[k].getName()+" ");
          if(k<exce.length-1){
            System.out.print(",");
          }
        }
      }else{
        System.out.print(")");
      }
      System.out.println();
    }
  }
}

【運(yùn)行結(jié)果】:

public java.lang.String  getSex ()

public void  setSex (java.lang.String arg0)

public void  sayChina ()

public void  sayHello (java.lang.String arg0,int arg1)

public final native void  wait (long arg0) throws java.lang.InterruptedException

public final void  wait () throws java.lang.InterruptedException

public final void  wait (long arg0,int arg1) throws java.lang.InterruptedException

public boolean  equals (java.lang.Object arg0)

public java.lang.String  toString ()

public native int  hashCode ()

public final native java.lang.Class  getClass ()

public final native void  notify ()

public final native void  notifyAll ()

【案例】接下來讓我們?nèi)〉闷渌惖娜繉傩园桑詈笪抑v這些整理在一起,也就是通過class取得一個類的全部框架

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
class hello {
  public static void main(String[] args) {
    Class<?> demo = null;
    try {
      demo = Class.forName("Reflect.Person");
    } catch (Exception e) {
      e.printStackTrace();
    }
    System.out.println("===============本類屬性========================");
    // 取得本類的全部屬性
    Field[] field = demo.getDeclaredFields();
    for (int i = 0; i < field.length; i++) {
      // 權(quán)限修飾符
      int mo = field[i].getModifiers();
      String priv = Modifier.toString(mo);
      // 屬性類型
      Class<?> type = field[i].getType();
      System.out.println(priv + " " + type.getName() + " "
          + field[i].getName() + ";");
    }
    System.out.println("===============實(shí)現(xiàn)的接口或者父類的屬性========================");
    // 取得實(shí)現(xiàn)的接口或者父類的屬性
    Field[] filed1 = demo.getFields();
    for (int j = 0; j < filed1.length; j++) {
      // 權(quán)限修飾符
      int mo = filed1[j].getModifiers();
      String priv = Modifier.toString(mo);
      // 屬性類型
      Class<?> type = filed1[j].getType();
      System.out.println(priv + " " + type.getName() + " "
          + filed1[j].getName() + ";");
    }
  }
}

【運(yùn)行結(jié)果】:

===============本類屬性========================

private java.lang.String sex;

===============實(shí)現(xiàn)的接口或者父類的屬性========================

public static final java.lang.String name;

public static final int age;

【案例】其實(shí)還可以通過反射調(diào)用其他類中的方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class hello {
  public static void main(String[] args) {
    Class<?> demo = null;
    try {
      demo = Class.forName("Reflect.Person");
    } catch (Exception e) {
      e.printStackTrace();
    }
    try{
      //調(diào)用Person類中的sayChina方法
      Method method=demo.getMethod("sayChina");
      method.invoke(demo.newInstance());
      //調(diào)用Person的sayHello方法
      method=demo.getMethod("sayHello", String.class,int.class);
      method.invoke(demo.newInstance(),"Rollen",20);
       
    }catch (Exception e) {
      e.printStackTrace();
    }
  }
}

  【運(yùn)行結(jié)果】:

hello ,china

Rollen  20

【案例】調(diào)用其他類的set和get方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
class hello {
  public static void main(String[] args) {
    Class<?> demo = null;
    Object obj=null;
    try {
      demo = Class.forName("Reflect.Person");
    } catch (Exception e) {
      e.printStackTrace();
    }
    try{
     obj=demo.newInstance();
    }catch (Exception e) {
      e.printStackTrace();
    }
    setter(obj,"Sex","男",String.class);
    getter(obj,"Sex");
  }
 
  /**
   * @param obj
   *      操作的對象
   * @param att
   *      操作的屬性
   * */
  public static void getter(Object obj, String att) {
    try {
      Method method = obj.getClass().getMethod("get" + att);
      System.out.println(method.invoke(obj));
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
 
  /**
   * @param obj
   *      操作的對象
   * @param att
   *      操作的屬性
   * @param value
   *      設(shè)置的值
   * @param type
   *      參數(shù)的屬性
   * */
  public static void setter(Object obj, String att, Object value,
      Class<?> type) {
    try {
      Method method = obj.getClass().getMethod("set" + att, type);
      method.invoke(obj, value);
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}// end class

【運(yùn)行結(jié)果】:

 【案例】通過反射操作屬性

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class hello {
  public static void main(String[] args) throws Exception {
    Class<?> demo = null;
    Object obj = null;
 
    demo = Class.forName("Reflect.Person");
    obj = demo.newInstance();
 
    Field field = demo.getDeclaredField("sex");
    field.setAccessible(true);
    field.set(obj, "男");
    System.out.println(field.get(obj));
  }
}// end class

 

【案例】通過反射取得并修改數(shù)組的信息:

?
1
2
3
4
5
6
7
8
9
10
11
12
import java.lang.reflect.*;
class hello{
  public static void main(String[] args) {
    int[] temp={1,2,3,4,5};
    Class<?>demo=temp.getClass().getComponentType();
    System.out.println("數(shù)組類型: "+demo.getName());
    System.out.println("數(shù)組長度 "+Array.getLength(temp));
    System.out.println("數(shù)組的第一個元素: "+Array.get(temp, 0));
    Array.set(temp, 0, 100);
    System.out.println("修改之后數(shù)組第一個元素為: "+Array.get(temp, 0));
  }
}

【運(yùn)行結(jié)果】:

數(shù)組類型: int

數(shù)組長度  5

數(shù)組的第一個元素: 1

修改之后數(shù)組第一個元素為: 100

【案例】通過反射修改數(shù)組大小

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class hello{
  public static void main(String[] args) {
    int[] temp={1,2,3,4,5,6,7,8,9};
    int[] newTemp=(int[])arrayInc(temp,15);
    print(newTemp);
    System.out.println("=====================");
    String[] atr={"a","b","c"};
    String[] str1=(String[])arrayInc(atr,8);
    print(str1);
  }
   
  /**
   * 修改數(shù)組大小
   * */
  public static Object arrayInc(Object obj,int len){
    Class<?>arr=obj.getClass().getComponentType();
    Object newArr=Array.newInstance(arr, len);
    int co=Array.getLength(obj);
    System.arraycopy(obj, 0, newArr, 0, co);
    return newArr;
  }
  /**
   * 打印
   * */
  public static void print(Object obj){
    Class<?>c=obj.getClass();
    if(!c.isArray()){
      return;
    }
    System.out.println("數(shù)組長度為: "+Array.getLength(obj));
    for (int i = 0; i < Array.getLength(obj); i++) {
      System.out.print(Array.get(obj, i)+" ");
    }
  }
}

【運(yùn)行結(jié)果】:

數(shù)組長度為: 15

1 2 3 4 5 6 7 8 9 0 0 0 0 0 0 =====================

數(shù)組長度為: 8

a b c null null null null null

動態(tài)代理

【案例】首先來看看如何獲得類加載器:

?
1
2
3
4
5
6
7
8
9
class test{
   
}
class hello{
  public static void main(String[] args) {
    test t=new test();
    System.out.println("類加載器 "+t.getClass().getClassLoader().getClass().getName());
  }
}

【程序輸出】:

類加載器  sun.misc.Launcher$AppClassLoader

其實(shí)在java中有三種類類加載器。

1)Bootstrap ClassLoader 此加載器采用c++編寫,一般開發(fā)中很少見。

2)Extension ClassLoader 用來進(jìn)行擴(kuò)展類的加載,一般對應(yīng)的是jre\lib\ext目錄中的類

3)AppClassLoader 加載classpath指定的類,是最常用的加載器。同時也是java中默認(rèn)的加載器。

如果想要完成動態(tài)代理,首先需要定義一個InvocationHandler接口的子類,已完成代理的具體操作。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
package Reflect;
import java.lang.reflect.*;
 
//定義項目接口
interface Subject {
  public String say(String name, int age);
}
 
// 定義真實(shí)項目
class RealSubject implements Subject {
  @Override
  public String say(String name, int age) {
    return name + " " + age;
  }
}
 
class MyInvocationHandler implements InvocationHandler {
  private Object obj = null;
 
  public Object bind(Object obj) {
    this.obj = 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.obj, args);
    return temp;
  }
}
 
class hello {
  public static void main(String[] args) {
    MyInvocationHandler demo = new MyInvocationHandler();
    Subject sub = (Subject) demo.bind(new RealSubject());
    String info = sub.say("Rollen", 20);
    System.out.println(info);
  }
}

【運(yùn)行結(jié)果】:

Rollen  20

類的生命周期

在一個類編譯完成之后,下一步就需要開始使用類,如果要使用一個類,肯定離不開JVM。在程序執(zhí)行中JVM通過裝載,鏈接,初始化這3個步驟完成。

類的裝載是通過類加載器完成的,加載器將.class文件的二進(jìn)制文件裝入JVM的方法區(qū),并且在堆區(qū)創(chuàng)建描述這個類的java.lang.Class對象。用來封裝數(shù)據(jù)。 但是同一個類只會被類裝載器裝載以前

鏈接就是把二進(jìn)制數(shù)據(jù)組裝為可以運(yùn)行的狀態(tài)。

鏈接分為校驗(yàn),準(zhǔn)備,解析這3個階段

校驗(yàn)一般用來確認(rèn)此二進(jìn)制文件是否適合當(dāng)前的JVM(版本),

準(zhǔn)備就是為靜態(tài)成員分配內(nèi)存空間,。并設(shè)置默認(rèn)值

解析指的是轉(zhuǎn)換常量池中的代碼作為直接引用的過程,直到所有的符號引用都可以被運(yùn)行程序使用(建立完整的對應(yīng)關(guān)系)

完成之后,類型也就完成了初始化,初始化之后類的對象就可以正常使用了,直到一個對象不再使用之后,將被垃圾回收。釋放空間。

當(dāng)沒有任何引用指向Class對象時就會被卸載,結(jié)束類的生命周期

將反射用于工廠模式

先來看看,如果不用反射的時候,的工廠模式吧:

http://www.cnblogs.com/rollenholt/archive/2011/08/18/2144851.html

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/**
 * @author Rollen-Holt 設(shè)計模式之 工廠模式
 */
 
interface fruit{
  public abstract void eat();
}
 
class Apple implements fruit{
  public void eat(){
    System.out.println("Apple");
  }
}
 
class Orange implements fruit{
  public void eat(){
    System.out.println("Orange");
  }
}
 
// 構(gòu)造工廠類
// 也就是說以后如果我們在添加其他的實(shí)例的時候只需要修改工廠類就行了
class Factory{
  public static fruit getInstance(String fruitName){
    fruit f=null;
    if("Apple".equals(fruitName)){
      f=new Apple();
    }
    if("Orange".equals(fruitName)){
      f=new Orange();
    }
    return f;
  }
}
class hello{
  public static void main(String[] a){
    fruit f=Factory.getInstance("Orange");
    f.eat();
  }
 
}

這樣,當(dāng)我們在添加一個子類的時候,就需要修改工廠類了。如果我們添加太多的子類的時候,改的就會很多。

現(xiàn)在我們看看利用反射機(jī)制:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
package Reflect;
 
interface fruit{
  public abstract void eat();
}
 
class Apple implements fruit{
  public void eat(){
    System.out.println("Apple");
  }
}
 
class Orange implements fruit{
  public void eat(){
    System.out.println("Orange");
  }
}
 
class Factory{
  public static fruit getInstance(String ClassName){
    fruit f=null;
    try{
      f=(fruit)Class.forName(ClassName).newInstance();
    }catch (Exception e) {
      e.printStackTrace();
    }
    return f;
  }
}
class hello{
  public static void main(String[] a){
    fruit f=Factory.getInstance("Reflect.Apple");
    if(f!=null){
      f.eat();
    }
  }
}

現(xiàn)在就算我們添加任意多個子類的時候,工廠類就不需要修改。

上面的愛嗎雖然可以通過反射取得接口的實(shí)例,但是需要傳入完整的包和類名。而且用戶也無法知道一個接口有多少個可以使用的子類,所以我們通過屬性文件的形式配置所需要的子類。

下面我們來看看: 結(jié)合屬性文件的工廠模式

首先創(chuàng)建一個fruit.properties的資源文件,

內(nèi)容為:

apple=Reflect.Apple
orange=Reflect.Orange

 然后編寫主類代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
package Reflect;
 
import java.io.*;
import java.util.*;
 
interface fruit{
  public abstract void eat();
}
 
class Apple implements fruit{
  public void eat(){
    System.out.println("Apple");
  }
}
 
class Orange implements fruit{
  public void eat(){
    System.out.println("Orange");
  }
}
 
//操作屬性文件類
class init{
  public static Properties getPro() throws FileNotFoundException, IOException{
    Properties pro=new Properties();
    File f=new File("fruit.properties");
    if(f.exists()){
      pro.load(new FileInputStream(f));
    }else{
      pro.setProperty("apple", "Reflect.Apple");
      pro.setProperty("orange", "Reflect.Orange");
      pro.store(new FileOutputStream(f), "FRUIT CLASS");
    }
    return pro;
  }
}
 
class Factory{
  public static fruit getInstance(String ClassName){
    fruit f=null;
    try{
      f=(fruit)Class.forName(ClassName).newInstance();
    }catch (Exception e) {
      e.printStackTrace();
    }
    return f;
  }
}
class hello{
  public static void main(String[] a) throws FileNotFoundException, IOException{
    Properties pro=init.getPro();
    fruit f=Factory.getInstance(pro.getProperty("apple"));
    if(f!=null){
      f.eat();
    }
  }
}

【運(yùn)行結(jié)果】:Apple

以上就是對Java 反射機(jī)制的詳解,后續(xù)繼續(xù)補(bǔ)充相關(guān)資料,謝謝大家對本站的支持!

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国语刺激对白勾搭视频在线观看 | 国产女主播福利在线 | 91精品国产高清久久久久 | 国产自一区 | 高清日韩在线 | 暖暖日本高清 | 偷拍自拍校园春色 | 成人精品mv视频在线观看 | 亚洲成年男人的天堂网 | 我的绝色岳每雯雯 | 成人啪啪漫画羞羞漫画www网站 | 4444亚洲国产成人精品 | 午夜久久久久久亚洲国产精品 | 亚洲激情网站 | japanese在线看 | 操美女骚b| 韩国甜性涩爱在线播放 | a一级一级 | 成人午夜毛片 | 精品视频一区在线观看 | 亚州一区二区 | 欧美黑人性猛交╳xx╳动态图 | 精品国产剧情在线观看 | 日韩欧美一区二区三区四区 | 大学生情侣在线 | 日本中文字幕一区二区三区不卡 | 亚洲视频久久 | 果冻传媒在线播放观看228集 | 动漫美女人物被黄漫小说 | 波多 在线播放 | 久久久久夜 | 69热精品视频在线看影院 | 久久不射网 | 久久受www免费人成_看片中文 | 国产精品香蕉夜间视频免费播放 | 亚洲精品一 | 扒开放荡老师裙子猛烈的进入 | 色综合视频一区二区三区 | 免费国产一级 | 日本一道一区二区免费看 | 九九精品免视看国产成人 |