Channel扇出模式

文章目录

      • 扇出模式
        • [reflectSelect 方式](#reflectSelect 方式)

扇出模式

有扇入模式,就有扇出模式,扇出模式是和扇入模式相反的。扇出模式只有一个输入源 Channel,有多个目标 Channel,扇出比就是 1 比目标 Channel 数的值,经常用在设计模式中的观察者模式中(观察者设计模式定义了对象间的一种一对多的组合关系。这样一来,一个对象的状态发生变化时,所有依赖于它的对象都会得到通知并自动刷新)。在观察者模式中,数据变动后,多个观察者都会收到这个变更信号。下面是一个扇出模式的实现。从源 Channel 取出一个数据后,依次发送给目标 Channel。在发送给目标 Channel 的时候,可以同步发送,也可以异步发送:

go 复制代码
// 扇出模式只有一个输入源 Channel,有多个目标 Channel,扇出比就是 1 比目标 Channel 数的值
func fanOut(ch <-chan interface{}, out []chan interface{}, async bool) {
    go func() {
        defer func() { //退出时关闭所有的输出chan
            for i := 0; i < len(out); i++ {
                close(out[i])
            }
        }()

        for v := range ch { // 从输入chan中读取数据
            v := v // 局部变量 v,以便在循环内部的闭包中使用
            for i := 0; i < len(out); i++ {
                i := i // 一个局部变量 i,以便在循环内部的闭包中使用
                if async {
                    go func() {
                        out[i] <- v // 放入到输出chan中,异步方式
                    }()
                } else {
                    out[i] <- v // 放入到输出chan中,同步方式
                }
            }
        }
    }()

}
reflectSelect 方式
go 复制代码
// 实现扇出模式的反射方式,将数据从一个通道发送到多个目标通道
func fanOutReflect2(ch <-chan interface{}, out []chan interface{}) {
	// 输入通道输出通道同时在一个cases 中
	cases := make([]reflect.SelectCase, len(out)+1)

	// Create a case for the input channel
	// 接收通道,监听,有数据即可发送
	cases[0] = reflect.SelectCase{
		Dir:  reflect.SelectRecv, // case <-Chan: 从一个通道发出
		Chan: reflect.ValueOf(ch),
	}

	// Create cases for the output channels
	// 发送通道,监听,
	for i := 0; i < len(out); i++ {
		cases[i+1] = reflect.SelectCase{
			Dir:  reflect.SelectSend, // Chan <- Send: 发到多个目标通道
			Chan: reflect.ValueOf(out[i]),
		}
	}

	for {
		chosen, recv, _ := reflect.Select(cases)
		if chosen == 0 {
			// Data received from the input channel 从输入通道接收的数据
			v := recv.Interface()
			for i := 1; i < len(cases); i++ {
				// Send the data to all output channels 将数据发送到所有输出通道
				cases[i].Send = reflect.ValueOf(v)
			}
		} else {
			// Data sent to an output channel 发送到输出通道的数据
			// 把接收通道的数据置空
			cases[chosen].Send = reflect.Value{}
		}
	}
}
相关推荐
27669582922 天前
阿里1688 阿里滑块 231滑块 x5sec分析
java·python·go·验证码·1688·阿里滑块·231滑块
Moment3 天前
在 NodeJs 中如何通过子进程与 Golang 进行 IPC 通信 🙄🙄🙄
前端·后端·go
唐僧洗头爱飘柔95273 天前
(Go基础)变量与常量?字面量与变量的较量!
开发语言·后端·golang·go·go语言初上手
黑心萝卜三条杠3 天前
【Go语言】深入理解Go语言:并发、内存管理和垃圾回收
google·程序员·go
不喝水的鱼儿3 天前
【LuatOS】基于WebSocket的同步请求框架
网络·websocket·网络协议·go·luatos·lua5.4
微刻时光4 天前
程序员开发速查表
java·开发语言·python·docker·go·php·编程语言
lidenger4 天前
服务认证-来者何人
后端·go
幼儿园老大*5 天前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
童先生5 天前
Go 项目中实现类似 Java Shiro 的权限控制中间件?
开发语言·go
幼儿园老大*5 天前
走进 Go 语言基础语法
开发语言·后端·学习·golang·go