设计模式之美:Role Object(角色对象)
索引
意图
通过明确地附加角色对象到目标对象中,以使对象可以适配不同的客户需求。每个角色对象都代表着目标对象在客户上下文中的一种角色。每种上下文都存在于特定的应用程序中,因此可使对象在不同的应用程序间解耦。
Adapt an object to different client’s needs through transparently attached role objects, each one representing a role the object has to play in that client’s context. Each context may be its own application, which therefore gets decoupled from the other applications.
结构

参与者
Component
- 定义关键的抽象接口。
- 定义添加、删除、测试和查询角色对象的接口协议。Client 可以指定一个 specification 来获取一个 ConcreteRole 实例。
ComponentCore
- 实现 Component 接口,包括角色管理协议。
- 负责创建 ConcreteRole 实例。
- 管理角色对象。
ComponentRole
- 存储一个经过装饰的 ComponentCore 对象的引用。
- 实现 Component 接口,并将接口请求转发到 ComponentCore 中。
ConcreteRole
- 定义和实现特定上下文的 Component 接口的实现。
- 构造函数包含一个 ComponentCore 参数。
Client
- 请求协作对象。
适用性
当以下情况成立时可以使用 Role Object 模式:
- 你需要在不同的客户上下文间保持关键的抽象定义,每种抽象定义的实现都存在于其特定的应用范围内,同时你不想将这些与客户上下文相关的接口耦合在同一个接口内。
- 你需要能够动态的处理角色,以便可以在运行时按需的添加或移除角色,而不是在编译时使用固定的对象定义。
- 你需要保持角色与客户成对儿关系,在不同的角色与客户对之间保持独立性,当更改一个角色时并不影响其他客户。
效果
- 可以简洁地定义对象的关键抽象。Component 接口仅需关注必要的状态的行为,而不会涉及具体上下文中的职责。
- 角色对象易于演进,彼此之间保持独立。
- 角色对象可以动态的添加和移除。
- 应用程序可以更好的解耦和。
相关模式
- Decorator模式与 Role Object 模式有着类似的结构,但行为不同。Decorator 模式使开发人员可以对一个对象进行链式的装饰,而 Role Object 模式则不允许这样。并且,Decorator 通常不会扩大对象的功能,而 Role Object 会引入新的操作。
- Extension Object 模式也在解决类似的问题,通过对对象的扩展来满足特定的客户上下文的需求。尽管如此,Extension Object 并没有考虑透明性(Transparently),即保持对象关键抽象(Key Abstraction)的定义,而 Role Object 则正是强调了这一点。
实现
实现方式(一):Role Object 的示例实现。
namespace RoleObjectPattern.Implementation1
{
public abstract class Customer
{
public abstract CustomerRole GetRole(string spec);
public abstract void AddRole(string spec);
public abstract void RemoveRole(string spec);
public abstract bool HasRole(string spec);
public abstract void SomeCommonOperation1();
} public class CustomerCore : Customer
{
private Dictionary<string, CustomerRole> _roles
= new Dictionary<string, CustomerRole>();
private CustomerRoleFactory _roleFactory; public CustomerCore(CustomerRoleFactory roleFactory)
{
_roleFactory = roleFactory;
} public override CustomerRole GetRole(string spec)
{
if (_roles.ContainsKey(spec))
return _roles[spec];
else
return null;
} public override void AddRole(string spec)
{
CustomerRole role = GetRole(spec);
if (role == null)
{
role = _roleFactory.CreateRole(spec, this);
if (role != null)
{
_roles.Add(spec, role);
}
}
} public override void RemoveRole(string spec)
{
_roles.Remove(spec);
} public override bool HasRole(string spec)
{
return _roles.ContainsKey(spec);
} public override void SomeCommonOperation1()
{
// do some business logic
}
} public class CustomerRole : Customer
{
private CustomerCore _decoratedCore; public CustomerRole(CustomerCore core)
{
_decoratedCore = core;
} public override CustomerRole GetRole(string spec)
{
return _decoratedCore.GetRole(spec);
} public override void AddRole(string spec)
{
_decoratedCore.AddRole(spec);
} public override void RemoveRole(string spec)
{
_decoratedCore.RemoveRole(spec);
} public override bool HasRole(string spec)
{
return _decoratedCore.HasRole(spec);
} public override void SomeCommonOperation1()
{
_decoratedCore.SomeCommonOperation1();
}
} public class Borrower : CustomerRole
{
public Borrower(CustomerCore core)
: base(core)
{
} public void SomeOperationForBorrower()
{
// do something for borrower
}
} public class Investor : CustomerRole
{
public Investor(CustomerCore core)
: base(core)
{
} public void SomeOperationForInvestor()
{
// do something for investor
}
} public class CustomerRoleFactory
{
public CustomerRole CreateRole(string spec, CustomerCore core)
{
CustomerRole newRole = null; if (spec == "Borrower")
{
newRole = new Borrower(core);
}
else if (spec == "Investor")
{
newRole = new Investor(core);
} return newRole;
}
} public class Client
{
public void TestCase1()
{
Customer customer = new CustomerCore(new CustomerRoleFactory());
customer.AddRole("Borrower");
customer.AddRole("Investor"); CustomerRole customerRole1 = customer.GetRole("Borrower");
Borrower borrower = (Borrower)customerRole1;
borrower.SomeCommonOperation1();
borrower.SomeOperationForBorrower(); CustomerRole customerRole2 = customer.GetRole("Investor");
Investor investor = (Investor)customerRole2;
investor.SomeCommonOperation1();
investor.SomeOperationForInvestor();
}
}
}
《设计模式之美》为 Dennis Gao 发布于博客园的系列文章,任何未经作者本人同意的人为或爬虫转载均为耍流氓。
设计模式之美:Role Object(角色对象)的更多相关文章
- 设计模式之美:Object Pool(对象池)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):实现 DatabaseConnectionPool 类. 实现方式(二):使用对象构造方法和预分配方式实现 ObjectPool ...
- Role Object(角色对象)
Role Object(角色对象) 索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):Role Object 的示例实现. 意图 通过明确地附加角色对象到目标对象中,以使对象可以 ...
- 设计模式之美:Type Object(类型对象)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):Type Object 的经典介绍. 实现方式(二):Type Object 在游戏设计中的使用. 意图 允许在运行时动态灵活的 ...
- 设计模式之美:Extension Object(扩展对象)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):使用示例结构实现 Extension Object. 实现方式(二):使用泛型实现 IExtensibleObject<T ...
- 设计模式之美:Null Object(空对象)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):Null Object 的示例实现. 意图 通过对缺失对象的封装,以提供默认无任何行为的对象替代品. Encapsulate t ...
- 设计模式之美:Product Trader(操盘手)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):Product Trader 的示例实现. 意图 使客户程序可以通过命名抽象超类和给定规约来创建对象. Product Trad ...
- 设计模式之美:Dynamic Property(动态属性)
索引 别名 意图 结构 参与者 适用性 效果 实现 实现方式(一):Dynamic Property 的示例实现. 别名 Property Properties Property List 意图 使对 ...
- 设计模式之美:Manager(管理器)
索引 意图 结构 参与者 适用性 效果 实现 实现方式(一):Manager 模式的示例实现. 意图 将对一个类的所有对象的管理封装到一个单独的管理器类中. 这使得管理职责的变化独立于类本身,并且管理 ...
- 设计模式之美:Visitor(访问者)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):Visitor 模式结构样式代码. 实现方式(二):使用 Visitor 模式解构设计. 实现方式(三):使用 Acyclic ...
随机推荐
- 搭建高可用mongodb集群(一)——配置mongodb
在大数据的时代,传统的关系型数据库要能更高的服务必须要解决高并发读写.海量数据高效存储.高可扩展性和高可用性这些难题.不过就是因为这些问题Nosql诞生了. NOSQL有这些优势: 大数据量,可以通过 ...
- ESM335x Linux输出脉冲计数
1.综述 ESM335X具有4路PWM输出,其中PWM1和PWM2除了可以用于产生标准的PWM信号,现已支持输出脉冲计数功能,可以在应用程序中设置脉冲个 数,当输出脉冲个数达到指定值时,驱动程序自 ...
- 不一样的角度 解读微信小程序
不一样的角度 解读微信小程序 七月在夏天· 2 天前 前段时间看完了雨果奖中短篇获奖小说<北京折叠>.很有意思的是,张小龙最近也要把应用折叠到微信里,这些应用被他称为:小程序. 含着金钥匙 ...
- C语言的关键字,标示符以及数据类型
1. 关键字 1> 关键字就是C语言提供的有特殊含义的符号,也叫做“保留字” 2> C语言一共提供了32个关键字,这些关键字都被C语言赋予了特殊含义 auto doub ...
- 自动生成Model层中对应表的各个字段
select 'public '+ case t.name when 'varchar' then 'string' when 'smallint' then 'Int16' when 'int' t ...
- Trie树(c++实现)
转:http://www.cnblogs.com/kaituorensheng/p/3602155.html http://blog.csdn.net/insistgogo/article/detai ...
- 【转】简易smtp调用类
用PHP发邮件的一个类,无需验证SMTP主机,类代码你可以不用管,只要按照后面的使用方法操作就行,这个类要先保存成一个php文件,文件名就叫smtp.php吧,下面是该文件代码: <?php c ...
- 字符串s中从第i个位置起取长度为len的子串,函数返回子串链表
/*已知字符串采用带结点的链式存储结构(详见linksrting.h文件),请编写函数linkstring substring(linkstring s,int i,int len),在字符串s中从第 ...
- GDB常用命令
一. gdb使用流程 1.编译生成可执行文件 gcc -g hello.c -o hello 2.启动gdb gdb hello 3. 在main处设置断点 break main 4.运行程序 run ...
- CodeForces 743B Chloe and the sequence (递归)
题意:给定n和k,求第n个序列中第k个数是多少,序列是这样构造,原来只有1,然后再copy一遍放在到后面再在中间放上一个没有出现过的最小整数,就变成了 121,下次就成了1213121. 析:很明显是 ...