[TOC]
# 简介
网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.socket也具有一个类似打开文件的函数调用:socket(),该函数返回有一个整形的socket描述符,随后的连接建立,数据传输等操作都是通过它实现的
常用的socket类型有两种:流式socket(`SOCK_STREAM`)和数据报式socket(`SOCK_DGRAM`)
* 流式是一种面向连接的socket,针对面向连接的tcp服务应用
* 数据报式socket是一种无连接的socket,对应于无连接的udp服务应用
![](https://box.kancloud.cn/d237f0f03353c1401e665eea53e5040f_1190x1260.png)
**listen创建的socket不用于通信,用于监听地址的,accept的socket是用来通信的**
![](https://box.kancloud.cn/2076d9f847b34c6783e6c6fb538553cf_1203x440.png)
read读服务/客户,有返回0,对端关闭和你连接了
# 函数
**server端**
Listen函数
~~~
func Listen(network, address string) (Listener, error)
~~~
network:tcp或者udp
address:ip端口号比如127.0.0.1:9194或者:8005
Listener接口
~~~
type Listener interface {
Accept() (Conn, error)
Close() error
Addr() Addr
}
~~~
Conn接口
~~~
type Conn interface {
Read(b []byte) (n int, err error)
Write(b []byte) (n int, err error)
Close() error
LocalAddr() Addr
RemoteAddr() Addr
SetDeadline(t time.Time) error
SetReadDeadline(t time.Time) error
SetWriteDeadline(t time.Time) error
}
~~~
# tcp服务代码
~~~
func main() {
listener, err := net.Listen("tcp", "0.0.0.0:8183")
if err != nil {
fmt.Println("err = ", err)
return
}
//退出前把监听关闭
defer listener.Close()
//阻塞等待用户链接
for {
conn, err := listener.Accept()
if err != nil {
fmt.Println("err = ", err)
continue
}
//接收用户的请求
//1024缓冲区
buf := make([]byte, 1024)
n, err1 := conn.Read(buf)
if err1 != nil {
fmt.Println("err1 = ", err1)
continue
}
fmt.Println("buf = ", string(buf[:n]))
//关闭当前用户的连接
defer conn.Close()
}
}
~~~
# tcp客户端
~~~
func main() {
conn, err := net.Dial("tcp", "127.0.0.1:8183")
if err != nil {
fmt.Println("err = ", err)
return
}
defer conn.Close()
//发送数据
conn.Write([]byte("are u ok?"))
}
~~~
# 服务端多人
用nc做客户端,telnet做客户端好像会多传东西
~~~
func HandleConn(conn net.Conn) {
//获取客户端的网络地址信息
addr := conn.RemoteAddr().String()
fmt.Println("连接成功: ", addr)
//关闭连接
defer conn.Close()
buf := make([]byte, 2048)
for {
//读取用户数据,如果客户端关闭了,好像是能发现这边的n是等于0的
n, err := conn.Read(buf)
if err != nil {
//客户端意外关闭,也能发现
fmt.Println("err = ", err)
return
}
//打印数据
fmt.Println("数据是: ", string(buf[:n]))
//看里面多了哪些东西
fmt.Println("长度是: ", len(string(buf[:n])))
//因为客户端传递来的有\n
if "exit" == string(buf[:n-1]) {
fmt.Println(addr, " exit")
return
}
//把数据转化为大写再发送
conn.Write([]byte(strings.ToUpper(string(buf[:n]))))
}
}
func main() {
listener, err := net.Listen("tcp", "0.0.0.0:8183")
if err != nil {
fmt.Println("err = ", err)
return
}
//退出前把监听关闭
defer listener.Close()
//阻塞等待用户链接
for {
conn, err := listener.Accept()
if err != nil {
fmt.Println("err = ", err)
return
}
//处理用户请求
go HandleConn(conn)
}
}
~~~
# 可输入回复客户端
~~~
func main() {
conn, err := net.Dial("tcp", "127.0.0.1:8183")
if err != nil {
fmt.Println("net.Dial err = ", err)
return
}
//main调用完毕,关闭连接
defer conn.Close()
//接收服务器回复的数据
go func() {
//从键盘输入内容,给服务器发送内容
str := make([]byte, 1024)
for {
n, err := os.Stdin.Read(str)
if err != nil {
fmt.Println("错误信息是: ", err)
return
}
//把输入的内容发给服务器
conn.Write(str[:n])
}
}()
//切片缓冲
buf := make([]byte, 1024)
for {
//如果服务端关闭和你连接,这边的n是能读取到0的
n, err := conn.Read(buf)
if err != nil {
fmt.Println("错误信息是: ", err)
return
}
fmt.Println(string(buf[:n]))
}
}
~~~
- 基础
- 简介
- 主要特征
- 变量和常量
- 编码转换
- 数组
- byte与rune
- big
- sort接口
- 和mysql类型对应
- 函数
- 闭包
- 工作区
- 复合类型
- 指针
- 切片
- map
- 结构体
- sync.Map
- 随机数
- 面向对象
- 匿名组合
- 方法
- 接口
- 权限
- 类型查询
- 异常处理
- error
- panic
- recover
- 自定义错误
- 字符串处理
- 正则表达式
- json
- 文件操作
- os
- 文件读写
- 目录
- bufio
- ioutil
- gob
- 栈帧的内存布局
- shell
- 时间处理
- time详情
- time使用
- new和make的区别
- container
- list
- heap
- ring
- 测试
- 单元测试
- Mock依赖
- delve
- 命令
- TestMain
- path和filepath包
- log日志
- 反射
- 详解
- plugin包
- 信号
- goto
- 协程
- 简介
- 创建
- 协程退出
- runtime
- channel
- select
- 死锁
- 互斥锁
- 读写锁
- 条件变量
- 嵌套
- 计算单个协程占用内存
- 执行规则
- 原子操作
- WaitGroup
- 定时器
- 对象池
- sync.once
- 网络编程
- 分层模型
- socket
- tcp
- udp
- 服务端
- 客户端
- 并发服务器
- Http
- 简介
- http服务器
- http客户端
- 爬虫
- 平滑重启
- context
- httptest
- 优雅中止
- web服务平滑重启
- beego
- 安装
- 路由器
- orm
- 单表增删改查
- 多级表
- orm使用
- 高级查询
- 关系查询
- SQL查询
- 元数据二次定义
- 控制器
- 参数解析
- 过滤器
- 数据输出
- 表单数据验证
- 错误处理
- 日志
- 模块
- cache
- task
- 调试模块
- config
- 部署
- 一些包
- gjson
- goredis
- collection
- sjson
- redigo
- aliyunoss
- 密码
- 对称加密
- 非对称加密
- 单向散列函数
- 消息认证
- 数字签名
- mysql优化
- 常见错误
- go run的错误
- 新手常见错误
- 中级错误
- 高级错误
- 常用工具
- 协程-泄露
- go env
- gometalinter代码检查
- go build
- go clean
- go test
- 包管理器
- go mod
- gopm
- go fmt
- pprof
- 提高编译
- go get
- 代理
- 其他的知识
- go内存对齐
- 细节总结
- nginx路由匹配
- 一些博客
- redis为什么快
- cpu高速缓存
- 常用命令
- Go 永久阻塞的方法
- 常用技巧
- 密码加密解密
- for 循环迭代变量
- 备注
- 垃圾回收
- 协程和纤程
- tar-gz
- 红包算法
- 解决golang.org/x 下载失败
- 逃逸分析
- docker
- 镜像
- 容器
- 数据卷
- 网络管理
- 网络模式
- dockerfile
- docker-composer
- 微服务
- protoBuf
- GRPC
- tls
- consul
- micro
- crontab
- shell调用
- gorhill/cronexpr
- raft
- go操作etcd
- mongodb