7.8-7.15

1 Template Method 模式

Template Method是带有模版功能的模式,组成模版的方法被定义在父类中,由于这些方法是抽象方法,所以只看父类的代码是无法知道这些方法最终会进行何种具体的处理的。

实现上述方法的是子类,但是不管子类具体如何实现,处理的流程都会按照父类所定义的进行

像这样在父类中定义处理流程的框架,在子类中实现具体处理的模式,称为Template Method模式.

1.1 示例程序

[外链图片转存失败(img-HyYBtVdQ-1563093798347)(http://note.youdao.com/yws/res/1737/04F3143D797C4F519A73CA1576CF4DE9)]

package designModel.TemplateModel;

public abstract class AbstractDisplay {
    public abstract void open();

    public abstract void print();

    public abstract void close();

    public final void display(){
        open();
        for(int i = 0; i < 5; i++){
            print();
        }
        close();
    }
}
package designModel.TemplateModel;

public class StringDisplay extends AbstractDisplay {

    private String string;

    private int width;

    public StringDisplay(String string) {
        this.string = string;
        this.width = string.getBytes().length;
    }

    @Override
    public void open() {
        printLine();
    }

    @Override
    public void print() {
        System.out.println("|" + string + "|");
    }

    @Override
    public void close() {
        printLine();
    }

    public void printLine(){
        System.out.print("+");
        for(int i = 0; i < width; i++){
            System.out.print("-");
        }
        System.out.println("+");
    }
}
package designModel.TemplateModel;

public class CharDisplay extends AbstractDisplay {

    private char ch;

    public CharDisplay(char ch) {
        this.ch = ch;
    }

    @Override
    public void open() {
        System.out.print("<<");
    }

    @Override
    public void print() {
        System.out.print(ch);
    }

    @Override
    public void close() {
        System.out.println(">>");
    }
}
package designModel.TemplateModel;

public class Main {
    public static void main(String[] args){
        AbstractDisplay d1 = new CharDisplay('H');
        AbstractDisplay d2 = new StringDisplay("Hello, World.");
        AbstractDisplay d3 = new StringDisplay("你好,世界");
        d1.display();
        d2.display();
        d3.display();
    }
}

输出如下结果:

<<HHHHH>>
+-------------+
|Hello, World.|
|Hello, World.|
|Hello, World.|
|Hello, World.|
|Hello, World.|
+-------------+
+---------------+
|你好,世界|
|你好,世界|
|你好,世界|
|你好,世界|
|你好,世界|
+---------------+

2 Factory Method 模式

2.1 示例程序

这段示例程序的作用是制作身份证,其中有5个类。Product和Factory类属于frameword包,这两个类组成了生成实例的框架。
IDCard类和IDCardFactory类负责实际的加工处理,她们属于idcard包
Main用于测试

2.1.1 Product类

frameword包中的Product类是用来表示产品的类,在该类中仅声明了use抽象方法。use方法的实现则被交给了Product类的子类.

package designModel.FactoryMethod.framework;

public abstract class Product {
    public abstract void use();
}

2.1.2 Factory类

createProduct的抽象方法和用于“注册产品”的registerProduct抽象方法。

在这个框架中,我们定义了工厂是用来“调用create方法生成Product实例”的,而create方法的实现是先调用createProduct生成产品,接着用registerProduct注册产品.

package designModel.FactoryMethod.framework;

public abstract class Factory {
    public final Product create(String owner){
        Product p = createProduct(owner);
        registerProduct(p);
        return p;
    }
    protected abstract Product createProduct(String owner);
    protected abstract void registerProduct(Product product);
}

2.1.3 IDCard类

package designModel.FactoryMethod.idcard;

import designModel.FactoryMethod.framework.Product;

public class IDCard extends Product {

    private String owner;

    public IDCard(String owner) {
        System.out.println("制作" + owner + "的ID卡");
        this.owner = owner;
    }

    @Override
    public void use() {
        System.out.println("使用" + owner + "的ID卡");
    }

    public String getOwner() {
        return owner;
    }
}

2.1.4 IDCardFactory类

package designModel.FactoryMethod.idcard;

import designModel.FactoryMethod.framework.Factory;
import designModel.FactoryMethod.framework.Product;

import java.util.ArrayList;
import java.util.List;

public class IDCardFactory extends Factory {

    private List owners = new ArrayList();

    @Override
    protected Product createProduct(String owner) {
        return new IDCard(owner);
    }

    @Override
    protected void registerProduct(Product product) {
        owners.add(((IDCard)product).getOwner());
    }

    public List getOwners() {
        return owners;
    }
}

2.1.5 Main类

package designModel.FactoryMethod;

import designModel.FactoryMethod.framework.Factory;
import designModel.FactoryMethod.framework.Product;
import designModel.FactoryMethod.idcard.IDCardFactory;

public class Main {
    public static void main(String[] args){
        Factory factory = new IDCardFactory();
        Product card1 = factory.create("小明");
        Product card2 = factory.create("小红");
        Product card3 = factory.create("小刚");
        card1.use();
        card2.use();
        card3.use();
    }
}

运行结果:

制作小明的ID卡
制作小红的ID卡
制作小刚的ID卡
使用小明的ID卡
使用小红的ID卡
使用小刚的ID卡

2.2 Factory Method登场角色

[外链图片转存失败(img-KyP1ouYq-1563093798348)(http://note.youdao.com/yws/res/1783/6CC513C023C94148BC790F43575EF053)]

  • Product(产品)

    它定义了在Factory Method模式中生成的那些实例所持有的API,但具体的处理由子类ConcreateProduct角色决定

  • Creator(创建者)

    负责生成Product角色的抽象类,但具体的处理则由子类ConcreteCreator角色决定。Creator角色对于实际负责生成实例的ConcreteCreator角色一无所知,它唯一知道的就是,只要调用Product角色和生成实例的方法,就可以生成Product的实例。在示例程序中,createProduct方法是用于生成实例的方法。不用new关键字来生成实例,而是调用生成实例的专用方法来生成实例,这样就可以防止父类与其他具体类耦合

  • ConcreteProduct(具体的产品)

  • ConcreteCreator(具体的加工者)

3 Abstract Facotry模式

抽象工厂的工作是将“抽象零件”组装成为“抽象的产品”,我们并不关心零件的具体实现,而是只关心接口,我们仅仅使用该接口将零件组装成为产品。

3.1 示例程序

<html>
<head>
    <title>LinkPage</title>
</head>
<body>
<h1>
    LinkPage
</h1>
<ul>
    <li>
        <ul>
            <li>
                <a href="http://www.people.com.cn">人民日报</a>
            </li>
            <li>
                <a href="http://www.gmw.cn">光明网</a>
            </li>
        </ul>
    </li>
    <ul>
        <li>
            Yahoo!
            <ul>
                <li><a href="http://www.yahoo.com/">Yahoo!</a></li>
                <li><a href="http://www.yahoo.co.jp/">Yahoo!Japan</a></li>
            </ul>
        </li>
        <li><a href="http://www.excite.com/">Excite</a></li>
        <li><a href="http://www.google.com/">Google</a></li>
    </ul>
</ul>
</body>
</html>

实例程序的功能是将导游层次关系的连接的集合制作成HTML文件,最后制作完成的HTML如上

在实例程序中,类被划分3个包

  • factory包:包含抽象工厂、零件、产品的包
  • listfactory包:包含具体工厂、零件、产品的包
  • Main的包

3.1.1 具体实现

抽象的零件:item类
Item类是Link和Tray类的父类,这样Link和Tray类就有可替换行了。
caption字段表示项目的标题
makeHTML方法交给子类实现

各个类:

package designModel.AbstractFactory.factory;

public abstract class Factory {
    public static Factory getFactory(String classname){
        Factory factory = null;
        try{
            factory = (Factory) Class.forName(classname).newInstance();
        }catch (ClassNotFoundException e){
            System.out.println("没有找到" + classname + " 类.");
        }catch (Exception e){
            e.printStackTrace();
        }
        return factory;
    }
    public abstract Link createLink(String caption, String url);
    public abstract Tray createTray(String caption);
    public abstract Page createPage(String title, String author);
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.factory;

public abstract class Item {
    protected String caption;

    public Item(String caption){
        this.caption = caption;
    }
    public abstract String makeHTML();
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.factory;

import java.io.PrintWriter;

public abstract class Link extends Item {
    protected String url;
    public Link(String caption, String url){
        super(caption);
        this.url = url;
    }
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.factory;

import java.io.FileWriter;
import java.io.IOException;
import java.io.Writer;
import java.util.ArrayList;

public abstract class Page {
    protected String title;
    protected String author;
    protected ArrayList content = new ArrayList();
    public Page(String title, String author){
        this.title = title;
        this.author = author;
    }

    public void add(Item item){
        content.add(item);
    }

    public void output(){
        try{
            String fileName = title + ".html";
            Writer writer = new FileWriter(fileName);
            writer.write(this.makeHTML());
            writer.close();
            System.out.println(fileName + "编写完成");
        }catch (IOException e ){
            e.printStackTrace();
        }
    }

    public abstract String makeHTML();
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.factory;

import java.util.ArrayList;

public abstract class Tray extends Item {
    protected ArrayList tray = new ArrayList();

    public Tray(String caption) {
        super(caption);
    }

    public void add(Item item) {
        tray.add(item);
    }
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.listfactory;

import designModel.AbstractFactory.factory.Factory;
import designModel.AbstractFactory.factory.Link;
import designModel.AbstractFactory.factory.Page;
import designModel.AbstractFactory.factory.Tray;

public class ListFactory extends Factory {
    @Override
    public Link createLink(String caption, String url) {
        return new ListLink(caption, url);
    }

    @Override
    public Tray createTray(String caption) {
        return new ListTray(caption);
    }

    @Override
    public Page createPage(String title, String author) {
        return new ListPage(title, author);
    }
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.listfactory;

import designModel.AbstractFactory.factory.Link;

public class ListLink extends Link {
    public ListLink(String caption, String url){
        super(caption, url);
    }
    @Override
    public String makeHTML() {
        return "<li><a href=\"" + url + "\">" + caption + "</a><li>\n";
    }
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.listfactory;

import designModel.AbstractFactory.factory.Item;
import designModel.AbstractFactory.factory.Page;

import java.util.Iterator;

public class ListPage extends Page {

    public ListPage(String title, String author){
        super(title, author);
    }

    @Override
    public String makeHTML() {
        StringBuffer buffer = new StringBuffer();
        buffer.append("<html><head><title>" + title + "</title></head>\n");
        buffer.append("<body>\n");
        buffer.append("<h1>" + title + "</h1>\n");
        buffer.append("<ul>\n");
        Iterator it = content.iterator();
        while(it.hasNext()){
            Item item = (Item)it.next();
            buffer.append(item.makeHTML());
        }
        buffer.append("</ul>\n");
        buffer.append("<hr><address>" + author + "</address>");
        buffer.append("</body></html>\n");
        return buffer.toString();
    }
}
/* -------------------------------------------------------- */
package designModel.AbstractFactory.listfactory;

import designModel.AbstractFactory.factory.Item;
import designModel.AbstractFactory.factory.Tray;

import java.util.Iterator;

public class ListTray extends Tray {

    public ListTray(String caption){
        super(caption);
    }

    @Override
    public String makeHTML() {
        StringBuffer buffer = new StringBuffer();
        buffer.append("<li>\n");
        buffer.append(caption + "\n");
        buffer.append("<ul>\n");
        Iterator it = tray.iterator();
        while(it.hasNext()){
            Item item = (Item) it.next();
            buffer.append(item.makeHTML());
        }
        buffer.append("<ul>\n");
        buffer.append("<li>\n");
        return buffer.toString();
    }
}
/* -------------------------------------------------------- */

[外链图片转存失败(img-ePGUBawz-1563093798348)(https://note.youdao.com/src/FB8F70F1087B484B8AFDBE7319FD35A4)]

4、Builder模式

组装具有复杂结构的实例叫做Builder模式

4.1 实例程序

采用Builder模式编写“文档”,文档具有

  • 含有一个标题
  • 含有几条字符串
  • 含有条目项目
package designModel.BuilderModel;

public abstract class Builder {
    public abstract void makeTitle(String title);
    public abstract void makeString(String str);
    public abstract void makeItems(String[] items);
    public abstract void close();
}
/* -------------------------------------------------------- */
package designModel.BuilderModel;

public class Directory {
    private Builder builder;

    public Directory(Builder builder) {
        this.builder = builder;
    }

    public void construct(){
        builder.makeTitle("Greeting");
        builder.makeString("从早上至下午");
        builder.makeItems(new String[]{
                "早上好.",
                "下午好."
        });
        builder.makeString("晚上");
        builder.makeItems(new String[] {
                "晚上好,",
                "晚安。",
                "再见."
        });
        builder.close();
    }
}
/* -------------------------------------------------------- */
package designModel.BuilderModel;

public class TestBuilder extends Builder {

    StringBuffer buffer = new StringBuffer();

    @Override
    public void makeTitle(String title) {
        buffer.append("===========================\n");
        buffer.append("[" + title +"]\n");
        buffer.append("\n");
    }

    @Override
    public void makeString(String str) {
        buffer.append("-" + str +"\n");
        buffer.append("\n");
    }

    @Override
    public void makeItems(String[] items) {
        for(String item:items){
            buffer.append("    ." + item + "\n");
        }
        buffer.append("\n");
    }

    @Override
    public void close() {
        buffer.append("===========================\n");
    }
    public String getResult(){
        return buffer.toString();
    }
}
/* -------------------------------------------------------- */

5 责任链模式

5.1 拦截器模式

首先定义一个拦截器接口:

package designModel.responseChain;

import java.lang.reflect.Method;

public interface Interceptor {
    public boolean before(Object proxy, Object target, Method method, Object[] args);
    public boolean around(Object proxy, Object target, Method method, Object[] args);
    public boolean after(Object proxy, Object target, Method method, Object[] args);
}

实现这个Inteceptor类:

package designModel.responseChain;

import java.lang.reflect.Method;

public class MyInterceptor implements Interceptor {
    @Override
    public boolean before(Object proxy, Object target, Method method, Object[] args) {
        System.out.println("反射方法前逻辑");
        return false;//不反射被代理对象原有方法
    }

    @Override
    public void around(Object proxy, Object target, Method method, Object[] args) {
        System.out.println("取代了被代理对象逻辑");
    }

    @Override
    public void after(Object proxy, Object target, Method method, Object[] args) {
        System.out.println("反射方法后逻辑");
    }
}
package designModel.responseChain;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class InterceptorJdkProxy implements InvocationHandler {

    private Object target; //真实对象

    private String interceptorClass = null; //拦截器全限定名

    public InterceptorJdkProxy(Object target, String interceptorClass) {
        this.target = target;
        this.interceptorClass = interceptorClass;
    }

    public static Object bind(Object target, String interceptorClass){
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),new InterceptorJdkProxy(target,interceptorClass));
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if(interceptorClass == null){
            return method.invoke(target, args);
        }
        Object result = null;
        Interceptor interceptor = (Interceptor) Class.forName(interceptorClass).newInstance();
        if(interceptor.before(proxy, target,method,args)){
            result = method.invoke(proxy,args);
        }else{
            interceptor.around(proxy, target, method, args);
        }
        interceptor.after(proxy,target,method,args);
        return result;
    }
}

5.2 责任链模式

责任链模式就是运用多个拦截器,一个拦截器拦截另外一个代理对象,以此类推。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值