Advanced Development

For short connections, each data send and receive operation closes the connection, resulting in relatively simple connection handling logic but lower communication efficiency. In most TCP communication scenarios, long connection operations are often used, with an asynchronous full-duplex TCP communication mode, meaning all communication is entirely asynchronous. In such scenarios, the gtcp.Conn link object may simultaneously be in multiple read and write operations (data read and write operations of the gtcp.Conn object are concurrency-safe), hence SendRecv operations will logically fail. Because after sending data in the same logical operation, immediately retrieving data might result in receiving results from other write operations.

Both the server and client need to encapsulate the use of the Recv* method to obtain data in an independent asynchronous loop and handle data through switch...case... (fully responsible for reading data in one goroutine), forwarding business processing based on the request data.

TCP Object - Senior - 图1tip

That is, Send*/Recv* methods are concurrency-safe, but data should be sent in one go. Because asynchronous concurrent writing is supported, the gtcp.Conn object is implemented without any buffering.

Example

We’ll illustrate how to implement asynchronous full-duplex communication in a program with a complete example, found at: https://github.com/gogf/gf/v2/tree/master/.example/net/gtcp/pkg_operations/common

  1. types/types.go

Define the data format for communication so we can use SendPkg/RecvPkg methods for communication.

For simplification of test code complexity, we’ll use the JSON data format to pass data here. In some scenarios where the message package size is strictly considered, the Data field can be encapsulated and parsed in binary as needed. Also, note that even using JSON data format, the Act field often defines constants for implementation, and uint8 type suffices for most cases to reduce message package size. Here, we use strings for demonstration purposes.

  1. package types
  2. type Msg struct {
  3. Act string // Operation
  4. Data string // Data
  5. }
  1. funcs/funcs.go

Custom definitions for sending/receiving data in defined formats, facilitating data structure encoding/parsing.

  1. package funcs
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "github.com/gogf/gf/v2/net/gtcp"
  6. "github.com/gogf/gf/.example/net/gtcp/pkg_operations/common/types"
  7. )
  8. // Custom format message package sending
  9. func SendPkg(conn *gtcp.Conn, act string, data...string) error {
  10. s := ""
  11. if len(data) > 0 {
  12. s = data[0]
  13. }
  14. msg, err := json.Marshal(types.Msg{
  15. Act : act,
  16. Data : s,
  17. })
  18. if err != nil {
  19. panic(err)
  20. }
  21. return conn.SendPkg(msg)
  22. }
  23. // Custom format message package receiving
  24. func RecvPkg(conn *gtcp.Conn) (msg *types.Msg, err error) {
  25. if data, err := conn.RecvPkg(); err != nil {
  26. return nil, err
  27. } else {
  28. msg = &types.Msg{}
  29. err = json.Unmarshal(data, msg)
  30. if err != nil {
  31. return nil, fmt.Errorf("invalid package structure: %s", err.Error())
  32. }
  33. return msg, err
  34. }
  35. }
  1. gtcp_common_server.go

Communication server. In this example, the server does not actively disconnect, but sends a doexit message to the client after 10 seconds, notifying the client to disconnect to end the example.

  1. package main
  2. import (
  3. "github.com/gogf/gf/v2/net/gtcp"
  4. "github.com/gogf/gf/v2/os/glog"
  5. "github.com/gogf/gf/v2/os/gtimer"
  6. "github.com/gogf/gf/.example/net/gtcp/pkg_operations/common/funcs"
  7. "github.com/gogf/gf/.example/net/gtcp/pkg_operations/common/types"
  8. "time"
  9. )
  10. func main() {
  11. gtcp.NewServer("127.0.0.1:8999", func(conn *gtcp.Conn) {
  12. defer conn.Close()
  13. // Test message, let the client exit voluntarily after 10 seconds
  14. gtimer.SetTimeout(10*time.Second, func() {
  15. funcs.SendPkg(conn, "doexit")
  16. })
  17. for {
  18. msg, err := funcs.RecvPkg(conn)
  19. if err != nil {
  20. if err.Error() == "EOF" {
  21. glog.Println("client closed")
  22. }
  23. break
  24. }
  25. switch msg.Act {
  26. case "hello": onClientHello(conn, msg)
  27. case "heartbeat": onClientHeartBeat(conn, msg)
  28. default:
  29. glog.Errorf("invalid message: %v", msg)
  30. break
  31. }
  32. }
  33. }).Run()
  34. }
  35. func onClientHello(conn *gtcp.Conn, msg *types.Msg) {
  36. glog.Printf("hello message from [%s]: %s", conn.RemoteAddr().String(), msg.Data)
  37. funcs.SendPkg(conn, msg.Act, "Nice to meet you!")
  38. }
  39. func onClientHeartBeat(conn *gtcp.Conn, msg *types.Msg) {
  40. glog.Printf("heartbeat from [%s]", conn.RemoteAddr().String())
  41. }
  1. gtcp_common_client.go

Communication client. As you can see, the code structure is similar to the server, with data retrieval independently residing within a for loop. Each business logic sends message packages directly using the SendPkg method.

Heartbeat messages are typically implemented with a gtimer timer. In this example, the client sends heartbeat messages to the server every 1 second and sends a hello test message to the server after 3 seconds. These are implemented using the gtimer timer, which is common in TCP communication.

After 10 seconds of client connection, the server sends a doexit message to the client, prompting the client to disconnect, ending the long connection.

  1. package main
  2. import (
  3. "github.com/gogf/gf/v2/net/gtcp"
  4. "github.com/gogf/gf/v2/os/glog"
  5. "github.com/gogf/gf/v2/os/gtimer"
  6. "github.com/gogf/gf/.example/net/gtcp/pkg_operations/common/funcs"
  7. "github.com/gogf/gf/.example/net/gtcp/pkg_operations/common/types"
  8. "time"
  9. )
  10. func main() {
  11. conn, err := gtcp.NewConn("127.0.0.1:8999")
  12. if err != nil {
  13. panic(err)
  14. }
  15. defer conn.Close()
  16. // Heartbeat message
  17. gtimer.SetInterval(time.Second, func() {
  18. if err := funcs.SendPkg(conn, "heartbeat"); err != nil {
  19. panic(err)
  20. }
  21. })
  22. // Test message, send hello message to server after 3 seconds
  23. gtimer.SetTimeout(3*time.Second, func() {
  24. if err := funcs.SendPkg(conn, "hello", "My name's John!"); err != nil {
  25. panic(err)
  26. }
  27. })
  28. for {
  29. msg, err := funcs.RecvPkg(conn)
  30. if err != nil {
  31. if err.Error() == "EOF" {
  32. glog.Println("server closed")
  33. }
  34. break
  35. }
  36. switch msg.Act {
  37. case "hello": onServerHello(conn, msg)
  38. case "doexit": onServerDoExit(conn, msg)
  39. case "heartbeat": onServerHeartBeat(conn, msg)
  40. default:
  41. glog.Errorf("invalid message: %v", msg)
  42. break
  43. }
  44. }
  45. }
  46. func onServerHello(conn *gtcp.Conn, msg *types.Msg) {
  47. glog.Printf("hello response message from [%s]: %s", conn.RemoteAddr().String(), msg.Data)
  48. }
  49. func onServerHeartBeat(conn *gtcp.Conn, msg *types.Msg) {
  50. glog.Printf("heartbeat from [%s]", conn.RemoteAddr().String())
  51. }
  52. func onServerDoExit(conn *gtcp.Conn, msg *types.Msg) {
  53. glog.Printf("exit command from [%s]", conn.RemoteAddr().String())
  54. conn.Close()
  55. }
  1. After execution

    • Server output result

      1. 2019-05-03 14:59:13.732 heartbeat from [127.0.0.1:51220]
      2. 2019-05-03 14:59:14.732 heartbeat from [127.0.0.1:51220]
      3. 2019-05-03 14:59:15.733 heartbeat from [127.0.0.1:51220]
      4. 2019-05-03 14:59:15.733 hello message from [127.0.0.1:51220]: My name's John!
      5. 2019-05-03 14:59:16.731 heartbeat from [127.0.0.1:51220]
      6. 2019-05-03 14:59:17.733 heartbeat from [127.0.0.1:51220]
      7. 2019-05-03 14:59:18.731 heartbeat from [127.0.0.1:51220]
      8. 2019-05-03 14:59:19.730 heartbeat from [127.0.0.1:51220]
      9. 2019-05-03 14:59:20.732 heartbeat from [127.0.0.1:51220]
      10. 2019-05-03 14:59:21.732 heartbeat from [127.0.0.1:51220]
      11. 2019-05-03 14:59:22.698 client closed
    • Client output result

      1. 2019-05-03 14:59:15.733 hello response message from [127.0.0.1:8999]: Nice to meet you!
      2. 2019-05-03 14:59:22.698 exit command from [127.0.0.1:8999]