【Go】聊天服务器

package main

import(
	"bufio"
	"fmt"
	"log"
	"net"
)

type client chan <- string

var(
	entering = make(chan client)
	leaving = make(chan client)
	messages = make(chan string)
)

func broadcaster(){
	clients := make(map[client]string)
	for{
		select{
		case msg := <- messages:
			for cli := range clients{
				cli <- msg
			}
		case cli := <- entering:
			clients[cli] = true
		case cli := <- leaving:
			delete(clients, cli)
			close(cli)
		}
	}
}

func handleConn(conn net.Conn){
	ch := make(chan string)
	go clientWriter(conn, ch)
	who := conn.RemoteAddr().String()
	ch <- "You are " + who
	messages <- who + " has arrived"
	entering <- ch

	input := bufio.NewScanner(conn)
	for input.Scan(){
		messages <- who + ": " + input.Text()
	}
	leaving <- ch
	messages <- who + " has left"
	conn.Close()
}

func clientWriter(conn net.Conn, ch <-chan string){
	for msg := range ch{
		fmt.Fprintln(conn, msg)
	}
}

func main(){
	listener, err := net.Listen("tcp", "localhost:8080")
	if err != nil{
		log.Fatal(err)
	}
	go broadcaster()
	for{
		conn, err := listener.Accept()
		if err != nil{
			log.Print(err)
			continue
		}
		go handleConn(conn)
	}
}
  1. 定义client为只能发送string类型的通道
  2. 定义entering和leaving为传输client类型的通道,嵌套定义通道,用于标记用户
  3. message为普通的string通道,用于发送和接收消息
  4. 主线程通过监听,接收每个用户,当有用户进入,为每个用户创建一个线程handleConn
  5. handleConn中通过创建string类型通道,作为entering和leaving的通道的通道的输入,标记用户进入和离开,同时通过message传输消息
  6. 在handleConn中通过给每个用户创造一个clientWriter线程,打印广播器的输出
  7. handleConn函数通过input.Scan阻塞,一直接收输入,直到连接断开
  8. 定义广播器broadcaster,使用clients记录用户,当有消息进入message通道,则广播这个消息给每个用户,每个用户都是可以发送string类型的通道,通过clientWriter接收这个通道并打印
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值