5.4 使用Zinx-V0.5完成应用程序

现在我们可以基于Zinx框架完成发送msg功能的测试用例了。

Server.go

  1. package main
  2. import (
  3. "fmt"
  4. "zinx/ziface"
  5. "zinx/znet"
  6. )
  7. //ping test 自定义路由
  8. type PingRouter struct {
  9. znet.BaseRouter
  10. }
  11. //Test Handle
  12. func (this *PingRouter) Handle(request ziface.IRequest) {
  13. fmt.Println("Call PingRouter Handle")
  14. //先读取客户端的数据,再回写ping...ping...ping
  15. fmt.Println("recv from client : msgId=", request.GetMsgID(), ", data=", string(request.GetData()))
  16. //回写数据
  17. err := request.GetConnection().SendMsg(1, []byte("ping...ping...ping"))
  18. if err != nil {
  19. fmt.Println(err)
  20. }
  21. }
  22. func main() {
  23. //创建一个server句柄
  24. s := znet.NewServer()
  25. //配置路由
  26. s.AddRouter(&PingRouter{})
  27. //开启服务
  28. s.Serve()
  29. }

当前Server端是先把客户端发送来Msg解析,然后返回一个MsgId为1的消息,消息内容是"ping…ping…ping"

Client.go

  1. package main
  2. import (
  3. "fmt"
  4. "io"
  5. "net"
  6. "time"
  7. "zinx/znet"
  8. )
  9. /*
  10. 模拟客户端
  11. */
  12. func main() {
  13. fmt.Println("Client Test ... start")
  14. //3秒之后发起测试请求,给服务端开启服务的机会
  15. time.Sleep(3 * time.Second)
  16. conn,err := net.Dial("tcp", "127.0.0.1:7777")
  17. if err != nil {
  18. fmt.Println("client start err, exit!")
  19. return
  20. }
  21. for {
  22. //发封包message消息
  23. dp := znet.NewDataPack()
  24. msg, _ := dp.Pack(znet.NewMsgPackage(0,[]byte("Zinx V0.5 Client Test Message")))
  25. _, err := conn.Write(msg)
  26. if err !=nil {
  27. fmt.Println("write error err ", err)
  28. return
  29. }
  30. //先读出流中的head部分
  31. headData := make([]byte, dp.GetHeadLen())
  32. _, err = io.ReadFull(conn, headData) //ReadFull 会把msg填充满为止
  33. if err != nil {
  34. fmt.Println("read head error")
  35. break
  36. }
  37. //将headData字节流 拆包到msg中
  38. msgHead, err := dp.Unpack(headData)
  39. if err != nil {
  40. fmt.Println("server unpack err:", err)
  41. return
  42. }
  43. if msgHead.GetDataLen() > 0 {
  44. //msg 是有data数据的,需要再次读取data数据
  45. msg := msgHead.(*znet.Message)
  46. msg.Data = make([]byte, msg.GetDataLen())
  47. //根据dataLen从io中读取字节流
  48. _, err := io.ReadFull(conn, msg.Data)
  49. if err != nil {
  50. fmt.Println("server unpack data err:", err)
  51. return
  52. }
  53. fmt.Println("==> Recv Msg: ID=", msg.Id, ", len=", msg.DataLen, ", data=", string(msg.Data))
  54. }
  55. time.Sleep(1*time.Second)
  56. }
  57. }

这里Client客户端,模拟了一个MsgId为0的"Zinx V0.5 Client Test Message"消息,然后把服务端返回的数据打印出来。

我们分别在两个终端运行

  1. $go run Server.go
  1. $go run Client.go

服务端结果:

  1. $ go run Server.go
  2. Add Router succ!
  3. [START] Server name: zinx v-0.5 demoApp,listenner at IP: 127.0.0.1, Port 7777 is starting
  4. [Zinx] Version: V0.4, MaxConn: 3, MaxPacketSize: 4096
  5. start Zinx server zinx v-0.5 demoApp succ, now listenning...
  6. Reader Goroutine is running
  7. Call PingRouter Handle
  8. recv from client : msgId= 0 , data= Zinx V0.5 Client Test Message
  9. Call PingRouter Handle
  10. recv from client : msgId= 0 , data= Zinx V0.5 Client Test Message
  11. Call PingRouter Handle
  12. recv from client : msgId= 0 , data= Zinx V0.5 Client Test Message
  13. ...

客户端结果:

  1. $ go run Client.go
  2. Client Test ... start
  3. ==> Recv Msg: ID= 1 , len= 18 , data= ping...ping...ping
  4. ==> Recv Msg: ID= 1 , len= 18 , data= ping...ping...ping
  5. ==> Recv Msg: ID= 1 , len= 18 , data= ping...ping...ping
  6. ...

好了,我们的Zinx已经成功的集成消息的封装功能了,这样我们就有Zinx的通信的基本协议标准了。