Python复习笔记(六)网络编程(udp/tcp)
一、网络-udp(用户数据报协议)
用户数据报协议
类似写信,不安全,数据有可能丢
1.1 ip地址
注意:
IP地址127.0.0.1 ~ 127.255.255.255 用于回路测试
私有ip地址,不在公网中使用
1.2 端口(重点)
- 端口是通过端口号来标记的,端口号只有整数,范围是从0~65535(2^16)
- 知名端口:(0~1023)(>1024的随便用)
- 80端口 分配给 HTTP 服务
- 21端口 分配给 ftp 服务
- 动态端口(1024~65535)
1.3 socket简介
1.31 不同电脑上进程之间如何通信
- 利用 ip地址、协议、端口 就可以标识网络的进程了。
- 进程间通信:运行的程序之间的数据共享。
1.32 创建socket
在Python中 使用socket 模块的函数 socket就可以完成:、
import socket # AddressFamily: ipv4/ipv6; Type: udp/tcp
socket.socket(AddressFamily, Type)
说明:
函数 socket.socket 创建一个 socket,该函数带有两个参数:
Address Family:可以选择 AF_INET (用于Internet 进程间通信) 或者 AF_UNIX(用于同一台机器进程间通信),实际工作常用 AF_INET
Type:套接字类型,可以是 SOCK_STREAM(流式套接字,主要用于TCP协议)或者 SOCK_DGRAM(数据报套接字,主要用于UDP协议)
创建一个 tcp socket (tcp套接字)
import socket # AddressFamily: ipv4/ipv6; Type: udp/tcp # 创建tcp的套接字
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # ... 这里是使用套接字的功能 (省略) # 不用的时候,关闭套接字
s.close()
创建一个 udp socket (udp 套接字)
import socket # AddressFamily: ipv4/ipv6; Type: udp/tcp # 创建tcp的套接字
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # ... 这里是使用套接字的功能 (省略) # 不用的时候,关闭套接字
s.close()
说明
- 套接字使用流程
- 创建套接字
- 使用套接字收/发数据
- 关闭套接字
1.4 udp网络程序-发送、接收数据
1.41 udp网络程序-发送程序
import socket # AddressFamily: ipv4/ipv6; Type: udp/tcp # 1. 创建udp的套接字
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 2. 准备接收方地址
# '192.168.43.74' 表示目标ip地址
# 8080表示目的端口
dest_addr = ('192.168.43.74', 8080) # 注意 是元组,ip是字符串,端口是数字 # 3. 从键盘获取数据
send_data = input("请输入要发送的数据:") # 4. 发送数据到指定的电脑上的指定程序中
udp_socket.sendto(send_data.encode('utf-8'), dest_addr) # 不用的时候,关闭套接字
udp_socket.close()
注意:可以用NetAssist测试
1.42 udp网络程序—接收程序 (绑定实例)
# -*- coding: utf-8 -*-
"""
Created on Fri Mar 1 22:25:32 2019 @author: Douzi
""" import socket def main():
# AddressFamily: ipv4/ipv6; Type: udp/tcp # 1. 创建udp的套接字
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 2. 绑定一个本地信息 '192.168.43.74'
local_addr = ("", 7788) # 注意 是元组,ip是字符串,端口是数字
udp_socket.bind(local_addr) # 必须绑定自己电脑的ip以及port,其他的不行 # 3. 等待接收对方发送的数据
recv_data = udp_socket.recvfrom(1024) #1024表示本次接收的最大字节数
# recv_data这个变量中存储的是一个元组(接收到的数据, (发送方ip, port))
recv_msg = recv_data[0]
send_addr = recv_data[1] # 4. 显示接收到的数据------------
print(recv_data)
print(recv_msg.decode("gbk"))
print(send_addr) # 关闭套接字
udp_socket.close() if __name__=="__main__":
main()
1.43 聊天程序
# -*- coding: utf- -*-
"""
Created on Sat Mar :: @author: Douzi
""" import socket def send_msg(udp_socket):
"""发送消息"""
# 获取内容
dest_ip = input("输入ip: ")
dest_port = int(input("输入对方port:"))
send_data = input("输入发送数据:")
udp_socket.sendto(send_data.encode("utf-8"), (dest_ip, dest_port)) def recv_msg(udp_socket):
recv_data = udp_socket.recvfrom()
print("%s:%s " % (str(recv_data[]), recv_data[].decode("utf-8"))) def main():
# 创建套接字
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 绑定信息
udp_socket.bind(("", )) # '192.168.43.74' while True: print("0----发送消息")
print("1----接收消息")
print("2----退出")
op = input("输入功能: ") if op == "":
# 发送
send_msg(udp_socket)
elif op == "":
# 接收并显示
recv_msg(udp_socket)
else:
return if __name__=="__main__":
main()
二、TCP
2.1 简介(传输控制协议)
- TCP协议,传输控制协议(Transmission Control Protocol)
- 类似打电话
- TCP通信需要经过 创建连接、数据传送、终止连接 三个步骤。
2.2 特点(安全复杂)
1. 面向连接
2. 可靠传输
- TCP采用发送应答机制
超时重传
发送端发出一个报文段之后,就 启动定时器,如果在定时时间没有收到应答就重新发送这个报文段
TCP为了保证不发生丢包,就给每个包一个序号,同时序号也保证了传送到接收端实体的包的 按序接收,然后 接收端实体 对已成功收到的包发回一个相应的确认(ACK);如果发送端实体在合理的 往返时延(RTT)内未收到确认,那么对应的数据包就被假设为已丢失,将会进行重传。
错误校验
- TCP用一个校验和函数来检验数据是否有错误;在发送和接收时都要计算校验和。
流量控制和阻塞管理
流量控制用来避免主机发送得快而使接收方来不及完全收下
TCP和UDP的不同点
面向连接(确认有创建三方交握,连接已创建才做传输)
有序数据传输
重发丢失的数据包
舍弃重复的数据包
无差错的数据传输
阻塞/流量控制
2.3 tcp客户端(重点)
2.31 tcp客户端构建流程
示例代码:
# -*- coding: utf-8 -*-
"""
Created on Sun Mar 3 14:14:03 2019 @author: Douzi
""" import socket def main():
# 1.创建socket
tcp_client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 2.链接服务器
# server_ip = input("输入服务器ip:")
# server_port = int(input("输入服务器port:"))
# tcp_client_socket.connect((server_ip, server_port))
tcp_client_socket.connect(("192.168.43.74", 8080)) # 提示用户输入数据
send_data = input("输入数据: ") tcp_client_socket.send(send_data.encode("gbk")) # 接收对方发送过来的数据,最大接收1024个字节
recvData = tcp_client_socket.recv(1024) print("接收到的数据: ", recvData.decode("gbk")) # 关闭套接字
tcp_client_socket.close() if __name__=="__main__":
main()
2.4 tcp服务器(重点)
流程如下:
1. socket创建一个套接字
2. bind绑定ip和port
3. listen使套接字变为 可以被动链接(监听套接字 负责 等待有新的客户端进行链接)
4. accept等待客户端的链接 (accept 产生新的套接字 用来为客户端服务)
5. recv/send接收发送数据
一个简单的tcp服务器如下:
# -*- coding: utf-8 -*-
"""
Created on Sun Mar 3 14:55:42 2019 @author: Douzi
""" import socket def main():
# 创建套接字
tcp_socket_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定本地信息 bind
tcp_socket_server.bind(("192.168.43.74", 6677))
# 让默认的套接字由主动变为被动 listen,这样就可以接收别人的链接了
tcp_socket_server.listen(128)
# 等待客户端的链接accept
# client_socket用来为这个客户端服务
# tcp_socket_server就可以省下来专门等待其他新客户端的链接
client_socket, clientAddr = tcp_socket_server.accept() # 接收对方发送过来的数据
recv_data = client_socket.recv(1024) # 接收1024个字节
print('接收到的数据: ', recv_data.decode('gbk')) # 发送一些数据到客户端
client_socket.send("Douzi is cute".encode('gbk')) # 关闭套接字
client_socket.close() if __name__=="__main__":
main()
为多个客户端服务
# -*- coding: utf-8 -*-
"""
Created on Sun Mar 3 14:55:42 2019 @author: Douzi
""" import socket def main():
# 创建套接字(买个手机)
tcp_socket_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定本地信息 bind (插入手机卡)
tcp_socket_server.bind(("192.168.43.74", 6677))
# (将手机设置为响铃)让默认的套接字由主动变为被动 listen,这样就可以接收别人的链接了
tcp_socket_server.listen(128) while True:
print("等待一个新的客户端的到来....")
# 等待客户端的链接accept (等待别人的电话)
# client_socket用来为这个客户端服务
# tcp_socket_server就可以省下来专门等待其他新客户端的链接
client_socket, client_addr = tcp_socket_server.accept() print(client_addr) # 为同一个客户端服务多次
while True:
# 接收对方发送过来的数据
recv_data = client_socket.recv(1024) # 接收1024个字节
print('接收到的数据: ', recv_data.decode("gbk")) # 如果recv 解堵塞,那么有2种方式:
# 1. 客户端发送过来数据
# 2. 客户端调用close导致了 这里recv解堵塞
if recv_data:
# 回送一部分数据到客户端
client_socket.send("Douzi is cute".encode('gbk'))
else:
break # 关闭套接字
client_socket.close() print("已经服务完毕....") tcp_socket_server.close() if __name__=="__main__":
main()
2.5 案例:文件下载器
文件下载的服务器
# -*- coding: utf-8 -*-
"""
Created on Sun Mar 3 17:46:37 2019 @author: Douzi
""" import socket
import sys def get_file_content(file_name):
"""获取文件内容"""
try:
with open(file_name, "rb") as f:
content = f.read()
return content
except:
print("没有下载的文件: %s" % file_name) def main(): if len(sys.argv) != 2:
print("请按如下方式运行: python3 xxx.py 7890")
return
else:
# 运行方式
port = int(sys.argv[1]) # 创建socket
tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 本地信息, port: 7890
address = ("192.168.43.74", port)
# 绑定本地信息
tcp_server_socket.bind(address)
# 将主动套接字变为被动套接字
tcp_server_socket.listen(128) while True:
# 等待客户端的链接,即为这个客户端发送文件
client_socket, client_addr = tcp_server_socket.accept()
# 接收对方发送过来的数据
recv_data = client_socket.recv(1024)
file_name = recv_data.decode("utf-8")
print("客户端需要下载的文件名为:%s" % file_name) file_content = get_file_content(file_name)
# 发送文件的数据给客户端
# 因为获取打开文件时是以rb方式打开,所以file_content中的数据已经是二进制的格式
# 所以不需要encode
if file_content:
client_socket.send(file_content)
# 关闭这个套接字
client_socket.close()
isOK = input("继续吗?Y/N")
if isOK == "Y":
break # 关闭监听套接字
tcp_server_socket.close() if __name__=="__main__":
main()
文件下载的客户端
# -*- coding: utf-8 -*-
"""
Created on Sun Mar 3 19:34:20 2019 @author: Douzi
""" import socket def main(): # 创建socket
tcp_socket_client = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 目的信息
server_ip = "192.168.43.74" # 服务器的ip
server_port = 7891 # 服务器的port # 链接服务器
tcp_socket_client.connect((server_ip, server_port)) # 输入需要下载的文件名
file_name = input("输入要下载的文件名: ") # 发送文件下载请求
tcp_socket_client.send(file_name.encode("utf-8")) # 接收对方发送过来的数据,最大接收1024个字节 (1k)
# 1024--1k, 1024*1024--1M,1024*1024*1024--1G
recv_data = tcp_socket_client.recv(1024*1024) # print('接收到的数据', recv_data.decode('utf-8'))
# 如果接收到数据再创建文件,否则不创建
if recv_data:
with open("[接收]"+file_name, "wb") as f:
f.write(recv_data) # 关闭套接字
tcp_socket_client.close() if __name__=="__main__":
main()
2.6 tcp注意点
tcp服务器 一般需要绑定,否则客户端找不到这个服务器
tcp客户端 一般不绑定,因为是主动链接服务器,所以只要确定好服务器的ip,port等信息就好,本地客户端可随意
tcp服务器通过listen可以将socket创建出来的主动套接字变成被动的,这是做tcp服务器必须要做的
当客户端需要链接服务时,服务器端会有1个新的套接字,这个套接字用来标记这个客户端,单独为这个客户端服务
listen后的套接字是被动套接字,用来接收 新的客户端 的链接请求,而accept返回的新套接字,是标记这个新客户端的。
关闭 listen 后的套接字 意味着 被动套接字被关闭了,会导致新的客户端不能够链接服务器,但是之前已经链接成功的客户端正常通信
关闭 accept 返回的套接字意味着这个客户端已经服务完毕。
当客户端的套接字 调用close后,服务器端会recv解阻塞,并且返回的长度为0,因此服务器可以通过返回数据的长度来区别客户端是否已经下线。
udp | tcp:client | tcp:server |
socket | socket | socket |
bind | connect | bind() |
sendto/recvfrom | send/recv | listen(128) |
close | close | accept: client_socket, client_addr |
recv/send | ||
close |
三. tcp的3次握手、4次挥手(重点 !)
- syn:标记请求的第一次
- ack:应答
3.1 3次握手
业务开始(客户端先请求)
3.2 4次挥手
业务结束后,关闭(客户端先关)
服务器先关闭,需要等待2-5分钟
设置当前服务器先close 即服务器4次挥手后资源能够立即是否,这样就保证了。下次运行程序时,可以立即执行
tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
Python复习笔记(六)网络编程(udp/tcp)的更多相关文章
- python学习笔记11 ----网络编程
网络编程 网络编程需要知道的概念 网络体系结构就是使用这些用不同媒介连接起来的不同设备和网络系统在不同的应用环境下实现互操作性,并满足各种业务需求的一种粘合剂.网络体系结构解决互质性问题彩是分层方法. ...
- python学习笔记10 ----网络编程
网络编程 网络编程需要知道的概念 网络体系结构就是使用这些用不同媒介连接起来的不同设备和网络系统在不同的应用环境下实现互操作性,并满足各种业务需求的一种粘合剂.网络体系结构解决互质性问题彩是分层方法. ...
- java 网络编程 UDP TCP
网络编程 网络编程主要用于解决计算机与计算机(手机.平板..)之间的数据传输问题. 网络编程: 不需要基于html页面就可以达到数据之间的传输. 比如: feiQ , QQ , 微信....网页编程: ...
- python学习笔记(六) 函数式编程
一 函数对象 函数同样可以作为对象复制给一个变量,如下: f = abs; print(f(-10)) f = 'abs'; print(f) def add(a,b,f): return f(a) ...
- 【网络编程1】网络编程基础-TCP、UDP编程
网络基础知识 网络模型知识 OSI七层模型:(Open Systems Interconnection Reference Model)开放式通信系统互联参考模型,是国际标准化组织(ISO)提出的一个 ...
- python网络编程05 /TCP阻塞机制
python网络编程05 /TCP阻塞机制 目录 python网络编程05 /TCP阻塞机制 1.什么是拥塞控制 2.拥塞控制要考虑的因素 3.拥塞控制的方法: 1.慢开始和拥塞避免 2.快重传和快恢 ...
- Socket网络编程-UDP编程
Socket网络编程-UDP编程 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.UDP编程概述 1>.UDP服务端编程流程 创建socket对象.socket.SOCK_ ...
- C#网络编程之---TCP协议的同步通信(二)
上一篇学习日记C#网络编程之--TCP协议(一)中以服务端接受客户端的请求连接结尾既然服务端已经与客户端建立了连接,那么沟通通道已经打通,载满数据的小火车就可以彼此传送和接收了.现在让我们来看看数据的 ...
- 嵌入式linux的网络编程(1)--TCP/IP协议概述
嵌入式linux的网络编程(1)--TCP/IP协议概述 1.OSI参考模型及TCP/IP参考模型 通信协议用于协调不同网络设备之间的信息交换,它们建立了设备之间互相识别的信息机制.大家一定都听说过著 ...
随机推荐
- synchronized 关键字解析
synchronized 关键字解析 同步锁依赖于对象,每个对象都有一个同步锁. 现有一成员变量 Test,当线程 A 调用 Test 的 synchronized 方法,线程 A 获得 Test 的 ...
- 解决Eclipse每次修改完代码后需要先Clean,不然修改的代码无效
工具栏 Project-->Build Automatically 勾选上即可
- 爬虫_糗事百科(scrapy)
糗事百科scrapy爬虫笔记 1.response是一个'scrapy.http.response.html.HtmlResponse'对象,可以执行xpath,css语法来提取数据 2.提取出来的数 ...
- ACM中的fread读入
fread可以加快读入速度,尤其是读特大的二进制文件. #include <cctype> typedef long long LL; char buf[100000],*p1=buf,* ...
- hihoCoder #1646 : Rikka with String II(容斥原理)
题意 给你 \(n\) 个 \(01\) 串 \(S\) ,其中有些位置可能为 \(?\) 表示能任意填 \(0/1\) .问对于所有填法,把所有串插入到 \(Trie\) 的节点数之和(空串看做根节 ...
- Hdoj 2046.骨牌铺方格 题解
Problem Description 在2×n的一个长方形方格中,用一个1× 2的骨牌铺满方格,输入n ,输出铺放方案的总数. 例如n=3时,为2× 3方格,骨牌的铺放方案有三种,如下图: Inpu ...
- 【CF908G】New Year and Original Order(动态规划)
[CF908G]New Year and Original Order(动态规划) 题面 洛谷 CF 题解 设\(f[i][j][k][0/1]\)表示当前填到了第\(i\)位,有\(j\)个大于等于 ...
- 华东交通大学2018年ACM“双基”程序设计竞赛部分题解
链接:https://ac.nowcoder.com/acm/contest/221/C来源:牛客网 C-公式题(2) 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32768K,其 ...
- java 字符与ASCII码互转
字符转对应ASCII码 // 方法一:将char强制转换为byte char ch = 'A'; byte byteAscii = (byte) ch; System.out.println(byte ...
- 洛谷P1896 互不侵犯
又是一道状压DP求方案数的题... 多了一个放k个的限制,于是我们把数组多开一维. f[i][j][k]表示前i行放了j个,第i行状态为k的方案数. 然后老套路DFS转移,这次要多记录一个cnt表示上 ...