java反射增加装饰模式的普适性

简介: 装饰模式(Decorator Pattern)的定义是:在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能。它是通过创建一个包装对象来包裹真实的对象。就增加功能来说,装饰模式相比于生成子类更为灵活。使用java的动态代理实现装饰模式会具有更强的灵活性、适应性。下面我们就来写一个使用java动态代理来实现装饰模式效果的例子。 定义一个职责的接口: /** * 能力 */

装饰模式(Decorator Pattern)的定义是:在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能它是通过创建一个包装对象来包裹真实的对象。就增加功能来说,装饰模式相比于生成子类更为灵活。使用java的动态代理实现装饰模式会具有更强的灵活性、适应性。下面我们就来写一个使用java动态代理来实现装饰模式效果的例子。

定义一个职责的接口:

/**
 * 能力
 */
interface Feature{
    void ability();
}
职责的两个实现类:

/**
 * 读书的能力
 */
class ReadAbility implements Feature {

    @Override
    public void ability() {
        System.out.println("我会读书、、、");
    }
}

/**
 * 写字的能力
 */
class WriteAbility implements Feature {

    @Override
    public void ability() {
        System.out.println("我会写字。、、、、");
    }
}
定义一个通用的接口,用来扩展实现类的职责:

/**
 * 有什么能力
 */
interface Ability {
    void sayYouAbility();
}
实现通用的接口,来通过反射来模拟装饰模式:

public class ReflectDecorateTest01 implements Ability{
    //用来包装真实对象
    private Ability ability;
    //下面要反射调用的Class对象。
    private Class<? extends Feature> clazz;

    @Override
    public void sayYouAbility() {
        /**
         * 动态代理
         */
        InvocationHandler invocationHandler = new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                Object obj = null;
                if(Modifier.isPublic(method.getModifiers())){
                    obj = method.invoke(clazz.newInstance(),args);
                }
                ability.sayYouAbility();//这个地方很重要,不能漏掉, 实现层层调用的功能。
                return obj;
            }
        };
	//生成动态代理对象
        Feature feature = (Feature) Proxy.newProxyInstance(getClass().getClassLoader(),clazz.getInterfaces(),invocationHandler);
        feature.ability();//代理类的调用方法
    }

    public ReflectDecorateTest01(Ability ability, Class<? extends Feature> clazz) {
        this.ability = ability;
        this.clazz = clazz;
    }

    public ReflectDecorateTest01() {
    }

    public static void main(String[] args){
        Ability ability = new Ability() {
            @Override
            public void sayYouAbility() {
                System.out.println("你有什么能力!");
            }
        };
        ability = new ReflectDecorateTest01(ability,ReadAbility.class);//给实现类增加读的能力
        ability = new ReflectDecorateTest01(ability,WriteAbility.class);//给实现类增加写的能力
        ability.sayYouAbility();
    }
}
这样一个通过反射来动态实现装饰模式的功能。下面多说一点装饰模式的东西。装饰模式有什么特点呢?

1、装饰对象和真实对象有相同的接口。这样调用者就能以和真实对象相同的方式和装饰对象交互。

2、装饰对象包含一个真实对象的引用(即上面例子中的Ability接口)。

3、装饰对象接受所有来调用者的请求,并把这些请求转发给真实的对象。

4、装饰对象可以在调用者的方法以前或以后增加一些附加功能。这样就确保了在运行时,不用修改给定对象的结构就可以在外部增加附加的功能。

那么在什么样的地方使用装饰模式呢?

1、需要动态扩展一个类的功能,或给一个类添加附加职责。

2、需要动态的给一个对象添加功能,这些功能可以再动态的撤销。

3、需要增加由一些基本功能的排列组合而产生的非常大量的功能,从而使继承关系变的不现实。

4、 当不能采用生成子类的方法进行扩充时。一种情况是,可能有大量独立的扩展,为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长。另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类。

下面再写一个不通过反射来实现装饰模式的例子:

/**
 * Created by zkn on 2016/11/14.
 * 测试装饰模式
 */
public class DecoratePatternTest01 implements DecorateInterface{

    private DecorateInterface decorateInterface;

    public DecoratePatternTest01(DecorateInterface decorateInterface) {
        this.decorateInterface = decorateInterface;
    }

    @Override
    public void read(){
        decorateInterface.read();
    }

    public static void main(String[] args){
        DecoratePatternTest01 decoratePatternTest01 = new DecoratePatternTest01(new DecorateTest02(new DecorateTest01()));
        decoratePatternTest01.read();
    }
}

/**
 * 装饰模式的接口
 */
interface DecorateInterface {

    void read();
}
/**
 * 装饰接口的第一个实现类
 */
class DecorateTest01 implements DecorateInterface {

    private DecorateInterface decorateInterface;

    @Override
    public void read() {

        System.out.println("我是第一个实现类");
    }

    public DecorateTest01(DecorateInterface decorateInterface) {
        this.decorateInterface = decorateInterface;
    }

    public DecorateTest01() {
    }
}








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

热门文章

最新文章