Skip to main content
 首页 » 编程设计

使用go实现责任链模式

2022年07月19日132duanxz

责任链模式

定义

责任链模式(Chain Of Responsibility):使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的藕合关系。将这个对象连成一条链,并沿着这条链传递该请求,直到一个对象处理完为止。

是一种处理请求的模式,它让多个处理器都有机会处理该请求,直到其中某个处理成功为止。责任链模式把多个处理器串成链,然后让请求在链上传递。

举个栗子

比如学生请假,小于等于 2 天班主任可以审批;小于等于 7 天,系主任可以审批;小于或等于 10 天,院长可以批准。其他情况,不予审批。

用责任链模式设计审批流程,每个审核者只关心自己责任范围内的请求,并且处理它。对于超出自己责任范围的,扔给下一个审核者处理,这样,将来继续添加审核者的时候,不用改动现有逻辑。

responsibility

优点

1、降低耦合度。它将请求的发送者和接收者解耦;

2、简化了对象。使得对象不需要知道链的结构;

3、增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任;

4、增加新的请求处理类很方便。

缺点

不能保证请求一定被接收

适用范围

1、多个对象可以处理一个请求,但具体由哪个对象处理该请求在运行时自动确定;

2、可动态指定一组对象处理请求,或添加新的处理者;

3、需要在不明确指定请求处理者的情况下,向多个处理者中的一个提交请求。

代码实现

上面学生请假的栗子:

学生请假,小于等于 2 天班主任可以审批;小于等于 7 天,系主任可以审批;小于或等于 10 天,院长可以批准。其他情况,不予审批。

type Teacher interface { 
	HaveRight(day int) bool 
	HandleApproveRequest(name string, day int) bool 
} 
 
type RequestChain struct { 
	Teacher 
	approver *RequestChain 
} 
 
func (r *RequestChain) SetApprover(m *RequestChain) { 
	r.approver = m 
} 
 
func (r *RequestChain) HandleApproveRequest(name string, day int) bool { 
	if r.Teacher.HaveRight(day) { 
		return r.Teacher.HandleApproveRequest(name, day) 
	} 
	if r.approver != nil { 
		return r.approver.HandleApproveRequest(name, day) 
	} 
	fmt.Println("请假时间太久了,不予审批") 
	return false 
} 
 
func (r *RequestChain) HaveRight(day int) bool { 
	return true 
} 
 
type HeadTeacher struct{} 
 
func NewHeadTeacherChain() *RequestChain { 
	return &RequestChain{ 
		Teacher: &HeadTeacher{}, 
	} 
} 
 
func (*HeadTeacher) HaveRight(day int) bool { 
	return day <= 2 
} 
 
func (*HeadTeacher) HandleApproveRequest(name string, day int) bool { 
	fmt.Println(fmt.Sprintf("班主任,批准了%s的请假申请,请假天数%d", name, day)) 
	return true 
} 
 
type DepTeacher struct{} 
 
func NewDepManagerChain() *RequestChain { 
	return &RequestChain{ 
		Teacher: &DepTeacher{}, 
	} 
} 
 
func (*DepTeacher) HaveRight(day int) bool { 
	return day <= 7 
} 
 
func (*DepTeacher) HandleApproveRequest(name string, day int) bool { 
	fmt.Println(fmt.Sprintf("系主任,批准了%s的请假申请,请假天数%d", name, day)) 
	return true 
} 
 
type DeanTeacher struct{} 
 
func NewDeanTeacherChain() *RequestChain { 
	return &RequestChain{ 
		Teacher: &DeanTeacher{}, 
	} 
} 
 
func (*DeanTeacher) HaveRight(day int) bool { 
	return day <= 10 
} 
 
func (*DeanTeacher) HandleApproveRequest(name string, day int) bool { 
	fmt.Println(fmt.Sprintf("院长,批准了%s的请假申请,请假天数%d", name, day)) 
	return true 
} 

测试代码

func TestApproveChain(t *testing.T) { 
	c1 := NewHeadTeacherChain() 
	c2 := NewDepManagerChain() 
	c3 := NewDeanTeacherChain() 
 
	c1.SetApprover(c2) 
	c2.SetApprover(c3) 
 
	var c Teacher = c1 
	assert.Equal(t, true, c.HandleApproveRequest("小明", 3)) 
	assert.Equal(t, true, c.HandleApproveRequest("小红", 2)) 
	assert.Equal(t, false, c.HandleApproveRequest("小龙", 30)) 
} 

输出

系主任,批准了小明的请假申请,请假天数3 
班主任,批准了小红的请假申请,请假天数2 
请假时间太久了,不予审批 

责任链模式对比装饰模式

责任链和装饰模式非常相似。两者都依赖递归组合将需要执行的操作传递给一系列对象。来看下两者的不同点。

1、责任链的管理者可以相互独立地执行一切操作,还可以随时停止传递请求。

2、装饰模式可以对对象进行扩展,但是不能中断请求的传递。

参考

【文中代码】https://github.com/boilingfrog/design-pattern-learning/tree/master/责任链模式
【大话设计模式】https://book.douban.com/subject/2334288/
【极客时间】https://time.geekbang.org/column/intro/100039001
【责任链模式(责任链模式)详解】http://c.biancheng.net/view/1383.html
【责任链模式】https://boilingfrog.github.io/2021/11/22/使用go实现责任链模式/


本文参考链接:https://www.cnblogs.com/ricklz/p/15586978.html
阅读延展