您的位置:首页 > 编程语言 > Java开发

Java23种设计模式02

2013-04-12 08:00 423 查看

1.2.5 外观模式

为子系统中的一组接口提供一个一致的界面,Facade模式定义了一个高层接口,这个接口使得这*子系统更加容易使用。

适用性

1.当你要为一个*杂子系统提供一个简单接口时。子系统往往因为不断演化而变得越来越

复杂。大多数模式使用时都会产生更多更小的类。这使得子系统更具可重用性,也更容

易对子系统进行定制,但这也给*些不需要定制子系统的用户带来一些使用上的困难。

Fa*ade可以提供一个简单的缺省视图,这一视图对大多数用户来说已经足*,而那些需

要更多的可定制性的用户可以越过facade层。

2.客户程序与抽象类的实现部分之间存在着很大的依赖性。引入facade将这个子系统与客

户以及其他的子系统分离,可以提高子系统的独立性和可移植性。

3.当你需要构建一个层次结构的子系统时,使用facade模式定义子系统中每层的入口点。

如果子系统之间是相互依赖的,你可以让它们仅通过facade进行通讯,从而简化了它们

之间的依赖关系。

参与者

1.Facade

知道哪些子系统类负责处理请求。

将客户的请求代理给适当的子系统对象。

2.Subsystemclasses

实现子系统的功能。

处理由Facade对象指派的任务。

没有facade的任何相关信息;即没有指向*acade的指针。

类图

例子

Facade

publi* class Facade {
ServiceA s*;

ServiceB sb;

ServiceC sc;

public Facade() {
sa = new S*rviceAImpl();
sb = new *erviceBImpl();
sc = new ServiceCImpl();
}

public void methodA() {
sa.methodA();
sb.methodB();
}

publi* void methodB() {
s*.methodB();
sc.methodC();
}

public void methodC() {
sc.methodC();
sa.methodA();
}
}
Subsystemclasse*

public *lass ServiceAImpl implements ServiceA {
public void methodA() {
System.out.println("这是服务A");
}
}
public class ServiceBImpl implements ServiceB {
public void methodB() {
System.out.println("这是服务B");
*
}
public class ServiceCImpl implements ServiceC {
public void methodC() {
System.out.println("这是服*C");
}
}
Test

public class Test {

public static voi* main(String[] args) {
ServiceA sa = new ServiceAImpl();
Ser*iceB sb = new ServiceBImpl();

sa.metho*A();
sb.methodB();

System.out.println("========");
//facade
Facade facade = new Facade();
facade.methodA();
facade.methodB();
}
}
resu*t

这是服务A
这是*务B
========
这是服务A
这是服务B
这是服务B
这是服务C

1.2.6 享元模式

运用共享技术有效地支持大量细粒度的对象。

适用性

当都具备下列情况时,使用Flyweight模式:

1.一个应用程序使用了大量的*象。

2.完全由于使用大量的对象,造成很大的存储开销。

3.对象*大多数状态都可变为外部状态。

4.如果删除对象的外部状态,那么可以*相对较少的共享对象取代很多组对象。

5.应用程序不依赖于对象标识。由于Flyweight对象可以被共享,对于*念上明显有别的对象,标识测试将返回真值。

参与者

1.Flyweight

描述一个接口,通过这个接口flyweight可以接受并作用于外部状态。

2.ConcreteFlyweight

实现Flyweight接口,并为内部状态(如果有的话)增加存储空间。

Concrete*lyweight对象必须是可共享的。它所存储的状态必须是内部的;即,它必须独立于ConcreteFlyweight对象的场景。

3.UnsharedConcreteFlyweigh*

并非所有的Flywe*ght子类都需要被共享。Flyweight接口使共享成为可能,但它并不强制共*。

在Flyweight对象结构的某些层次,UnsharedConcreteFlyweight对象通常将ConcreteFlyweight对象作为子节点。

4.Flyweigh*Factory

创建并管理flywe*ght对象。

确保合理地共享flyweight。当用户请求一个flywei*ht时,Fl*weightFactory对象提供一个已创建的实例或者创建一个(如果不存在的话)。

类*

例子

Flyweight

public int*rface Flyweight {
void a*tion(int arg);
}
ConcreteFlyweight

public class FlyweightImpl implements Flyweight {
public void action(int arg) {
// T*DO Auto-genera*ed method stub
System.out.println(*参数值: " + arg);
}
}
FlyweightFactory

public class Flyweigh*Factory {
private static Map flyweights = new HashMap();

public FlyweightF*ctory(String arg) {
flyweights.put(arg, new FlyweightImpl());
}

public static Flyweight getFly*eight(String key) {
if (flyweights.get(key) == null) {
flyweights.p*t(key, new FlyweightImpl());
}
return flyweights.get(key);
}

public static int g*tSize() {
retu*n flyweights.size();
}
}
Test

public class Test {
public static v*id main(String[] args) {
// TODO Auto-generated method stub
Flyweight fly1 = Flyw*ightFact*ry.getFlyweight(*a");
fly1.action(1);

Flyweight fly* = FlyweightFactory.getF*yweight("a");
System.out.println(fly1 == fly2);

Flyweight fl*3 = FlyweightFactory.getFlywei*ht("b");
fly3.action(2);

Flyweight fly4 = Flyweigh*Factory.getF*yweight("c");
fly4.action(3);

Flyweigh* fly5 = FlyweightFactory.getFlyweight("d");
fly4.action(4);

System.out.println(FlyweightFactory.getSize())*
}
}
result

参数值: 1
true
参数值: 2
*数值: 3
参数值: 4
4

1.2.7 代理模式

为其他对象提供一种代理以控制对这个对象的访问。

适用性

1.远程代理(RemoteProxy)为一个对象在不同的地址空间提供局部代表。

2.虚*理(VirtualProxy)根据需*创建开销很大的对象。

3.保护代理(ProtectionProxy)控制对原始对象的访问。

4.智能指引(SmartReference)取代了简单的指针,它在访问对象时执行一些附加操作。

参与者

1.Proxy

保存一个引用使得代理可以访问实体。若RealSubject和Subject的接口相同,Proxy会引用Subject。

*供一个与Subject的接口相同的接口,这样代理就可以用来替代实体。

控制对实体的*取,并可能负责创建和删除它。

其他功能依赖于*理的类型:

2.RemoteProxy负责对请求及其参数进行编码,并向不同地址空间中的实体发送已编码的请求。

*.VirtualProxy可以缓存实体的附加信息,以便延迟对它的访问。

4.ProtectionProxy检查调用者是*具有实现一个请求所必需的访问权限。

5.Subjec*

定义RealSubject和Proxy的共用接口,这样就在任何使用RealSubject的地方都*以使用Proxy。

6.RealSubject

*义Proxy所代表的实体。

类图

例子

Proxy

public class ProxyObject implements Object {
Objec* obj;

public ProxyObject() {
System.out.println("这是代理类");
o*j = new ObjectImpl();
}

public void ac*ion() {
System.out.p*intln("代理开始");
obj.action*);
System.out.println(*代理结束");
}
}
Subject

public interface Obje*t {
void action();
}
RealSubject

public class ObjectImpl implements Object {
pu*lic void action() {
System.out.println("========");
System.out.println("========");
System.out.pr*ntln("这是被代理的类");
System.out.println("========");
System.out.println("========");
}
}
Test

public class Test {
publi* static void main() {
Object ob* = new ProxyObject();
obj.action();
*
}
result

这是代理类
代理开始
========
=*======
这是被代理的类
========
======*=
代理结束

1.3 行为型模式

Chain of Responsibility ( 责任链模式 )

Command ( 命令模式 )

Interpreter ( 解释器模式 )

Iterator ( 迭代器*式 )

Mediator ( 中介者模式 )

Memento ( 备忘录模式 )

Observer ( 观察者模式 )

State ( 状*模式 )

Strategy ( 策略模式 )

TemplateMethod ( 模板方法 )

Vis*tor ( 访问者模式 )

1.3.1 责任链模式

使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一*链,

并*着这条链传递该请求,直到有一个对象处理它为止。

这一模式的想法是,给多个对象处理一个请求的机会,从而解耦发送者和接受者.

适用性

1.有多个的对象可以处理一个请求,哪个对象处理该请求运行时刻自动确定。

2.你*在不明确指定接收者的情况下,向多个对象中的一个提交一个请求。

3.可处理一个请求的对象集合应被动态指定。

参与者

1.Handler

定义一个处理请求的接口。

(可选)实现后继链。

2.ConcreteHandler

处理它所负责的请*。

可访问它的后继者。

如果可处理该*求,就处理*;否则将该请求转发给它的后继者。

3.Client

向链上的具体处理者(ConcreteHandler)对象提交请求。

类图

例子

Hand*er

public interface RequestHandle {
void handleRequest(R*quest request);
}
ConcreteHandler

public class HRRequestHandle implements RequestHandle {
public void handleRequest(Request request) {
if (request instanceof DimissionRequest) {
System.out.println("要离职, 人事审批!");
}
System.out.println("请求完*");
}
}
public class PMRequestHandle implements RequestHandle {
Req*estHandle rh;

public PMRequestHandle(RequestHandle *h) {
this.rh = rh;
}

public void handle*equest(Request request) {
if (request instanceof AddMoneyRequest) {
System.out.println("要加薪, 项目经理审批!*);
} else {
rh.handleRequest(request);
}
}
}
public class TLRequestHandle implements RequestHandle {
RequestHandle rh;

public TLRequestHandle(RequestHand*e rh) {
this.rh = rh;
}
public void handleRequest(Request request) {
if (request instanceof LeaveRe*uest) {
System.ou*.println("要请假, 项目组长审批!");
} else {
rh.handleRequest(request);
}
}
}
Client

public *lass Test {

public static v*id main(String[] args) {
RequestHa*dle hr = *ew HRRequ*stHandle();
Requ*stHandle pm = new P*RequestHandle(hr);
RequestHandle tl = new TLRequestHandle(pm);

//team leader处理离职请求
Request request = new DimissionRequest()*
tl.handleRequest(request);

System.out.println("===========");
//team leader处理加薪请求
request = new AddMoneyRequest();
tl.handleRequ*st(request);

System.out.println("========");
//项目经理上理辞职请求
requ*st = ne* Dimissio*Request();
pm.handleRequest(request);
}
}
result

要离职, 人事审批!
请求完毕
=======*===
要加薪, 项目经理审批!
========
要离职, 人事审批!
请求完毕

1.3.2 命令模式

将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤消的*作。

适用性

1.抽象出待执行的动作以参数化某对象。

2.在不同的时刻指定、排列和执行请求。

3.支持取消操作。

4.支持修改日志,这样当系统崩溃时,这*修改可以被重做一遍。

5.用构建在原语操作上的高层操作构造一个系统。

参与者

1.Comma*d

声明执行操作的接口。

2.ConcreteCommand

将一个接收者对象绑定于一个动作。

调用接收者相应的操作,以实现Execute。

3.Client

创建一个具体命令对象并设定它的接收者。

4.Invoker

要求该命令执行这个请求。

5.Recei*er

知道如何实*与执行一个请求相关的操作。任何类都可能作为一个接收者。

类图

例子

Command

public abstract class *ommand {

protecte* Receiver receiver;

public Command(Receiver re*eiver) {
this.receiver = receiver;
}

public abstract *oid execute();
}
ConcreteCommand

public class CommandImpl extends Comman* {
public CommandImpl(Receiv*r receiver) {
super(receiver);
}

pu*lic void *xecute*) {
receiver.request();
}
}
Invoker

public cl*ss Invoker {
private Command command;

pub*ic void setCommand(Command c*mmand) {
this.command = command;
}

public void execute*) {
command.execute();
}
}
Receiver

public class Receiver {
public void receive() {
S*stem.out.println("This is Receive class!");
}
}
Test

publ*c class Test {
pub*ic static void main*String[] args) {
R*ceiver rec = new Receiver();
Command cmd = n*w CommandImpl(rec);
Invoker i = new Invoker();
i.setCom*and(cmd);
i.execut*();
}
}
result

This is Receive class!

1.3.3 解释器模式

给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。

适用性

当有一个语言需要解释执行,并且你可将该语言中的句子表示为一个抽象语法树时,可使

用解释器模式。而当存在*下情况时该模式效果最好:

1.该文法简单对于复杂的文法,文法的*层次变得庞大而无法管理。

2.效率不是一个关键问题最高效的解释器通常不是通过直接解释语法分析树实现的,而是首先将它们转换成另一种形式。

参与者

1.AbstractExpression(抽象表达式)

声明一个抽象的解释操作,这个接口为抽象语法树中所有的节点所共享。

2.TerminalExpression(终结符表达式)

实现与文法中的终结符相关联的解释操作。

一个句子中的每个终结符需要该类的一个实例。

3.N*nterminalExpression(非终结符表达式)

为文法中的非终结符实现解释(Interpret)操作。

4.Context(上下文)

包含解释器之外的一些全局信息。

5.Client(客户)

构建(或被给定)表示该文法定义的语言中*个特定的句子的抽象*法树。

该抽象语法树由NonterminalExpression和TerminalExpression的实例装配而成。

调用解*操作。

类图

例子

AbstractExpression

pu*lic abstract class Expression {
abstract void interpret(Context ctx);
}
Expression

public class AdvanceExpressio* extends Expression {
void interpr*t(Context ctx) {
System.out.println("这是高级解析器!");
}
}
public class SimpleExpression extends Expressio* {
void interpret(*ontext ctx) {
System.out.pri*tln("这是普通解析器!");
}
}
Context

public class Co*text {
private S*ring content;

*rivate List list = new ArrayList();

public void setContent(String content) {
this.content = content;
}

pu*lic String getContent() {
return this.con*ent*
}

public void add(Expression ep*) {
list.add(eps);
}

public List getList() {
return list;
}
}
Test

public class Test {
public static void main(String[] args) {
Context *tx = new Context();
ctx.*dd(new SimpleExpression());
ctx.add(new AdvanceExpression());
ctx.add(new SimpleExpression());

for *Expression eps : ctx.getL*st()) {
eps.interpret*ctx);
}
}
}
res*lt

*是普通解析器!
这是高级解析器!
*是普通解析器!

1.3.4 迭代器模式

给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。

适用性

1.访问一个聚合对象的内容而无需暴露它的内部表示。

2.支持对聚合对象的多种遍历。

3.为遍历不同的聚合结构提供一*统一的接口(即,支持多态迭代)。

参与者

1.Iterator

迭代器定义访问和遍历元素的接口。

2.ConcreteIterator

具*迭代器实现迭代器接口。

对该聚合遍历时跟踪当前位置。

3.Aggregate

聚合定义创建相应迭代器*象的接口。

4.ConcreteAggregate

具体聚合实现创建相应迭代器的接口,该操作返回ConcreteIterator的一个适当的实例.

类图

例子

Iterator

public interface Iterator {
Object nex*();

void first();

voi* last();

boolean hasNext();
}
ConcreteIterator

public class IteratorImpl implements It*rator {
private List list;

private int index;

public Ite*atorImpl(List list* {
index = 0;
this.list = list;
}

public void first() {
index = 0;
}
publ*c void last() {
index = list.getSize();
}
public Object next() {
Object obj = list.get(index);
index++;
ret*rn obj;
}
public boolean hasNext() {
return index < list.getSize();
}
}
Aggregate

p*blic interface List {
Iterator iterator();

Object get(int index);

int *etSize();

void add(Object ob*);
}
ConcreteAggregate

public class ListImpl implements List {
private Object[] list;

private int index;

private int size;

public ListImpl() {
index = 0;
size = 0;
list = new Object[100];
}

public Iterator iterator() {
return new IteratorImpl(this);
}

public O*ject get(int index) {
return list[index];
}

public int getSize() {
return this.size;
}

public void add(Object obj) {
list[index++] = obj;
size++;
}
}
Test

public class Test {
public stati* void main(String[] arg*) {
List list = new ListImpl();
list.add("a");
list.add("b");
list.add("c");
//第一种迭代方式
Iterator it = list.iterator();
while (*t.ha*Next()) {
S*stem.out.println(it.next());
}

Syst*m.out.println("=====");
//第二种迭代方式
for (int i = 0; i < list.getSize(); i++) {
System.out.println(list.get(i));
}
}
}
result

a
b
c
=====
a
b
c

1.3.5 中介者模式

用一个中介对象来封装一系列的对象交互。中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。

适用性

1.一组对象以定义良好但是复杂的方式进行通信。产生的相互依赖关系结构混乱且难以理解。

2.一个对象引用其他很多对象并且直接与这些对象通信,导致难以复*该对象。

3.想定制一个分布在多个类中的行为,*又不想生成太多的子类。

参与者

1.Mediator

中介者定义一个接口用于与各同事(Colleague)对象通信。

2.ConcreteMediator

具*中介者通过协调各同事对象实现协作行为*

了解并维护它的各个同事。

3.Colleagueclass

每一个同事类都知道它的中介者对象。

每一个同事对象在需与其他的同事通信的时候*与它的中介者通信

类图

例子

Mediator

public abstract class Mediator {
public abstract void notice(String content);
}
ConcreteMediator

public *lass ConcreteMediator e*tends Mediator {
private ColleagueA ca;

pri*ate ColleagueB cb;

public ConcreteMediator() {
ca = new ColleagueA();
cb = new Col*eagueB();
}

public void no*ice(String content) {
if (co*tent.equals("boss")) {
//老板来了, 通知员工A
ca*action();
}
if (content.equals("client")) {
//客户来了, *知前台B
cb.action();
}
}
}
Colleagueclass

public class ColleagueA extends *olleague {

public void action(* {
System.out.println("普通员工努力工作");
}
*
public class ColleagueB extends Colleague {
public void action() {
System.out.println("前台注意了!");
}
}
Test

public class Test {
public static void main(String[] args) {
Mediator med = new Concr*teMediator();
*/老板来了
med.notice("boss");

//客户来*
med.n*tice("client");
}
}
result

普通员工努力工作
前台注意了!

1.3.6 备忘录模式

在不破坏封装性*前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。

适用性

1.必须*存一个对象在某一个时刻的(部分)状态,这样以后需要时它才能恢复到先前的状态。

2.如果一个用接口来让其它对象直接得到这些状态,将会暴露对象的实现细节并破坏对象的封装性。

参与者

1.Memento

备忘录存储原发器对象的内部状态。

2.Originator

原发器创建一个备忘录,用以记录当前时刻*的内部状态。

使用备忘录恢复内部状态.

3.Caretaker

负责保存好备忘录。

不能对备忘录的内*进行操作或检查。

类图

例子

Memento

public class Memento {
private String state;
public Meme*to(String state) {
this.state = state;
}
public String getState() {
*eturn state;
}
public void setSt*te(String state) {
this.stat* = s*ate;
}
}
Or*ginator

public class Originator {
private String state;
public String getState() {
return state;
}
public void setState(String state) {
this.state = state;
}

*ublic Memento createMemento() {
return new Memento(state);
}

p*blic void setMemento(Memento meme*to) {
state = memento.ge*State();
}

p*blic void showState(){
System.out.println(state);
}
}
Caretaker

public class Caretaker {

p*ivate Memento memento;

public Memento getMemento(){
return this.memento;
}

public void setMemento(Memento memento){
this.memento = memento;
}
}
Test

public class Test {
public static void main(String[] args) {
Originator org = *ew Originator();
org*setState("开会中");

C*retaker ctk = new Ca*etaker();
ctk.setMemento(org.createMemento());//将数据封装在Caretaker

or*.setState("睡觉中");
org.sh*wState();*/显示

org.setMemento(ctk.getMemento());//将数据重新导入
or*.showState();
}
}
result

睡觉中
开会中

1.3.7 观察者模式

定义对象间的一种一*多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

适用性

1.当一个抽象模型有两个方面,其中一个方面依赖于另一方面。

将这二者封装*独立的对象中以使它们可以各自独立地改变和复用。

2.当对一个对象的改变需要同时改变其它对象,而不知道具体*多少对象有待改变。

3.当一个对象必须通知其它对象,而它又不能假定其它对象是谁。

参与者

1.Subject(目标)

目标知道它的观*者。可以有任意多个观察者观察同一个目标。

提供注册和删除观察者对象的接口。

2.Obser*er(观察者)

为那些在目标发生改变时需获得*知的对象定义一个更新*口。

3.ConcreteSubject(具体目标)

将有关状态存入各ConcreteObserver对象。

当它的状态发生改变时,向它的各个观察者发出通知。

4.ConcreteObserver(具体观察者)

维护一个指向ConcreteSubject对象的引用。

存储有关状态,这些状态应与目标的状态保持一致。

实现Observer的更新接口*使自身状态与目标的状态保持一致

类*

例子

Subject

public abstract class Citizen {

List po*s;

String help = "normal";

public void setHelp(String help) {
this.help = *elp;
}

public String getHelp() {
return this.help;
}

abstract void sendMessage(String help);
public void setPolicemen() {
t*is.pols = new ArrayList();
}

public void register(*oliceman pol) {
this.pols.add(pol);
}
public void unRegister(Policeman pol) {
this.pols.remove(pol);
}
}
Observer

public interface Policeman {
void action(Citizen ci);
}
*oncreteSubjec*

public class Hua*gPuCitizen extends Citiz*n {
public HuangPuCitizen(P*liceman pol) {
setPolicemen();
register*pol);
}

public void sendMessage(String help) {
setHelp(h*lp);
for(int i = 0; i < pols.size(); i++) {
Policeman pol = pols.get(i);
//通知警察行动
pol.action(this);
}
}
}
public class TianHeCitizen extends Citizen {
public TianHeCitizen(Policeman pol) {
setPolicemen();
re*ister(pol);
}

public void sendM*ssage(Str*ng help) {
setHelp(help);
for (i*t i = 0; i < pols.size(); i++) {
Policeman pol = pols.get(i);
//通知警察行动
pol.action(this);
}
}
}
ConcreteObserver

public clas* HuangPuPoliceman implements Pol*ceman {
public void action(Citizen ci) {
String help = ci.getHelp();
if (help.equals("normal")) {
System.o*t.println("一切正常, 不用出动");
}
if (help.*quals("unnormal")) *
System.out.println("有犯罪行为, 黄埔警察出动!");
}
}
}
public c*ass TianHePoliceman implements Policem*n {
public void action(Citizen ci) {
String help = *i.getHelp();
if (help.equals("*ormal")) {
System.out.println("一切正常, 不用出动");
*
if (help.equals("unnormal")) {
System.out.println("有犯罪行为, 天河警察出动!");
}
}
}
Test

public class Test{
public st*tic void main(String[] args) {
Policeman thPol = new TianHePoliceman();
Pol*ceman hpPol = new HuangPuPoliceman();

Ci*izen citizen = new HuangPuCitizen(hpPol);
citizen.sendMessage("unnormal");
citizen.sendMessage("normal");

System.out.println("===========");

citizen = new TianH*Citizen(thPol);
citizen.sendMessage("normal");
citi*en.sendMessage("unnormal");
}
}
result

有犯罪行为, 黄埔警察出动!
一切正常, 不用出动
======*====
一切正常, 不用出动
有犯罪行为, 天河警察出动!

1.3.8 状态模式

定义对象间*一种一对多的依赖关系,当一个对象的状态*生改变时,所*依赖于它的对象都得到通知并被自动更新。

适用性

1.一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改*它的行为。

2.一个操作中含有庞大的多分支的条件语句,且这些分支依赖于该对象的状态。

这个状态通常用一个或多个枚举常量表示。

通常,有多个操作包含这一相同的条件结构。

State模式将每一个条件分支放入一个独立的类中。

这使得你可以根据对象自身的情况将对象的状态作为一个对象,这一对象可以不依赖于其他对象而独立变化。

参与者

1.Context

定义客户感兴趣的接口。

维护一个ConcreteState子类的实例,这个实例定义当前状态。

2.State

定义一个接口以封装与Context的一个特定状态相关的行为。

3.ConcreteStatesubclasses

每一子类实现一个与Context的一个状态*关的行为。

类图

例子

*ontext

pu*lic class Context {
private Weather weather;
public voi* setWeather(Weather weather) {
this.*eather = weather;
}
pu*lic Weather getWeather() {
return this.weather;
}
public String weatherMessage() {
return w*ather.getWeather();
}
}
State

public interface Weath*r {
St*ing getWeather();
}
Concrete*tatesubclasses

public class Rain implements Weather {
public Stri*g getWeather() {
return "下雨";
}
*
public class Sunshine implements Weather {
public Str*ng getWeather() {
return "阳光";
}
*
Test

public class Test{
public static void main(String[] args) {
Context ctx1 = new Context();
ctx1.setWeather(new Sunshine());
System.out.println(*tx1.weatherMessage());
System.*ut.println("===============");
Context ctx2 = new Context();
ctx2.setWeather(new Rain());
S*stem.out.println(ctx2.weatherMessage());
}
}
result

阳光
===============
下雨

1.3.9 策略模式

定义一系列的算法,把它们*个个封装起来,并且使它们可相互替换。本模式使得算法可独立于使用它的客户而变化。

适用性

1.许多相关的类仅仅是行为有异。“策略”提供了一种用多个行为中的一个行为来配置一个类的方法。

2.需要使用一个算法的不同变体。

3.算法使用客户不应该知*的数据。可使用策略模式以避免暴露复杂的、与算法相关的数据结构。

4*一个类定义了多种行为,并且这些行为在这个类的操作中以*个条件语句的形式出现。

将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句。

参与者

1.Strategy

定义所有支持的算法的公共接口。Context使用这个接口来调用某ConcreteStrate*y定义的算法。

2.ConcreteStrategy

以S*rategy接口实现某具体算法。

*.Context

用一个ConcreteStrateg*对象来配置。

维护一个对Strategy对象的引用。

可定义一个接口来让Stategy访问它的数据。

类图

例子

Strategy

public abstra*t class Strategy {
pu*lic abstract void method();
}
ConcreteStrategy

public class *trategyImplA extends Strategy {
public voi* method() {
System.out.println("这是第一个实现");
}
}
public class StrategyImplB extends Strategy {
public void method() {
System.out.println("这是第二个实现");
}
}
public class StrategyImplC extends Strategy {
public void method() {
Syst*m.out.println("这是第三个实现");
}
}
Context

pub*ic class Context {
Strategy stra;

public Cont*xt(Strategy stra) {
this.stra = stra;
}

pub*ic void doMethod() {
stra*method();
}
}
Test

public class Test {

public static void main(String[] ar*s) {
Context ctx = new C*ntext(new StrategyImplA());
ctx.doMethod();

ctx = new Context(new *trategyImplB());
ctx.doMethod();

ctx = new Context(new StrategyImplC());
ctx.doMethod();
}
}
result

这是第一个实现
这是第二个实现
这是第三个实现

1.3.10 模板方法

定义*个操作中的算法的骨架,*将一些步骤延迟到子类中。

TemplateMethod使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。

适用性

1.一次性实现一个算法的不变的部分,并将可变的*为留给子类来实现。

2.各子类中公共的行为应被提取出来并集中到一个公共父类中以避免代码重复。

首先识别现有*码中的不同之处,并且将不同之处分离为新的操作。

最后,用*个调用这些新的操作的模板方法来替换这些不同的代码。

3.控制子类*展。

参与者

1.AbstractClass

定义抽象的原语操作(primitiveoperation),具体的子类将重定义它们以实现一个算法的各步骤。

实现一个模板方法,定义一个算法的骨架。

该模板方法不仅调用原语操作,也调用定义在AbstractClass或其他对象中的操作。

*.ConcreteClass

实现*语操作以完成算法中与特定子类相关的步骤。

类图

例子

AbstractClass

public abstract class Template {
public abstract void print();

public void update() {
System.out.println("开始打印");
for (int i = 0; i < 10; i++) {
print();
}
}
}
ConcreteClass

public class TemplateConcrete extends Template {
@*verride
public void print() {
System.out.println("这是子类的实现");
}
}
Test

public class Test {
pu*lic static void main(String[] args) {
Te*plate temp = new TemplateConcrete();
temp.update();
}
}
result

开始打印
这是子类的*现
这是子类的实现
这是子类的实现
这是子类的实现
这是子类的实现
这是子类的实现
这是子类的实现
这*子类的实现
这是子类的实现
这是子类的实现

1.3.11 访问者模式

表*一个作用于某对象结构中的各元素的操作。

它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。

适用性

1.一个对象结构包含很多类对象,它们有不同的接口,而你想对这些对象实施一些依赖于其具体类的操*。

2.需要对一个对象结构中的对象进行很多不同的并且不相关的操作,*你想避免让这些操作“污染”这些对象的类。

Visitor使得你可以将相关的操作集中起来定义在一个类中。

当该对象结构被很多应用共享时,用Visitor模式让每个应用仅包含需要用到的操作。

3.定义对象结构的类很少改变,但经常需要在此结构上定义新的操作。

改变对象结构类需要重定义对所有访问者的接口,这可能*要很大的代价。

如果对象结构类经常改变,那么可能还是在这些类中定义这些操作较好。

参与者

1.Visitor

为该对象结构中ConcreteEle*ent的每一个类声明一个Visit操作。

该操作的名字和特征标识了发送*isit请求给该访问者的那个类。

这使得访问者可以确定正被访问元素*具体的类。

这样访问者就可以通过该元素的特定接口直*访问它。

2.Concret*Visitor

实现每个由Visitor声明的操作。

每个操作实现本算法的一部分,而该算法片断乃是对应于结构中对象的类。

Concret*Visitor*该算法提供了上下文并存*它的局部状态。

这一状态常常在遍历该结构的过程中累*结果。

3.Element

定义一个Accept操作,它*一个访问者为参数。

4.ConcreteElement

实现Accept操作,该操作以一个访问者为参数。

5.ObjectStru*ture

能枚举它的元素。

*以提供一个高层的接口以允许该访问者访问它的元素。

可以是一个复合或是一个集合,如一个列表或一个无序集合。

类图

例子

Visitor

public interface Visitor {
public void visitString(StringElement stringE);

public void visitFloat(FloatElement floatE);

public void visitCollection(Collection collection);
}
ConcreteVisitor

public class C*ncreteVisitor implements Visitor {
public void visitCollectio*(Collection colle*tion) {
// TODO Auto-generated method stub
Iterator iterator = collection.iterator();
while (iterator.hasNext()) {
Object o = iterato*.next();
if (o in*tanceof Visitable) {
(*Visitable)o).accept(this);
}
}
}
public void visitFloat(FloatElement floatE) {
System.out.println(floatE.getFe*));
}
public void visitString(StringElement stringE) {
System.out.println(stringE.getSe());
}
}
Element

public interface Visitabl* {
publ*c void accept(Visitor visitor);
}
ConcreteElement

public class FloatElement implements Visitable {
private Float fe;

public FloatElement(Float fe) {
this.fe = fe;
}

public Float getFe() {
return this.fe;
}

public void accept(Visitor visitor) {
visitor.*isitFloat(this);
}
}
public class StringElement implements Visitable *
private String se;

public String*lement(String se) {
this.se = se;
}

public String getS*() {
return thi*.se;
}

public void accept(Visitor visitor) {
visitor.visitString(this);
}
}
Test

public class Test {
public static void main(String[] args) {
Visitor visitor = new ConcreteVisitor();
StringElement se = new StringElement("abc");
s*.accep*(visitor);

Fl*atElement fe = new FloatElement(n*w Float(1.5));
fe.accept(visitor);
S*stem.out.println("===========");
List result = new ArrayList();
result.add(new StringEle*ent("abc"));
result.a*d(new StringElement("abc"));
result.add(*ew StringElement("abc"));
result.add(new FloatElement(new Float(1.5)));
result.add(new FloatElement(new Float(1.5)));
result.add(new FloatElement(new Float(1.5)));
visitor.visitCollection(result);
}
}
result

abc
1.5
===========
abc
abc
abc
1.5
1.5
1.5
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: