JAVA设计模式理解与总结(下)代理模式&适配器模式&观察者模式

码了好几年代码的打字机器我,对于设计模式这个词,肯定是一点也不陌生,但是对于设计模式的理解,因为日常开发中,增删改查较多,使用设计模式思想来优化代码的机会就很少。也不乏在翻阅源码的时候,叹服于别人优秀高效的设计。所有今天抽出点时间,对设计模式做个归纳、记录,以便日后读到优秀的源码,可以自信的说,这**不就是那啥吗,我也会写~~~

设计模式

设计模式(Design Pattern)是前辈们对代码开发经验的总结,是解决特定问题的一系列套路。它不是语法规定,而是一套用来提高代码可复用性、可维护性、可读性、稳健性以及安全性的解决方案

代理模式

一个类代表另一个类的功能,在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。

代理模式在我们的生活中也很常见,例如程序员的女友想买一个法国产的某包包,她不需要亲自去法国某个卖驴的专卖店去买,而是可以在代购手中购买,并获得一些小礼物,折扣优惠等等。

静态代理

在代码中,代理就是在使用者和服务提供者中间的服务,提供一些在提供的服务前后处理一些事物的功能。

卖东西人的认证(共同实现的接口)

public interface Seller {
    void sell();
}

法国某经销商:

public class LvFactory implements Seller{
    @Override
    public void sell() {
        System.out.println("LV厂家卖包包");
    }
}

你女朋友朋友圈某个不知名好友:

class Purchase implements Seller{
    private Seller seller;

    public Purchase(Seller factory) {
        this.seller = factory;
    }

    @Override
    public void sell() {
        System.out.println("坐飞机去法国");
        seller.sell();
        System.out.println("坐飞机回国");
    }
}

你女朋友:

public static void main(String[] args) {
    Seller lv = new LvFactory();
    new Purchase(lv).sell();
}

你:

信用卡 boom~~~~~~~~~~~~~~~~~

坐飞机去法国

LV厂家卖包包

坐飞机回国

动态代理

动态代理有别于静态代理,是根据代理的对象,动态创建代理类。这样,就可以避免静态代理中代理类接口过多的问题。动态代理是实现方式,是通过反射来实现的,借助Java自带的java.lang.reflect.Proxy,通过固定的规则生成。

举个🌰,女朋友朋友圈这个代购,代购包,也代购手表,那我找他一起买一块手表呢?

手表接口和天梭类:

public interface Watch {
    void sellWatch();
}
public class TissotWatch implements Watch {
    @Override
    public void sellWatch() {
        System.out.println("卖天梭手表");
    }
}

使用 InvocationHandler 定义代理类:

public class PurchaseProxy implements InvocationHandler {
    private Object object;
    public PurchaseProxy(Object object) {
        this.object = object;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("坐飞机出国");
        Object result = method.invoke(object, args);
        System.out.println("坐飞机回国");
        return result;
    }
}

在朋友圈买包买表:

public static void main(String[] args) {
    //卖包
    Seller lvseller = new LvFactory();
    PurchaseProxy lvproxy = new PurchaseProxy(lvseller);
    ((Seller) Proxy.newProxyInstance(lvseller.getClass().getClassLoader(), new Class[] { Seller.class }, lvproxy)).sell();
    //卖表
    Watch tissot = new TissotWatch();
    PurchaseProxy tissotproxy = new PurchaseProxy(tissot);
    ((Watch) Proxy.newProxyInstance(tissot.getClass().getClassLoader(), new Class[] { Watch.class }, tissotproxy)).sellWatch();
}

坐飞机出国

LV厂家卖包包

坐飞机回国

坐飞机出国

卖天梭手表

坐飞机回国

动态代理详解

InvocationHandler

InvocationHandler 是一个接口,当代理方法被调用时,会调用起相对应的实现类的 invoke 方法。

@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    return null;
}

三个参数:

  • proxy 代理对象
  • method 代理对象调用的方法
  • args 调用的方法中的参数
Proxy

Java自带的java.lang.reflect.Proxy

public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)
        throws IllegalArgumentException
  • loader 类加载器
  • interfaces 代码要用来代理的接口
  • h 一个 InvocationHandler 对象

适配器模式

我们日常使用的适配器是一个接口转换器,它可以是一个独立的硬件接口设备,允许硬件或电子接口与其它硬件或电子接口相连,也可以是信息接口

在代码中,我想实现一个接口中的部分功能,或者想用一个类来冒充另一个类来实现它的功能,就需要适配器模式了。

适配器模式使接口不兼容的那些类可以一起工作,其别名为包装器

类适配器模式

在类适配器模式中,适配器与适配者之间是继承,适配着实现通用接口

举个🌰,我国标准电压是220v,手机充电器也是220v,日本标准电压是110v,女朋友去日本玩,想给220v的手机用110v电压充电,就需要一个适配器了。

日本插座:

public class JapanAC {
    private final static int v = 110;
    public void jcharge() {
        System.out.printf("使用%dv电压充电\n", v);
    }
}

充电通用接口:

public interface Charger {
    void charge();
}

买一个 110v升220变压器:

public class Adaptee extends JapanAC implements Charger {
    @Override
    public void charge() {
        System.out.println("使用适配器110v转220v");
        super.jcharge();
        System.out.println("充电完成");
    }
}

女朋友使用适配器充电:

public static void main(String[] args) {
    Charger charger = new Adaptee();
    charger.charge();
}

使用适配器110v转220v

使用110v电压充电

充电完成

对象适配器模式

把一只猴子伪装成人:

有那么一种猴子:

public interface Monkey {
    //吃香蕉
    void eatBanana();
    //骑独轮车
    void rideABike();
}
public class Shuang implements Monkey {
    @Override
    public void eatBanana() {
        System.out.println("猴子吃香蕉");
    }
    @Override
    public void rideABike() {
        System.out.println("猴子独轮车"); //猴子不懂人性,只会骑独轮车
    }
}

有那么个人:

public interface Human {
    //吃香蕉
    void eatBanana();
    //开车
    void drive();
}
public class Da implements Human{
    @Override
    public void eatBanana() {
        System.out.println("人吃香蕉");
    }
    @Override
    public void drive() {
        System.out.println("人开车");
    }
}

猴子伪装成人的大型真猴修表演:

public class MonkeyMan implements Human{
    private Monkey monkey;
    public MonkeyMan(Monkey monkey) {
        this.monkey = monkey;
    }
    @Override
    public void eatBanana() {
        monkey.eatBanana();
    }
    @Override
    public void drive() {
        monkey.rideABike();
    }
}

猴子伪装成人:

public static void main(String[] args) {
    Monkey shuang = new Shuang();
    Human human = new MonkeyMan(shuang); //猴子成人
    human.drive();
}

猴子独轮车

观察者模式

当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知依赖它的对象。观察者模式属于行为型模式。

又被称为发布-订阅(Publish/Subscribe)模式

例如微博关注,多个用户关注同一博主,博主更新,会通知到所有用户。

罗老师微博:

public interface Bloger {
    void add(Fan fan);
    void del(Fan fan);
    void notifySms(String sms);
}
public class Luo implements Bloger {
    private List<Fan> fans;
    {
        fans = new ArrayList<>();
    }
    @Override
    public void add(Fan fan) {
        fans.add(fan);
    }
    @Override
    public void del(Fan fan) {
        fans.remove(fan);
    }
    @Override
    public void notifySms(String sms) {
        fans.forEach(fan -> fan.notifySms(sms));
    }
}

粉丝:

public interface Fan {
    void notifySms(String sms);
}
public class Xigua implements Fan {
    private String name;
    public Xigua(String name) {
        this.name = name;
    }
    @Override
    public void notifySms(String sms) {
        System.out.println(name + " | " + sms);
    }
}

罗老师直播买手机 :

public static void main(String[] args) {
    Fan leijun = new Xigua("雷军");
    Fan dongmingzhu = new Xigua("董明珠");
    Fan renzhengfei = new Xigua("任正非");

    Bloger luo = new Luo();
    luo.add(leijun);
    luo.add(dongmingzhu);
    luo.add(renzhengfei);

    luo.notifySms("买锤子,送锤子啦!!!");
}

雷军 | 买锤子,送锤子啦!!!

董明珠 | 买锤子,送锤子啦!!!

任正非 | 买锤子,送锤子啦!!!

实际生产过程中,观察者模式往往用消息中间件来实现

收工





更多好玩好看的内容,欢迎到我的博客交流,共同进步        WaterMin


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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