设计模式-结构型模式

结构型模式

类或者对象按照一定布局组成更大的对象。类结构模式采用继承接口,对象模式采用组合或聚合来组合对象。
结构型模式中除了适配器外,其余都是对象模式

一、代理模式

  • 代理模式在客户端和目标对象之间起到中介和保护作用
  • 优点:中介可以扩展目标对象功能、解耦
  • 缺点:增加代理影响性能,增加复杂度
6proxy.png

抽象主题 Abstract Subject

接口或者抽象类定义真实主题和代理对象实现的方法

interface AbstractSubject{
    void request();
}

真实主题 Concrete subject

实现抽象主题具体接口

class ConcreteSubject implement AbstractSubject{
    
    void request(){
        System.out.println("处理请求");
    }
}

代理类 Proxy

提供真实主题相同的接口,内部包括真实主题的引用,可以访问、控制、扩展真实主题功能。

class Proxy implements AbstractSubject{
    private ConcreteSubject realSub;
    public request(){
        if(null==realSub){
            realSub=new ConcreteSubject();
        }
        preRequest();
        realSub.request();
        posRequest();
    }
    
    private void preRequest(){
        System.out.println("调用前");
    }
    
    private void PosRequest(){
        System.out.println("调用后");
    }
}

调用类

class  TestProxy{
    
    public static void mian(String[] args){
        Proxy pro=new Proxy();
        pro.request();
    }
}

代理模式扩展-动态代理 DynamicProxy

JDK动态代理依赖接口,可以没有实现类

抽象主题
interface AbstractDynamicSubject{
     void request();
}
真实主题
class DynamicSubject implements AbstractDynamicSubject{
    void request(){
        System.out.println("dynamicSubject");
    }
}
执行处理类 InvocationHandler
class MyInvocationHandler implements InvocationHandler{
    @Override
    public Object invok(Object proxy,Method method,Object[] arg) throw ThrowAble{
        System.out.print("testDynamicProxy");
        return null;
    }
}
调用类
public class TestDynamicProxy{
    
    public static void main(String[] args){
        System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");
        AbstractDynamicSubject proxySubject= (AbstractDynamicSubject) Proxy.newProxyInstance(DynamicSubject.class.getClassLoad(),DynamicSubject
                .class.getInterfaces(),new MyInvocationHandler());
    }
}

二、适配器模式

将类的接口转换为客户希望的另一个接口,分为类结构模型和对象结构模型,类需要了解组件内部细节较少使用
适用老的接口使用,第三方接口匹配

7adapter.png

目标 target

interface Target{
    void request();
}

(被)适配者 Adaptee

class Adaptee {

void adapteeRequest(){
    
   System.out.println("adaptee");
}
    
}

适配器(类) ClassAdapter

class ClassAdapter extends Adaptee implements Target{
    
    void request(){
        adapteeRequest();
    }
}

对象适配器 ObjectAdapter

class ObjectAdapter implements Target{
    private Adaptee ad;
    
    Public ObjectAdapter(Adaptee cad){
        ad=cad;
    }
    
    void request(){
        ad.adapteeRequest();
    }
}

调用类

public class TestAdapter{
    public void static main(String[] args){
        Target oba=new ObjectAdapter(new ClassAdapter());
        oba.request();
    }
}

扩展,双向适配器

interface Adpteeinf {
    void adpteeRequest();
}
class TwoObjectAdapter implements Adpteeinf,Target{
    private Adaptee da;
    private Target tg;
    public TwoObjectAdapter(Adaptee cda){
        da=cda;
    }
    public TwoObjectAdapter(Target tga){
        tg=tga;
    }
    
    void request(){
        da.adapteeRequest();
    }
    
    void adapteeRequest(){
        tg.request();
    }
    
}

三、桥接模式

将抽象和实现分离,使其独立变化。用组合关系代替集成关系来实现,降低了抽象和实现两个可变维度耦合度。
当一个类有两种独立的变化维度,且都需要扩展,例如包有颜色、用途两种维度属性,且都需要扩展变化。

8bridge.png

实现化角色 implementor

interface Implementor {
    void operation();
}

具体实现化角色 ConcreterImplement

class ConcreteImplement implements Implementor{
    
    void operation(){
        System.out.print("concrete");
    }
}

抽象化角色 Abstraction

abstract Abstraction{
    private Implementor imp;
    
    public Abstraction(Implementor imp){
        this.imp=imp;
    }
    
     void operationImp();
}

扩展抽象化角色

class ReAbstraction extends Abstraction{
    public ReAbstraction(Implementor imp){
        super(imp);
    }
    
    public void operationImp(){
         imp.operation();
        System.out.print("Reimp");
    }
}

调用类

public class TestBridge{
    public static void main(String[] args){
        Implementor imp= new ConcreteImplement();
        ReAbstraction reAb=new ReAbstraction(imp); 
        reAb.operationImp;
    }
}

模式扩展

有时桥接模式可与适配器模式联合使用。

四、装饰模式 decorator

不改变现有结构的情况下,动态增加职责功能。最著名的应用java中的I/O 设计
例如,Reader增加缓冲区

BufferedReader in =new BufferedReader(new FileReader("D:\t.text"));
String s=in.readLine();
9decorator.png

抽象构件 Component

定义接口规范需要附加职责的对象

interface Component{
    void read();
}

抽象构件的实现 ConcreteComponent

class ConcreteComponent implements Component{

void read(){
    System.out.println("read");
}
    
}

抽象裝飾 AbstractDecorator

定义需要扩展的功能,由子类集成实现

abstract Decorator implements Component{
    private Component cp;
    
    public Decorator(Component cp){
        this.cp=cp;
    }
    
    void read(){
        cp.read();
    }
    
}

具体抽象装饰类 ConcreteDecorator

class ConcreteDecorator extends Decorator{
    
    public ContreteDecorator(Component cp){
    super(cp);
        
    }
    
    void read(){
        super.read();
        addFn();
    }
    
    private addFn(){
        System.out.print("扩展功能");
    }
}

五、外观模式 facade

为多个子系统提供统一个一致的接口,方便外部访问。
降低了子系统与客户端直接的耦合,是子系统的变化不会影响调用。减少了对象
处理的数目
新增子系统需要修改外观类和客户端,违背了开闭原则。


10facde.png

外观角色 Facade

对子系统提供统一的接口

class Facade{
    private SubSys1 s1=new SubSys1();
    private SubSys2 s2 =new SubSys2();
    void method(){
    s1.method1();
    s2.method2();
        
    }
}

子系统 SubSystem

class SubSys1{
    void method1{
        System.out.println("print1()");
    }
}
class SubSys2{
    void method2{
        System.out.println("print2()");
    }
}

客户端 TestFacade

class TestFacade{
    public static void main(String[] args){
        
          Facade fe=new Facade();
          fe.method();
    }
}

六、享元模式 Flyweight

运用共享的技术来有效支撑,大量颗粒度的对象复用。

11flyweight.png

抽象享元角色 Flyweight

规范要实现的公共接口,非享元通过接口参数传入。

interface Flyweight{
    void setUnFlyweight(UnFlyweight uflyweight);
}

具体享元角色 ConcreteFlyweight

class ConcreteFlyweight implements Flyweight{
    private String key;
    public ConcreteFlyweight(String key){
        this.key=key;
    }
    public void setUnflyweight(Unflyweight uflyweight){
        System.out.println("具体享元被调用");
        System.out.println("非享元信息是"+uflyweight.info());
    }
}

非享元角色 UnsharedFlyweight

class UnshareFlyweight{
    String info;
    public UnshareFlyweight(String info){
        this.info=info;
    }
    public String getInfo(){
        return info;
    }
    pblic void setInfo(String info){
        this.info=info;
    }
}

享元工厂角色 FlyweightFactory

负责创建和管理享元角色。客户请求时返回原有对象或创建新对象。

class FlyweightFactory{
    private HashMap<String,Flyweight> flyweights=new HashMap<>();
    
    public getFlyweight(String key){
        Flyweight flyweight=flyweights.get(key);
        if(null==flyweight){
            flyweight =new ConcreteFlyweight(key);
            flyweights.put(key,flyweight);
        } 
        return flyweight;
    }
}

享元模式扩展

单纯享元模式 、符合享元模式

复合享元模式 CompositeConcreteFlyweight

public class CompositeCOncreteFlyweight implements Flyweight{
    private List<FLyweight> flyweights=new ArraryList<>();
    
    void setFlyweight(UnFlyweight uFlyweight){
        System.out.println(uFLyweight.getInfo());
    }
    
    public addFlyweight(Flyweight flyweight){
        flyweights.add(flyweight);
    }
    
    public removeFlyweight(Flyweight flyweight){
        flyweights.remove(flyweight);
    }
}

七、组合模式 Composite

将对象组合成树状层次结构的模式,客户端可以一致底处理单个对象或组合。分为透明式和安全式

12composite.png

安全式为例:

抽象构件 Component

abstract Component{
    
    void operat();
}

树叶构件角色 Leaf

class Leaf implements Component{
    
    void operat(){
        System.out.print("leaf");
    }
}

树枝构件 Composite

class Commposite implements Component{
    private ArrayList<Component> leafs =new ArrayList<>();
    void operat(){
    for(Component leaf:leafs)
        System.out.print("composite");
    }
    
    public addLeaf(Leaf leaf){
        leafs.add(leaf);
    }
    
    pbulic remove(Leaf leaf){
        leafs.remove(leaf);
    }
    public List<Component> getChild(){
    return leafs;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342