package org.rui.pattern;

import junit.framework.*;

/**
* 为了使同一个方法调用能够产生不同的行为,State 模式在代理(surrogate)的
* 生命周期内切换它所相应的实现(implementation)。当你发现,在决定怎样实现任 对象去耦(Object decoupling)
* http://blog.csdn.net/lxwde 28 何一个方法之前都必须作非常多測试的情况下,这是一种优化实现代码的方法。比如,
* 童话故事青蛙王子就包括一个对象(一个生物),这个对象的行为取决于它自己所处 的状态。你能够用一个布尔(boolean)值来表示它的状态,測试程序例如以下:
*
* @author Administrator
*
*/
class Creature
{
private boolean isFrog = true; public void greet()
{
if (isFrog)
System.out.println("Ribbet!");
else
System.out.println("Darling!");
} public void kiss()
{
isFrog = false;
}
} public class KissingPrincess extends TestCase
{
Creature creature = new Creature(); public void test()
{
creature.greet();
creature.kiss();
creature.greet();
} public static void main(String args[])
{
junit.textui.TestRunner.run(KissingPrincess.class);
}
} // /:~
package org.rui.pattern;

import junit.framework.*;

/**
* 状态模式:改变对象的行为 一个用来改变类的(状态的)对象。 迹象:差点儿全部方法里都出现(同样的)条件(表达式)代码。
*
*
* 可是,greet() 方法(以及其他全部在完毕操作之前必须測试 isFrog 值的那些方
* 法)终于要产生一大堆难以处理的代码。
* 假设把这些操作都托付给一个能够改变的状 态对象(State object),那代码会简单非常多。
*
* @author Administrator
*
*/
class Creature2
{ private interface State
{
String response();
} private class Frog implements State
{
public String response()
{
return "Ribbet!";
}
} private class Prince implements State
{
public String response()
{
return "Darling!";
}
} private State state = new Frog(); public void greet()
{
System.out.println(state.response());
} public void kiss()
{
// 改变对象状态
state = new Prince();
}
} public class KissingPrincess2 extends TestCase
{
Creature2 creature = new Creature2(); public void test()
{
creature.greet();
creature.kiss();
creature.greet();
} public static void main(String args[])
{
junit.textui.TestRunner.run(KissingPrincess2.class);
}
} // /:~
package org.rui.pattern;

import org.junit.Test;

import junit.framework.*;

/**
* 以下的代码演示了 State 模式的基本结构
*/ interface State
{
void operation1(); void operation2(); void operation3();
} class ServiceProvider
{
private State state; public ServiceProvider(State state)
{
this.state = state;
} public void changeState(State newState)
{
state = newState;
System.out.println("========changeState==========="
+ newState.getClass().getSimpleName());
} // 通过方法调用实现
public void service1()
{
// ...
state.operation1();
// ...
state.operation3();
} public void service2()
{
// ...
state.operation1();
// ...
state.operation2();
} public void service3()
{
// ...
state.operation3();
// ...
state.operation2();
}
} class Implementation1 implements State
{
public void operation1()
{
System.out.println("Implementation1.operation1()"); } public void operation2()
{
System.out.println("Implementation1.operation2()");
} public void operation3()
{
System.out.println("Implementation1.operation3()");
}
} class Implementation2 implements State
{
public void operation1()
{
System.out.println("Implementation2.operation1()");
} public void operation2()
{
System.out.println("Implementation2.operation2()");
} public void operation3()
{
System.out.println("Implementation2.operation3()");
}
} public class StateDemo extends TestCase
{
static void run(ServiceProvider sp)
{
sp.service1();
sp.service2();
sp.service3();
} ServiceProvider sp = new ServiceProvider(new Implementation1()); @Test
public void test()
{
run(sp);
sp.changeState(new Implementation2());
run(sp);
} /*
* public static void main(String args[]) {
* junit.textui.TestRunner.run(StateDemo.class); }
*/
} // /:~ /*
* 在 main( )函数里,先用到的是第一个实现,然后转入第二个实现。 当你自己实现 State 模式的时候就会碰到非常多细节的问题,你必须依据自己的需
* 要选择合适的实现方法,比方用到的状态(State)是否要暴露给调用的客户,以及如 何使状态发生变化。有些情况下(比方 Swing 的
* LayoutManager),,client能够直接 传对象进来,可是在 KissingPrincess2.java 那个样例里,状态对于client来说是不
* 可见的。此外,用于改变状态的机制可能非常easy也可能非常复杂-比方本书后面将要提 到的状态机(State
* Machine),那里会讲到一系列的状态以及改变状态的不同机制。 上面提到 Swing 的 LayoutManager 那个样例非常有趣,它同一时候体现了
* Strategy 模 式和 State 模式的行为。 Proxy 模式和 State 模式的差别在于它们所解决的问题不同。《设计模式》里是这 么描写叙述
* Proxy 模式的一般应用的: 1. 远程代理(Remote Proxy)为一个对象在不同的地址空间提供局部代理。 RMI 编译器(rmic)在创建
* stubs 和 skeletons 的时候会自己主动为你创建一 个远端代理。 2. 虚代理(Virtual proxy),依据须要,在创建复杂对象时使用
* “延迟初 始化(lazy initialization)” . 3. 保护代理(protection proxy) 用于你不希望client程序猿全然控制被代
* 理对象(proxied object)的情况下。 4. 智能引用(smart reference). 当訪问被代理对象时提供额外的动作。
* 比如,它能够用来对特定对象的引用进行计数,从而实现写时复制 (copy-on-write),进而避免对象别名(object aliasing). 更简单的
* 一个样例是用来记录一个特定方法被调用的次数。 你能够把 java 里的引用(reference)看作是一种保护代理,它控制对分配在堆
* (heap)上的实际对象的訪问(并且能够保证你不会用到一个空引用(null reference))。 『重写:在《设计模式》一书里,Proxy 模式和
* State 模式被觉得是互不相干的, 由于那本书给出的用以实现这两种模式的结构是全然不同的(我觉得这样的实现有点武 断)。尤其是 State
* 模式,它用了一个分离的实现层次结构,但我觉着全然没有必 要,除非你认定实现代码不是由你来控制的(当然这也是一种可能的情况,可是假设
* 代码是由你来控制的,那还是用一个单独的基类更简洁有用)。此外,Proxy 模式的实
* 现不须要用一个公共的基类,由于代理对象仅仅是控制对被代理对象的訪问。虽然有细 节上的差异,Proxy 模式和 State
* 模式都是用一个代理(surrogate)把方法调用传递 给实现对象。』
*/

java 状态模式 解说演示样例代码的更多相关文章

  1. java文件夹相关操作 演示样例代码

    java文件夹相关操作 演示样例代码 package org.rui.io; import java.io.File; import java.io.FilenameFilter; import ja ...

  2. java 线程、线程池基本应用演示样例代码回想

    java 线程.线程池基本应用演示样例代码回想 package org.rui.thread; /** * 定义任务 * * @author lenovo * */ public class Lift ...

  3. java 又一次抛出异常 相关处理结果演示样例代码

    java 又一次抛出异常 相关处理结果演示样例代码 package org.rui.ExceptionTest; /** * 又一次抛出异常 * 在某些情况下,我们想又一次掷出刚才产生过的违例,特别是 ...

  4. java I/O进程控制,重定向 演示样例代码

    java I/O进程控制,重定向 演示样例代码 package org.rui.io.util; import java.io.*; /** * 标准I/O重定向 */ public class Re ...

  5. 10分钟理解Android数据库的创建与使用(附具体解释和演示样例代码)

    1.Android数据库简单介绍. Android系统的framework层集成了Sqlite3数据库.我们知道Sqlite3是一种轻量级的高效存储的数据库. Sqlite数据库具有以下长处: (1) ...

  6. [Python] SQLBuilder 演示样例代码

    用Python写一个SQLBuilder.Java版能够从 http://www.java2s.com/Code/Java/Database-SQL-JDBC/SQLBuilder.htm 看到. 附 ...

  7. Linux下用OTL操作MySql(包含自己封装的类库及演示样例代码下载)

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/ClamReason/article/details/23971805 首先重点推荐介绍otl介绍及使 ...

  8. C编程规范, 演示样例代码。

    /*************************************************************** *Copyright (c) 2014,TianYuan *All r ...

  9. AppCan移动应用开发平台新增9个超有用插件(内含演示样例代码)

    使用AppCan平台进行移动开发.你所须要具备的是Html5+CSS +JS前端语言基础.此外.Hybrid混合模式应用还需结合原生语言对功能模块进行封装,对于没有原生基础的开发人员,怎样实现App里 ...

随机推荐

  1. MSSQL - Sqlcommand

    Command对象:·Command对象也称为数据库连接对象,Command对象主要执行包括添加.删除.修改.查询数据的操作命令.也可以用来执行存储过程. 属性:CommandType属性. 执行存储 ...

  2. mac下和windows下清空DNS缓存

    在WIN下: > ipconfig /flushdns 在mac下: 对于低版本系统,在命令行窗口(terminal)输入 lookupd -flushcache 即可: $ sudo look ...

  3. 团队 / Staff_VidaMiaTangoClub_新浪博客

    团队 / Staff_VidaMiaTangoClub_新浪博客 团队 / Staff

  4. Swift - 添加、修改、删除通讯录联系人

    使用AddressBook.framework框架,我们除了可以很方便的获取通信录里的联系人.同时,还能对通讯录进行新增.修改.删除联系人操作. (注意:这些操作同查询一样,首先需要发起授权请求) 1 ...

  5. Android实时获取音量(单位:分贝)

    基础知识 度量声音强度,大家最熟悉的单位就是分贝(decibel,缩写为dB).这是一个无纲量的相对单位,计算公式如下: 分子是测量值的声压,分母是参考值的声压(20微帕,人类所能听到的最小声压).因 ...

  6. SQL Server使用问题总结

    1.datetime,smalldatetime,date的区别 1)datetime 从1753年1月1日到9999年12月31日的日期和时间数据,精确度为百分之三秒(等于   3.33毫秒或0.0 ...

  7. Irvine的专业汇编网站

    http://asmirvine.com/ http://download.csdn.net/download/stupid_boy2007/3890853 http://download.csdn. ...

  8. 使用apache daemon让java程序在unix系统上以服务方式运行

    通过使用apache_commons_daemon,可以让Java程序在unix系统上以服务器的方式运行. 当然,通过wrapper也是可以达到这样的目的,wrapper还可以指定java应用中用到的 ...

  9. jni 入门 android的C编程之旅 --->环境搭建&&helloworld

    需要进行jni的开发有一下几个条件: 1:能初步使用C/C++如果不会,请参读 谭浩强的  C编程语言 2:android应用开发已经基本入门,如果没有,请先行学习 这两个条件基本满足后,我们开始了: ...

  10. alv行可编辑时带出描述

    ALV显示可以编辑的状态下可以带出描述信息等,比如维护表程序输入公司代码时需要带出公司代码的描述,这时就需要通过下面事件来触发 定义一个类: CLASS lcl_event_receiver DEFI ...