Socket(套接字)

套接字是一个抽象层,应用程序可以通过它发送或接收数据,可对其进行像文件一样的打开、读写和关闭等操作。套接字允许应用程序将I/O插入到网络中,并与网络中的其他应用程序进行通信。网络套接字是IP地址与端口的组合。

发展:套接字最初是由加利福尼亚大学Berkely分校为Unix系统开发的网络通信接口。后来随着TCP/IP网络的发展,套接字成为最为通用的应用程序接口,也是在Internet上进行应用开发最为通用的API.

应用:

实现:

基于TCP协议实现Sever端和Client端的信息互通

Sever端:

import socket
sk = socket.socket() #创建服务器的套接字
sk.bind(('127.0.0.1',8080)) #把地址绑定到套接字 sk.listen() #监听链接
conn,addr = sk.accept() #接收到客户端的连接和地址
ret = conn.recv(1024) #接收客户端信息
print(ret) #打印客户端的信息 conn.send(b'hi') #向客户端发送信息
conn.close() #关闭客户端的连接
sk.close() #关闭服务器套接字

Client端:

import socket
sk = socket.socket() #创建客户端的套接字
sk.connect(('127.0.0.1',8080)) #尝试连接服务器 sk.send(b'hello') #向服务器发送消息
ret = sk.recv(1024) #接收服务器发送的消息
print(ret) #打印服务器发送的消息 sk.close() #关闭客户端的套接字

Sever端和Client端两边都要对应接收和发送信息,不能只接不收或只收不接且执行时要先执行Sever端再去执行Client端,执行结果:

当重启服务器时遇到地址已被使用的问题

解决方案:

如何让客户端和服务器能够多次进行对话?

Sever端:

import socket
from socket import SOL_SOCKET,SO_REUSEADDR
sk = socket.socket()   #创建服务器的套接字
sk.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)
sk.bind(('127.0.0.1',8080))   #把地址绑定到套接字
sk.listen()   #监听链接
conn,addr = sk.accept()   #接收到客户端的连接和地址 print(addr)   #打印客户端的地址
ret = conn.recv(1024)   #接收客户端信息
print(ret)   #打印客户端的信息
conn.send(b'hi')   #向客户端发送信息
print(addr)   #打印客户端的地址
ret = conn.recv(1024)   #接收客户端信息
print(ret)   #打印客户端的信息
conn.send(b'hi')   #向客户端发送信息 conn.close()   #关闭客户端的连接
sk.close()   #关闭服务器套接字

Client端:

import socket
sk = socket.socket() #创建客户端的套接字
sk.connect(('127.0.0.1',8080)) #尝试连接服务器 sk.send(b'hello') #向服务器发送消息
ret = sk.recv(1024) #接收服务器发送的消息
print(ret) #打印服务器发送的消息
sk.send(b'hello') #向服务器发送消息
ret = sk.recv(1024) #接收服务器发送的消息
print(ret) #打印服务器发送的消息 sk.close() #关闭客户端的套接字

执行结果:

因此我们只要在Sever端获取到Client端后让它去循环接收和发送消息、让Client端在连接上Sever端后去循环发送和接收消息就能完成一个一直发送接收消息的过程,但这个过程是一个死循环,我们需要加上一些条件让它更加完善

Sever端:

import socket
import time
from json import dumps
from socket import SOL_SOCKET,SO_REUSEADDR
sk = socket.socket() #创建服务器的套接字
sk.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)
sk.bind(('127.0.0.1',8080)) #把地址绑定到套接字
sk.listen() #监听链接
conn,addr = sk.accept() #接收到客户端的连接和地址 while 1:
ret = conn.recv(1024).decode('utf-8') #接收客户端信息
if ret.strip() == 'bye' or ret.strip() == 'bye'.capitalize(): #跳出循环的条件
conn.send(b'bye') #发送bytes类型的Bye
print('Client has disconnected!')
break
print(time.asctime(time.localtime(time.time())) + ' ' + dumps(addr) + ':' + ret) #格式化打印客户端发来的消息
sendInfo = input(time.asctime(time.localtime(time.time())) + ' <<<')
conn.send(sendInfo.encode('utf-8')) #向客户端发送信息 conn.close() #关闭客户端的连接
sk.close() #关闭服务器套接字

Client端:

import socket
import time
sk = socket.socket() #创建客户端的套接字
sk.connect(('127.0.0.1',8080)) #尝试连接服务器 while 1:
sendInfo = input(time.asctime(time.localtime(time.time())) + ' <<<')
sk.send(sendInfo.encode('utf-8')) #向服务器发送消息
ret = sk.recv(1024).decode('utf-8') #接收服务器发送的消息
if ret.strip() == 'bye' or ret.strip() == 'bye'.capitalize(): #跳出循环的条件
sk.send(b'bye') #发送bytes类型的Bye
print('Sever has disconnected!')
break
print(time.asctime(time.localtime(time.time())) + ' ["127.0.0.1", 8080]:' + ret)#格式化打印服务器发来的消息 sk.close() #关闭客户端的套接字

执行结果:

基于UDP协议实现Sever端和Client端的信息互通

实现简单传递信息

Sever端:

import socket
sk = socket.socket(type=socket.SOCK_DGRAM) #创建服务器套接字
sk.bind(('127.0.0.1',8080)) #绑定服务器套接字 msg,addr = sk.recvfrom(1024) #接收到客户端的信息和地址
print(msg.decode('utf-8')) #将客户端传递来的信息解码并打印出来
sk.sendto(b'Hello client!',addr) #将bytes类型的信息发送到客户端的地址 sk.close() #关闭服务器套接字

Client端:

import socket
sk = socket.socket(type=socket.SOCK_DGRAM) #创建客户端套接字
ip_port = ('127.0.0.1',8080) sk.sendto(b'Hello sever!',ip_port) #客户端发送信息
ret,addr = sk.recvfrom(1024) #接收到服务器的地址和信息
print(ret.decode('utf-8')) #打印收到的信息 sk.close() #关闭客户端套接字

执行结果:

多人版:

Sever端:

import socket
sk = socket.socket(type=socket.SOCK_DGRAM) #创建服务器套接字
sk.bind(('127.0.0.1',8080)) #绑定服务器套接字 while 1:
msg,addr = sk.recvfrom(1024) #接收到客户端的信息和地址
str_msg = msg.decode('utf-8')
print(addr,str_msg)
info = input('<<<').encode('utf-8')
sk.sendto(info,addr) sk.close() #关闭服务器套接字

Client1:

import socket
sk = socket.socket(type=socket.SOCK_DGRAM) #创建客户端套接字
ip_port = ('127.0.0.1',8080) while 1:
info = input('Client1:')
info = ('\033[32m来自Client1的消息:%s\033[0m'%info).encode('utf-8')
sk.sendto(info,ip_port)
msg,addr = sk.recvfrom(1024)
str_msg = msg.decode('utf-8')
print(str_msg) sk.close() #关闭客户端套接字

Client2:

import socket
sk = socket.socket(type=socket.SOCK_DGRAM) #创建客户端套接字
ip_port = ('127.0.0.1',8080) while 1:
info = input('Client2:')
info = ('\033[36m来自Client2的消息:%s\033[0m' % info).encode('utf-8')
sk.sendto(info, ip_port)
msg, addr = sk.recvfrom(1024)
str_msg = msg.decode('utf-8')
print(str_msg) sk.close() #关闭客户端套接字

执行结果:

udp的Sever不需要进行监听也不需要建立连接,在启动服务之后只能被动得等待客户端发送消息来,客户端发送消息的同时还会自带地址信息,消息回复的时候不仅要发送消息还要把对方的地址给填上

Python学习日记(三十) Socket模块使用的更多相关文章

  1. Python学习日记(三十二) hmac检验客户端的合法性和socketsever模块

    Hmac模块 其实这个模块类似hashlib模块,它能将一些重要的信息通过算法加密成密文,让信息更具有安全性. 关于hmac加密算法的了解:它的全名是哈希运算消息认证码(Hash-based Mess ...

  2. Python学习日记(三十六) Mysql数据库篇 四

    MySQL作业分析 五张表的增删改查: 完成所有表的关系创建 创建教师表(tid为这张表教师ID,tname为这张表教师的姓名) create table teacherTable( tid int ...

  3. Python学习日记(三十八) Mysql数据库篇 六

    Mysql视图 假设执行100条SQL语句时,里面都存在一条相同的语句,那我们可以把这条语句单独拿出来变成一个'临时表',也就是视图可以用来查询. 创建视图: CREATE VIEW passtvie ...

  4. Python学习日记(三十四) Mysql数据库篇 二

    外键(Foreign Key) 如果今天有一张表上面有很多职务的信息 我们可以通过使用外键的方式去将两张表产生关联 这样的好处能够节省空间,比方说你今天的职务名称很长,在一张表中就要重复的去写这个职务 ...

  5. Python学习日记(三十九) Mysql数据库篇 七

    Mysql函数 高级函数 1.BIN(N) 返回N的二进制编码 ); 执行结果: 2.BINARY(str) 将字符串str转换为二进制字符串 select BINARY('ASCII'); 执行结果 ...

  6. Python学习日记(三十五) Mysql数据库篇 三

    使用Navicate 创建一个连接去使用Mysql的数据库,连接名可以取任意字符但是要有意义 新增一个数据库 填写新数据库名,设置它的字符集和排序规则 新建一个表 增加表中的信息 点击保存再去输入表名 ...

  7. Python学习日记(二十八) hashlib模块、configparse模块、logging模块

    hashlib模块 主要提供字符加密算法功能,如md5.sha1.sha224.sha512.sha384等,这里的加密算法称为摘要算法.什么是摘要算法?它又称为哈希算法.散列算法,它通过一个函数把任 ...

  8. Python学习日记(十)—— 杂货铺(全局变量补充、Python参数传递、字符串格式化、迭代器、生成器)

    全局变量补充 python自己添加了些全局变量 print(vars()) """结果: {'__name__': '__main__', '__doc__': None ...

  9. Python学习(三十九)—— Django之Form组件

    一.构建一个表单 假设你想在你的网站上创建一个简单的表单,以获得用户的名字.你需要类似这样的模板: <form action="/your-name/" method=&qu ...

随机推荐

  1. continue & tag in GO

    Go语言中 continue 语句可以结束当前循环,开始下一次的循环迭代过程,仅限在 for 循环内使用,在 continue 语句后添加标签时,表示开始标签对应的循环,例如: package mai ...

  2. 关于Spring Cloud的思考和总结

    博主开发使用了Spring Cloud,使用过程中的一些学习和经验,体会通过本博文进行记录. 以前的架构,单体应用: 加入ESB,通过总线进行调用: 微服务架构: 区别: SOA架构主要针对企业级,采 ...

  3. 阿里云环境安装K8S步骤

    1. 安装docker yum install -y docker 2. 修改 /etc/docker/daemon.json 文件并添加上 registry-mirrors 键值 $ vim /et ...

  4. @vue/cli 4.1.1安装

    按照安装步骤,先卸载,再安装,最终,查看vue -V 的版本都是3.8.2,也就是说并没有安装成功,于是,考虑用yarn去安装 1,首先清除缓存: yarn cache clean 2,yarn设置淘 ...

  5. zxing解析生成一维码二维码

    @web界面实现扫一扫 二维码工具类 package util; import java.awt.BasicStroke; import java.awt.Graphics; import java. ...

  6. EasyNVR摄像机网页无插件直播方案H5前端构建之:使用BootstrapPagination以分页形式展示数据信息

    背景介绍 EasyNVR核心在于摄像机的音视频流的获取.转换.转码与高性能分发,同时同步完成对实时直播流的录像存储,在客户端(PC浏览器.Android.iOS.微信)进行录像文件的检索.回放和下载. ...

  7. myeclipse An internal error occurred during: "Initialize metrics".

    重新安装的myeclipse,在打开的时候弹出:    An internal error occurred during: "Initialize metrics". com/g ...

  8. war包部署在tomcat下,使用windows service服务方式启动tomcat服务器,在包含调用dll的模块,报dll找不到问题的解决办法

    问题描述: 开发了一个需要调用dll的java web程序,在idea开发环境下运行调试没问题,可以正常运行,在tomcat/bin下,运行批处理startup.bat,启动tomcat服务器,也可以 ...

  9. 【windows】win10局域网共享文件夹

    1.打开 2.共享文件夹 另一台局域网电脑可根据共享的电脑ip访问就可以了[格式:\\ip] 如果右键文件夹没有共享选项

  10. 分布式事务的 N 种实现

    转自:http://myfjdthink.com/2019/04/26/%E5%88%86%E5%B8%83%E5%BC%8F%E4%BA%8B%E5%8A%A1%E7%9A%84-n-%E7%A7% ...