TCP客户端和服务器间传输数据遇到的TypeError: a bytes-like object is required, not 'str'问题
使用python实现python核心编程3第472页和474页的TCP时间戳服务器和客户端服务器间数据传输编程时遇到TypeError: a bytes-like object is required, not 'str'报错问题,经查找资料知道是Python中的字符串数据和字节数据在python3中不能混用所致,一博文 https://eli.thegreenplace.net/2012/01/30/the-bytesstr-dichotomy-in-python-3 对python3中字符串和字节数据间区别联系做了较详细解释。
下面是https://www.cnblogs.com/abclife/p/7445222.html 进行的译文
python 3中最重要的新特性可能就是将文本(text)和二进制数据做了更清晰的区分。文本总是用unicode进行编码,以str类型表示;而二进制数据以bytes类型表示。
在python3中,不能以任何隐式方式将str和bytes类型二者混合使用。不可以将str和bytes类型进行拼接,不能在str中搜索bytes数据(反之亦然),也不能将str作为参数传入需要bytes类型参数的函数(反之亦然)。
字符串和字节符之间划分界线是必然的。下面这个图解要牢记于心:
strings可以被编码(encode)成字bytes,bytes也可以解码(decode)成strings:
>>> '€20'.encode('utf-8')
b'\xe2\x82\xac20'
>>> b'\xe2\x82\xac20'.decode('utf-8')
'€20'
可以这样理解:
string是文本(text)的抽象表示。字符串(string)由字符组成,字符也是抽象的实体且与任何二进制表示无关。
当操纵字符串的时候,很多细节是不用了解的。我们可以分割、切片和拼接字符串,在字符串内部进行搜索。但并不在乎内部是如何表示的,也不用在意底层一个字符要花费多少byte。
只有在需要将string编码(encode)成byte的时候,比如:通过网络传输数据;或者需要将byte解码(decode)成string的时候,我们才会关注string和byte的区别。
传入encode和decode的参数是编码方式。编码是一种用二进制数据表示抽象字符的方式。目前有很多种编码。上面给出的UTF-8是其中一种,下面是另一种:
>>> '€20'.encode('iso-8859-15')
b'\xa420'
>>> b'\xa420'.decode('iso-8859-15')
'€20'
编码是这个转换过程中至关重要的一部分。若不编码,bytes对象b'\xa420'只是一堆比特位而已。编码赋予其含义。采用不同的编码,这堆比特位的含义就会大不同:
>>> b'\xa420'.decode('windows-1255')
'₪20'
因而服务器端编码
from socket import *
from time import ctime HOST=''
PORT=21567
BUFSIZ=1024
ADDR=(HOST,PORT) tcpSerSock=socket(AF_INET,SOCK_STREAM)
tcpSerSock.bind(ADDR) #绑定地址到套接字
tcpSerSock.listen(5) #开始TCP监听 while True:
print('waiting for connection...')
tcpSerSock,addr=tcpSerSock.accept()#被动接受TCP客户端连接,等待连接的到来
print('...connected from:',addr) while True:
data=tcpSerSock.recv(BUFSIZ)
print(data.decode()) #服务器接受到客户端发送来的数据是字节类型,需要转换为字符串类型
if not data:
break
tcpSerSock.send(('[%s] %s'%(ctime(),data)).encode()) # 网络传输数据前将字符串数据转化为字节类型
tcpSerSock.close()
tcpSerSock.close()
客户端编程
from socket import * HOST='localhost'
PORT=21567
BUFSIZ=1024
ADDR=(HOST,PORT) tcpCliSock=socket(AF_INET,SOCK_STREAM)
tcpCliSock.connect(ADDR) while True:
data=input('>')
if not data:
break
tcpCliSock.send(data.encode()) #网络传输数据前将字符串数据转化为字节类型
data=tcpCliSock.recv(BUFSIZ).decode() #将收到的字节类型数据转化为字符串数据
if not data:
break
print(data) tcpCliSock.close()
客户端显示
>what??
[Sat Mar 31 16:03:20 2018] b'what??'
>excuse me?
[Sat Mar 31 16:03:52 2018] b'excuse me?'
>
服务器端显示
waiting for connection...
...connected from: ('127.0.0.1', 50782)
what??
excuse me?
udp服务器端编程
from socket import *
from time import ctime HOST=''
PORT=21567
BUFSIZ=1024
ADDR=(HOST,PORT) udpSerSock=socket(AF_INET,SOCK_DGRAM)
udpSerSock.bind(ADDR) while True:
print('waiting for message...')
data,addr=udpSerSock.recvfrom(BUFSIZ) #data是字节对象,addr是字符串对象
print(data)
print(addr)
data=data.decode()
data='[%s]%s'%(ctime(),data)
udpSerSock.sendto(data.encode(),addr)
print('..received from and returned to:',addr) udpSerSock.close()
客户端编程
from socket import * HOST='localhost'
PORT=21567
BUFSIZ=1024
ADDR=(HOST,PORT) udpCliSock=socket(AF_INET,SOCK_DGRAM)
while True:
data=input('>').encode()
if not data:
break
udpCliSock.sendto(data,ADDR)
data,ADDR=udpCliSock.recvfrom(BUFSIZ)
if not data:
break
else:
print(data.decode())
# print(udpCliSock.close()) udpCliSock.close()
服务器端显示
waiting for message...
b'who is that?'
('127.0.0.1', 60959)
..received from and returned to: ('127.0.0.1', 60959)
waiting for message...
b'I am Jack'
('127.0.0.1', 60959)
..received from and returned to: ('127.0.0.1', 60959)
waiting for message...
b'sending through UDP'
('127.0.0.1', 60959)
..received from and returned to: ('127.0.0.1', 60959)
waiting for message...
客户端显示
>who is that?
[Sat Mar 31 16:47:15 2018]who is that?
>I am Jack
[Sat Mar 31 16:47:24 2018]I am Jack
>sending through UDP
[Sat Mar 31 16:47:32 2018]sending through UDP
>
TCP客户端和服务器间传输数据遇到的TypeError: a bytes-like object is required, not 'str'问题的更多相关文章
- 利用Python中SocketServer 实现客户端与服务器间非阻塞通信
利用SocketServer模块来实现网络客户端与服务器并发连接非阻塞通信 版权声明 本文转自:http://blog.csdn.net/cnmilan/article/details/9664823 ...
- 【python】网络编程-SocketServer 实现客户端与服务器间非阻塞通信
利用SocketServer模块来实现网络客户端与服务器并发连接非阻塞通信.首先,先了解下SocketServer模块中可供使用的类:BaseServer:包含服务器的核心功能与混合(mix-in)类 ...
- [JAVA] Tcp客户端和服务器简单代码
服务器: import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; im ...
- Java实验四 TCP客户端和服务器的应用
实验内容 1.掌握Socket程序的编写: 2.掌握密码技术的使用: 3.设计安全 4.对通信内容进行摘要计算并验证 实验步骤 1.信息安全传送: 发送方A——————>接收方B A加密时,用B ...
- socket编程:客户端与服务器间的连接以及各函数的用法
在认真的看UNP之前,一直被socket编程说的云里雾里,今天我要让大家从整天上认识socket编程,让我们知道socket编程的整个流程和各个函数的用法.这样:我们在写一些简单的socket编程时就 ...
- Socket TCP客户端和服务器的实现
import java.io.*; import java.net.Inet4Address; import java.net.InetSocketAddress; import java.net.S ...
- TCP客户端与服务器的实现
为了更容易理解,我们举一个小例子来说明服务器与客户端之间的连接过程. 有一个饭店,饭店里有服务员,服务员用于招待客人 特别要注意的是:要记住相关函数的各个参数都是什么,什么时候返回SOCKET_ERR ...
- tcp客户端从服务器下载文本文件
代码讲解: server import socket def send_file_client(new_client_socket, new_client_addr): # 接收客户端需要下载的文件名 ...
- 基于Qt的tcp客户端和服务器实现摄像头帧数据处理(客户端部分)
项目简述 实现客户端调用摄像头,并以帧的形式将每一帧传输到服务端,服务端将图片进行某些处理后再返回给客户端.(客户端与服务端通信代码部分参考<Qt5 开发及实例>) 项目步骤 客户端的编写 ...
随机推荐
- springMVC集成 -- shiro(配置)
备注:文中配置基本来自尚硅谷视频教程,也可自行参照shiro官方教程:http://shiro.apache.org/spring.html 1.首先通过maven导入shiro相关依赖jar包,修改 ...
- Ordering Tasks UVA - 10305 图的拓扑排序
John has n tasks to do. Unfortunately, the tasks are not independent and the execution of one task i ...
- Servlet 编写过滤器
Servlet 过滤器可以动态地拦截请求和响应,以变换或使用包含在请求或响应中的信息. 可以将一个或多个 Servlet 过滤器附加到一个 Servlet 或一组 Servlet.Servlet 过滤 ...
- python 案例:使用BeautifuSoup4的爬虫
我们以腾讯社招页面来做演示:http://hr.tencent.com/position.php?&start=10#a 使用BeautifuSoup4解析器,将招聘网页上的职位名称.职位类别 ...
- 【HDU 5305】Friends 多校第二场(双向DFS)
依据题意的话最多32条边,直接暴力的话 2 ^ 32肯定超时了.我们能够分两次搜索时间复杂度降低为 2 * 2 ^ 16 唯一须要注意的就是对眼下状态的哈希处理. 我採用的是 十进制表示法 跑的还是 ...
- winform 的 checklistbox动态绑定并选中值
绑定的代码:这里绑定的是一个泛型 BLL.PowerBLL powerbll = new BLL.PowerBLL(); checkpower.DataSource = powerbll.GetAll ...
- Laravel5.1 -控制器(初步了解)
首先道个歉 这篇笔记是前两天就应该写的,可大K有点事儿要忙 就耽误了,今天抽空学了学控制器,并写个笔记分享下. 为什么要使用控制器 像我们之前写一些逻辑呢都是在Route(路由)中,搞得Route文件 ...
- http://localhost:8080请求用户名和密码。信息为:“XDB” 解决办法
windows查看端口占用情况 cmd下 netstat -ano 查看端口和对应的服务 为2520 Oracle的服务 源博客: http://blog.163.com/jxguo_05/blog/ ...
- TP系统常量信息
[系统常量信息] 获取系统常量信息: 如果加参数true,会分组显示: 显示如下: [跨控制器调用] 一个控制器在执行的时候,可以实例化另外一个控制,并通过对象访问其指定方法. 跨控制器调用可以节省我 ...
- 《从零开始学Swift》学习笔记(Day 36)——静态方法
原创文章,欢迎转载.转载请注明:关东升的博客 静态方法与静态属性类似,Swift中定义了静态方法,也称为类型方法.静态方法的定义与静态属性类似,枚举和结构体的静态方法使用的关键字是static:类静态 ...