专栏首页python3java反射详解(3)

java反射详解(3)

动态代理

【案例】首先来看看如何获得类加载器:

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

其实在java中有三种类类加载器。

1)Bootstrap ClassLoader 此加载器采用c++编写,一般开发中很少见。

2)Extension ClassLoader 用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类

3)AppClassLoader 加载classpath指定的类,是最常用的加载器。同时也是java中默认的加载器。

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

package Reflect; 
import java.lang.reflect.*; 
  
//定义项目接口 
interface Subject { 
    public String say(String name, int age); 
} 
  
// 定义真实项目 
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); 
    } 
} 

【运行结果】:

Rollen  20

类的生命周期

在一个类编译完成之后,下一步就需要开始使用类,如果要使用一个类,肯定离不开JVM。在程序执行中JVM通过装载,链接,初始化这3个步骤完成。

类的装载是通过类加载器完成的,加载器将.class文件的二进制文件装入JVM的方法区,并且在堆区创建描述这个类的java.lang.Class对象。用来封装数据。 但是同一个类只会被类装载器装载以前

链接就是把二进制数据组装为可以运行的状态。

链接分为校验,准备,解析这3个阶段

校验一般用来确认此二进制文件是否适合当前的JVM(版本),

准备就是为静态成员分配内存空间,。并设置默认值

解析指的是转换常量池中的代码作为直接引用的过程,直到所有的符号引用都可以被运行程序使用(建立完整的对应关系)

完成之后,类型也就完成了初始化,初始化之后类的对象就可以正常使用了,直到一个对象不再使用之后,将被垃圾回收。释放空间。

当没有任何引用指向Class对象时就会被卸载,结束类的生命周期

将反射用于工厂模式

先来看看,如果不用反射的时候,的工厂模式吧:

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

/** 
 * @author Rollen-Holt 设计模式之 工厂模式 
 */ 
  
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 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(); 
    } 
  
} 

这样,当我们在添加一个子类的时候,就需要修改工厂类了。如果我们添加太多的子类的时候,改的就会很多。

现在我们看看利用反射机制:

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(); 
        } 
    } 
} 

现在就算我们添加任意多个子类的时候,工厂类就不需要修改。

上面的例子虽然可以通过反射取得接口的实例,但是需要传入完整的包和类名。而且用户也无法知道一个接口有多少个可以使用的子类,所以我们通过属性文件的形式配置所需要的子类。

下面我们来看看: 结合属性文件的工厂模式

首先创建一个fruit.properties的资源文件,

内容为:

apple=Reflect.Apple 
orange=Reflect.Orange 

然后编写主类代码:

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(); 
        } 
    } 
} 

【运行结果】:Apple

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • python flask 消息闪现

    消息闪现参考文档:http://www.pythondoc.com/flask/patterns/flashing.html

    py3study
  • python记录day_20 多继承

    在python中,支持多继承,一个类可以拥有多个父类。但是多继承中, 存在着这样一个问题,当两个父类中出现了重名方法的时候该怎么办呢? 这时就涉及到如何查找父类...

    py3study
  • Java循环练习:求1+(1*2)+(1

    py3study
  • Java描述设计模式(02):简单工厂模式

    1)、OCP原则:软件实体,如类、模块和函数,应当对扩展开放,但对修改关闭。 2)、违反设计模式的OCP原则,新增食品类不方便扩展,代码改动较大。

    知了一笑
  • 【java开发系列】—— 嵌套类与内部类

    嵌套类与内部类在java中使用的很广泛,为了避免难以理解,这里用了几个小例子,说明它的用法。   嵌套类与内部类的结构如下图 ?   静态嵌套类     ...

    用户1154259
  • 九、委派模式与模板模式详解

    委派模式(Delegate Pattern ) 又叫委托模式, 是一种面向对象的设计模式, 允许对象组合实现与 继承相同的代码重用。它的基本作用就是负责任务的...

    编程之心
  • 锻造正则神兵之Java源码分析器-V0.01

    张风捷特烈
  • 在Java中12个常见的语法糖!

    本文从 Java 编译原理角度,深入字节码及 class 文件,抽丝剥茧,了解 Java 中的语法糖原理及用法,帮助大家在学会如何使用 Java 语法糖的同时,...

    Java3y
  • 不懂这12个语法糖,别说你会Java!

    本文从 Java 编译原理角度,深入字节码及 class 文件,抽丝剥茧,了解 Java 中的语法糖原理及用法,帮助大家在学会如何使用 Java 语法糖的同时,...

    纯洁的微笑
  • 不了解这 12 个语法糖,别说你会 Java!

    本文从 Java 编译原理角度,深入字节码及 class 文件,抽丝剥茧,了解 Java 中的语法糖原理及用法,帮助大家在学会如何使用 Java 语法糖的同时,...

    Java技术江湖

扫码关注云+社区

领取腾讯云代金券