七叶笔记 » golang编程 » net包 dial – golang

net包 dial – golang

最近一直和硬件打交道,而和这些硬件交互很多用到了 socket 。于是在使用的过程中同时简单的学习了一下net包的dial。

在go中,大致提供了5种dial,包括:

1. Dial(network, address string) (Conn, error)

 2. Dial tcp (network string, laddr, raddr *TCPAddr) (*TCPConn, error)

 3. DialUDP(network string, laddr, raddr *UDPAddr) (*UDPConn, error)

 4. DialIP(network string, laddr, raddr *IPAddr) (*IPConn, error)

 5. DialUnix(network string, laddr, raddr *UnixAddr) (*UnixConn, error)  

其中dial是对其他几个的封装,同时实现了Conn接口。

而其他几个,虽然返回TCPConn,UDPConn,IPConn,UnixConn。但是他们的底层是一样的如下图:

最终这几个返回都是对netfd结构体的返回。而且除了DialUnix 其余三个最终都是走的 internetSocket() 只不过在调用这个方法前,各自的处理有一些不同。

而DialUnix最终调用的是一个 unixSocket()。

如果大家想看具体实现,可以去tcpsock,udpsock,ipsock,unixsock这几个文件看看。

因为他们基本都有相同的实现,和相同的返回。所以他们的使用也是基本相同,下面就以dial,tcpdial为例。

//创建连接

 	conn,err := net.Dial("tcp","192.168.1.254:4001")

 	//发送数据

 	conn.Write([] byte {0x00,0x07,0x00,0x00,0x00,0x06,0x01,0x03,0x00,0x00,0x00,0x0A})

 	defer conn. Close ()

 	buffer := make([]byte,32)

 	//读取返回数据

 	result,err := conn.Read(buffer)

 	if err !=  nil  {

 		fmt.Println(err)

 	}

 	fmt.Println(result,buffer)

 //进行地址化转换

 	tcpAddr,err:=net.ResolveTCPAddr("tcp","192.168.1.254:4001")

 	//创建一个连接

 	conn,err:=net.DialTCP("tcp",nil,tcpAddr)

 	if err != nil {

 		fmt.Println(err)

 	}

 	str := []byte{32, 0, 0, 0, 7, 85, 35, 160, 176, 7, 226, 12, 18, 15, 45, 0}

 

 	//向连接端发送一个数据

 	_,err=conn.Write(str)

 	if err != nil {

 		fmt.Println(err)

 	}

 	//读取连接端返回来的数据,

 	result,err:=ioutil. ReadAll (conn)

 	if err != nil {

 		fmt.Println(err)

 	}

 	fmt.Println(string(result))  

从这两个请求可以看出,dial只需要直接填入连接地址,而dialtcp却需要提前把地址转换为对应的类型才能填入。在dial中,连接会根据你填入的连接类型,自动把地址转换为对应类型地址。

同时,我们还会发现我们dial中用了buffer接收数据,而在dialtcp用了ioutil.ReadAll()返回所有数据。

这两种方法有什么不同,第一种是把buffer填充完后就直接执行后面的代码了,但是buffer的你需要事先定义大小。第二种ioutil.ReadAll()要是拿不到结束符,会一直等到连接超时才执行后面代码。这样就会导致阻塞很长时间。

相关文章