~~网络编程(四):socket套接字~~
进击のpython
网络编程——socket
socket的中文意思叫做套接字,socket方法其实也叫套接字方法
我们研究过TCP/UDP协议,但是要是让我们自己搭建,就十分困难了
而这时候,socket就出来了
socket他是存在在应用层和传输层之间的一堆接口
他把复杂的协议都封装好了,你想用的时候,直接调用就行
就跟我们写模块一样,把麻烦的代码写进去,等到用的时候只需要调用就行了
那你想想,咱们在研究模块的时候,是不是学会方法怎么用就行??
那咱们学套接字方法,也是学会方法就行
我要是想给你打电话的话应该是怎么个过程?
# 首先我得有电话
# 输入想打的电话
# 等待你接听
# 跟你bb两句
# 听你bb两句
那你接电话呢?
# 首先你得有电话
# 你还得有电话号码
# 然后你得有电话线
# 等着接电话
# 铃响接电话
# 听你bb两句
没问题吧!
那我要是想写成socket通信呢?
首先我得先创建两个py文件是吧,通信嘛,当然要两个:客户端和服务端
然后要在两个文件都调用socket模块是吧
接下来就继续吧!
解释一下啊,socket里面的两个属性,第一个是套接字的类型,我们这个是基于网络的嘛,所以是AF_INET
第二个是你数据的传输方式,SOCK_STREAM是流式传输,也就是TCP协议
在这我们要学习一个方法 bind 绑定,因为是手机绑定,所以是phone.bind
里面要添加一个元组参数 IP 和 端口
其中ip 我设置的是127.0.0.1,代表着本机的意思,我这不是要在自己电脑上操作嘛
端口最大到65535 其中,1-1024 是系统的端口,1024之后的就是应用的了,你随便选个就行
这里就是又一个方法了 listen 监听
里面要传一个参数,这个5是什么意思呢?
我这个电话接通的时候,别人又给我打电话,我也不能挂了啊
那这种状态可以维持几个手机同时给我打电话呢?5个
那最后,我就开始等电话了是吧
那客户端呢?是不是要买手机啊我用绑定ip嘛?
换句话说我用手机绑定手机卡,说只能用这个手机的这张卡才能给你打电话???
这显然不对啊,所以,客户端不用绑定
![1565065172405](D:\python 学习笔记\assets\1565065172405.png)
客户端的拨号,用到了connet方法,里面放一个元组参数,元组由服务端的ip和端口构成
那我要是运行,是不是应该先运行服务端,再运行客户端.这个能理解吧
那我们执行一下
看到了吧,这就算是连接上了,那我们看看我们拿的是什么东西
(<socket.socket fd=484, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 52621)>, ('127.0.0.1', 52621))
首先可以肯定这是个元组是吧,第一个元素是个什么东西?是不是套接字对象啊
第二个元素呢?是不是客户端的ip以及端口信息啊
那我们是不是可以通过解构的方式把这两个信息拿到啊
connet, client_addr = phone.accept()
那你说我这是不是就算是拿到了管道啊,那我有了管道了,我是不是就可以进行传输了啊
那我就可以收信息了,怎么收?又用到一个方法recv
connet.recv(1024)
它里面要传个参数,传接受的最大字节数
你给我传一个,我就收一个,你要是给我传1025个,我也收1024个
至于为什么是1024呢?我们后面说,你现在就先这么写
那我要是发消息呢?send方法
connet.send()
里面传发送的东西
打完电话就该挂电话了对吧
connet.close()
挂完电话是不是就关机了啊
phone.close()
那服务端完事了,客户端是不是也应该写点什么了
服务端是接收-发送
那我客户端是不是应该是发送-接收
客户端发送怎么写?send
phone.send("hello".encode("utf-8"))
别忘了把编码缀上
接收呢??recv
k = phone.recv(1024)
print(k)
然后呢?关机!
phone.close()
行啦!我们来看看写完的长什么样
执行一下吧(别忘了先执行服务端,再执行客户端)
这样,就完成了一个简单的C/S结构!
话不多说,自己多敲几遍
正常人谁那么搞啊,不都是有来言有去语的吗
那我们是不是就应该用循环来搞这个需求
循环放在哪呢?我们不是要循环收发这个过程嘛
那我们就应该循环收发信息这一块的代码啊
# 服务端
import socket
# 买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 绑定手机卡
phone.bind(("127.0.0.1", 8080))
# 开机
phone.listen(5)
# 等电话
connet, client_addr = phone.accept()
# 收发消息
while 1:
k = connet.recv(1024)
print(f'从客户端接收的消息:{k}')
connet.send(k.upper())
# 挂电话
connet.close()
# 关机
phone.close()
# 客户端
import socket
# 买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 拨号
phone.connect(("127.0.0.1", 8080))
# 发收信息
while 1:
msg = input(">>>")
phone.send(msg.encode("utf-8"))
k = phone.recv(1024)
print(f"从服务端接收的消息:{k}")
# 关闭
phone.close()
# 服务端
从客户端接收的消息:b'as'
从客户端接收的消息:b'ds'
从客户端接收的消息:b'fa'
# 客户端
>>>as
从服务端接收的消息:b'AS'
>>>ds
从服务端接收的消息:b'DS'
>>>fa
从服务端接收的消息:b'FA'
>>>
还有一点就是想说,你们在调试的时候没遇到这个报错吗?
OSError:[Errno 48] Address already in use
这个问题就是说你刚关那个程序,然后系统还没有把你定义的端口给回收,你就又用了,就会出这个错误
怎么解决呢?
一种办法是修改客户端和服务端的端口,但是你不觉得太傻逼了吗?
所以还有另一种方法,就是在程序服务端绑定ip和端口下面添加一段这个代码
phone.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
这样就没事了,这个就是如果我发现这个端口被占用了,我就重新启用他
不是,那你们没有遇到bug嘛?
比如我在服务端直接敲回车,结果是什么???
你会发现你的程序执行不了了
那是谁的问题呢?是没发过去还是没接过来呢?
我们在send的下面写一个打印,发现可以被打印,那就说明我发过去了
而当我在接收的下面写一个打印,发现没有被打印,说明我没有接受到
而且服务端也没有执行原来有的print
那就说明是我发过去了,但是他什么都没有接收,所以她始终都在接受的这个位置等我传
也就是说空字符串是没办法单独传过去的,怎么解决这个问题呢?加个判断就好了啊
if not msg:continue
这样bug就没了
还有问题!你在断开客户端的时候,你会发现,服务端报错了!
ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。
这是我们不能忍的啊,我不跟你链接了,你就报错?这不符合常识
所以我们就可以用异常捕获来处理
try:
k = connet.recv(1024)
print(f'从客户端接收的消息:{k}')
connet.send(k.upper())
except ConnectionResetError:
break
这样就没问题了!
这里多说一句啊,我这个代码是在windows里运行的,如果你是linux运行的
服务端是不会报错的,他会不停的接收一个空,一直循环不报错
这样就会使程序变成死循环!,这是我们所不能接受的!
所以,在服务端加个判断
if not k:break
这样就万无一失了!
毕竟正常人不会发送个空过来!