分享
Golang在windows下的socket编程(不使用net包和Cgo)
u011774512 · · 3767 次点击 · · 开始浏览这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。
go也可以像普通的socket编程那样: 创建套接字 -> 绑定 -> 监听 ...
当然,go中有net包可以使用,但是如果要实现一些比较底层的操作,例如自己构造数据包,就可以通过这种比较原始的方式来进行socket编程。
代码如下:
package main
import (
. "fmt"
"strconv"
"strings"
"syscall"
)
func MAKEWORD(low, high uint8) uint32 {
var ret uint16 = uint16(high)<<8 + uint16(low)
return uint32(ret)
}
func inet_addr(ipaddr string) [4]byte {
var (
ips = strings.Split(ipaddr, ".")
ip [4]uint64
ret [4]byte
)
for i := 0; i < 4; i++ {
ip[i], _ = strconv.ParseUint(ips[i], 10, 8)
}
for i := 0; i < 4; i++ {
ret[i] = byte(ip[i])
}
return ret
}
func main() {
var (
sock syscall.Handle
addr syscall.SockaddrInet4
wsadata syscall.WSAData
err error
)
if err = syscall.WSAStartup(MAKEWORD(2, 2), &wsadata); err != nil {
Println("Startup error")
return
}
defer syscall.WSACleanup()
if sock, err = syscall.Socket(syscall.AF_INET, syscall.SOCK_STREAM, syscall.IPPROTO_IP); err != nil {
Println("Socket create error")
return
}
defer syscall.Closesocket(sock)
addr.Addr = inet_addr("61.135.169.105")
addr.Port = 80
if err = syscall.Connect(sock, &addr); err != nil {
Println("Connect error")
return
}
var (
data syscall.WSABuf
sendstr string = "hello"
SendButes uint32
overlapped syscall.Overlapped
)
data.Len = uint32(len(sendstr))
data.Buf = syscall.StringBytePtr(sendstr)
//如果使用syscall.Sendto或syscall.Write会发送失败,原因未知
err = syscall.WSASend(sock, &data, 1, &SendButes, 0, &overlapped, nil)
if err != nil {
Println("Send error")
} else {
Println("Send success")
}
}
注意:syscall.Sendto和syscall.Write两个函数试过了会发送失败,而且我不知道什么原因,在windows环境下看来只能使用WSA****系列的函数进行发送数据包了,不过证明了Go也可以进行底层的网络编程而不借助Cgo
如果转载请注明出处:http://blog.csdn.net/gophers/article/details/18666287
有疑问加站长微信联系(非本文作者)
入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889
关注微信3767 次点击
添加一条新回复
(您需要 后才能回复 没有账号 ?)
- 请尽量让自己的回复能够对别人有帮助
- 支持 Markdown 格式, **粗体**、~~删除线~~、
`单行代码` - 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
- 图片支持拖拽、截图粘贴等方式上传
收入到我管理的专栏 新建专栏
go也可以像普通的socket编程那样: 创建套接字 -> 绑定 -> 监听 ...
当然,go中有net包可以使用,但是如果要实现一些比较底层的操作,例如自己构造数据包,就可以通过这种比较原始的方式来进行socket编程。
代码如下:
package main
import (
. "fmt"
"strconv"
"strings"
"syscall"
)
func MAKEWORD(low, high uint8) uint32 {
var ret uint16 = uint16(high)<<8 + uint16(low)
return uint32(ret)
}
func inet_addr(ipaddr string) [4]byte {
var (
ips = strings.Split(ipaddr, ".")
ip [4]uint64
ret [4]byte
)
for i := 0; i < 4; i++ {
ip[i], _ = strconv.ParseUint(ips[i], 10, 8)
}
for i := 0; i < 4; i++ {
ret[i] = byte(ip[i])
}
return ret
}
func main() {
var (
sock syscall.Handle
addr syscall.SockaddrInet4
wsadata syscall.WSAData
err error
)
if err = syscall.WSAStartup(MAKEWORD(2, 2), &wsadata); err != nil {
Println("Startup error")
return
}
defer syscall.WSACleanup()
if sock, err = syscall.Socket(syscall.AF_INET, syscall.SOCK_STREAM, syscall.IPPROTO_IP); err != nil {
Println("Socket create error")
return
}
defer syscall.Closesocket(sock)
addr.Addr = inet_addr("61.135.169.105")
addr.Port = 80
if err = syscall.Connect(sock, &addr); err != nil {
Println("Connect error")
return
}
var (
data syscall.WSABuf
sendstr string = "hello"
SendButes uint32
overlapped syscall.Overlapped
)
data.Len = uint32(len(sendstr))
data.Buf = syscall.StringBytePtr(sendstr)
//如果使用syscall.Sendto或syscall.Write会发送失败,原因未知
err = syscall.WSASend(sock, &data, 1, &SendButes, 0, &overlapped, nil)
if err != nil {
Println("Send error")
} else {
Println("Send success")
}
}
注意:syscall.Sendto和syscall.Write两个函数试过了会发送失败,而且我不知道什么原因,在windows环境下看来只能使用WSA****系列的函数进行发送数据包了,不过证明了Go也可以进行底层的网络编程而不借助Cgo
如果转载请注明出处:http://blog.csdn.net/gophers/article/details/18666287