菜鸟教程 -- 学的不仅是技术,更是梦想!

Python 3 教程
Python3 教程 Python3 简介 Python3 环境搭建 Python3 VScode Python3 基础语法 Python3 基本数据类型 Python3 数据类型转换 Python3 解释器 Python3 注释 Python3 运算符 Python3 数字(Number) Python3 字符串 Python3 列表 Python3 元组 Python3 字典 Python3 集合 Python3 条件控制 Python3 循环语句 Python3 编程第一步 Python3 推导式 Python3 迭代器与生成器 Python3 with Python3 函数 Python3 lambda Python3 装饰器 Python3 数据结构 Python3 模块 Python __name__ Python3 输入和输出 Python3 File Python3 OS Python3 错误和异常 Python3 面向对象 Python3 命名空间/作用域 Python 虚拟环境的创建 Python 类型注解 Python3 标准库概览 Python3 实例 Python 测验

Python3 高级教程

Python3 正则表达式 Python3 CGI编程 Python3 MySQL(mysql-connector) Python3 MySQL(PyMySQL) Python3 网络编程 Python3 SMTP发送邮件 Python3 多线程 Python3 XML 解析 Python3 JSON Python3 日期和时间 Python3 内置函数 Python3 MongoDB Python3 urllib Python uWSGI 安装配置 Python3 pip Python3 operator Python math Python requests Python random Python OpenAI Python 有用的资源 Python AI 绘画 Python statistics Python hashlib Python 量化 Python pyecharts Python selenium 库 Python 爬虫 Python Scrapy 库 Python Markdown Python sys 模块 Python Pickle 模块 Python subprocess 模块 Python queue 模块 Python StringIO 模块 Python logging 模块 Python datetime 模块 Python re 模块 Python csv 模块 Python threading 模块 Python asyncio 模块 Python PyQt Python for 循环 Python while 循环
(追記) (追記ここまで)

Python3 网络编程

Python 提供了两个级别访问的网络服务。:

  • 低级别的网络服务支持基本的 Socket,它提供了标准的 BSD Sockets API,可以访问底层操作系统Socket接口的全部方法。
  • 高级别的网络服务模块 SocketServer, 它提供了服务器中心类,可以简化网络服务器的开发。

什么是 Socket?

Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求,使主机间或者一台计算机上的进程间可以通讯。


socket()函数

Python 中,我们用 socket() 函数来创建套接字,语法格式如下:

socket.socket([family[, type[, proto]]])

参数

  • family: 套接字家族可以是 AF_UNIX 或者 AF_INET
  • type: 套接字类型可以根据是面向连接的还是非连接分为SOCK_STREAMSOCK_DGRAM
  • proto: 一般不填默认为0.

Socket 对象(内建)方法

函数 描述
服务器端套接字
s.bind() 绑定地址(host,port)到套接字, 在AF_INET下,以元组(host,port)的形式表示地址。
s.listen() 开始TCP监听。backlog指定在拒绝连接之前,操作系统可以挂起的最大连接数量。该值至少为1,大部分应用程序设为5就可以了。
s.accept() 被动接受TCP客户端连接,(阻塞式)等待连接的到来
客户端套接字
s.connect() 主动初始化TCP服务器连接,。一般address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。
s.connect_ex() connect()函数的扩展版本,出错时返回出错码,而不是抛出异常
公共用途的套接字函数
s.recv() 接收TCP数据,数据以字符串形式返回,bufsize指定要接收的最大数据量。flag提供有关消息的其他信息,通常可以忽略。
s.send() 发送TCP数据,将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。
s.sendall() 完整发送TCP数据。将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。
s.recvfrom() 接收UDP数据,与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。
s.sendto() 发送UDP数据,将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。
s.close() 关闭套接字
s.getpeername() 返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。
s.getsockname() 返回套接字自己的地址。通常是一个元组(ipaddr,port)
s.setsockopt(level,optname,value) 设置给定套接字选项的值。
s.getsockopt(level,optname[.buflen]) 返回套接字选项的值。
s.settimeout(timeout) 设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。一般,超时期应该在刚创建套接字时设置,因为它们可能用于连接的操作(如connect())
s.gettimeout() 返回当前超时期的值,单位是秒,如果没有设置超时期,则返回None。
s.fileno() 返回套接字的文件描述符。
s.setblocking(flag) 如果 flag 为 False,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用 recv() 没有发现任何数据,或 send() 调用无法立即发送数据,那么将引起 socket.error 异常。
s.makefile() 创建一个与该套接字相关连的文件

简单实例

服务端

我们使用 socket 模块的 socket 函数来创建一个 socket 对象。socket 对象可以通过调用其他函数来设置一个 socket 服务。

现在我们可以通过调用 bind(hostname, port) 函数来指定服务的 port(端口)

接着,我们调用 socket 对象的 accept 方法。该方法等待客户端的连接,并返回 connection 对象,表示已连接到客户端。

完整代码如下:

实例

#!/usr/bin/python3
# 文件名:server.py

# 导入 socket、sys 模块
import socket
import sys

# 创建 socket 对象
serversocket = socket.socket(
socket.AF_INET, socket.SOCK_STREAM)

# 获取本地主机名
host = socket.gethostname()

port = 9999

# 绑定端口号
serversocket.bind((host, port))

# 设置最大连接数,超过后排队
serversocket.listen(5)

while True:
# 建立客户端连接
clientsocket,addr = serversocket.accept()

print("连接地址: %s" % str(addr))

msg='欢迎访问菜鸟教程!'+ "\r\n"
clientsocket.send(msg.encode('utf-8'))
clientsocket.close()

客户端

接下来我们写一个简单的客户端实例连接到以上创建的服务。端口号为 9999。

socket.connect(hostname, port ) 方法打开一个 TCP 连接到主机为 hostname 端口为 port 的服务商。连接后我们就可以从服务端获取数据,记住,操作完成后需要关闭连接。

完整代码如下:

实例

#!/usr/bin/python3
# 文件名:client.py

# 导入 socket、sys 模块
import socket
import sys

# 创建 socket 对象
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 获取本地主机名
host = socket.gethostname()

# 设置端口号
port = 9999

# 连接服务,指定主机和端口
s.connect((host, port))

# 接收小于 1024 字节的数据
msg = s.recv(1024)

s.close()

print (msg.decode('utf-8'))

现在我们打开两个终端,第一个终端执行 server.py 文件:

$ python3 server.py

第二个终端执行 client.py 文件:

$ python3 client.py 
欢迎访问菜鸟教程!

这时我们再打开第一个终端,就会看到有以下信息输出:

连接地址: ('192.168.0.118', 33397)

Python Internet 模块

以下列出了 Python 网络编程的一些重要模块:

协议功能用处端口号Python 模块
HTTP网页访问80httplib, urllib, xmlrpclib
NNTP阅读和张贴新闻文章,俗称为"帖子"119nntplib
FTP文件传输20ftplib, urllib
SMTP发送邮件25smtplib
POP3接收邮件110poplib
IMAP4获取邮件143imaplib
Telnet命令行23telnetlib
Gopher信息查找70gopherlib, urllib

更多内容可以参阅官网的 Python Socket Library and Modules

AI 思考中...

3 篇笔记 写笔记

  1. #0

    miss

    zho***[email protected]

    199

    TCP 服务端

    服务端结构:

    tcps = socket() #创建服务器套接字
    tcps.bind() #把地址绑定到套接字
    tcps.listen() #监听链接
    while True: #服务器无限循环
     tcpc = tcps.accept() #接受客户端链接
     while True: #通讯循环
     tcpc.recv()/tcpc.send() #对话(接收与发送)
     tcpc.close() #关闭客户端套接字
    tcps.close() #关闭服务器套接字(可选)

    时间戳服务端实例:

    #!/usr/bin/python3
    # -*-coding:utf-8 -*-
    from socket import *
    import time
    COD = 'utf-8'
    HOST = '192.168.164.141' # 主机ip
    PORT = 21566 # 软件端口号
    BUFSIZ = 1024
    ADDR = (HOST, PORT)
    SIZE = 10 
    tcpS = socket(AF_INET, SOCK_STREAM) # 创建socket对象
    tcpS.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #加入socket配置,重用ip和端口
    tcpS.bind(ADDR) # 绑定ip端口号
    tcpS.listen(SIZE) # 设置最大链接数
    while True:
     print("服务器启动,监听客户端链接")
     conn, addr = tcpS.accept() 
     print("链接的客户端", addr)
     while True:
     try:
     data = conn.recv(BUFSIZ) # 读取已链接客户的发送的消息
     except Exception:
     print("断开的客户端", addr)
     break
     print("客户端发送的内容:",data.decode(COD))
     if not data:
     break
     msg = time.strftime("%Y-%m-%d %X") #获取结构化事件戳
     msg1 = '[%s]:%s' % (msg, data.decode(COD))
     conn.send(msg1.encode(COD)) #发送消息给已链接客户端
     conn.close() #关闭客户端链接
    tcpS.closel()

    miss

    zho***[email protected]

    7年前 (2019年07月27日)
  2. #0

    miss

    zho***[email protected]

    153

    TCP 客户端

    客户端结构:

    tcpc = socket() # 创建客户端套接字
    tcpc.connect() # 尝试连接服务器
    while True: # 通讯循环
     tcpc.send()/tcpc.recv() # 对话(发送/接收)
    tcpc.close() # 关闭客户套接字

    时间戳客户端实例:

    #!/usr/bin/python3
    # -*-coding:utf-8 -*-
    from socket import *
    from time import ctime
    HOST = '192.168.164.141' #服务端ip
    PORT = 21566 #服务端端口号
    BUFSIZ = 1024
    ADDR = (HOST, PORT)
    tcpCliSock = socket(AF_INET, SOCK_STREAM) #创建socket对象
    tcpCliSock.connect(ADDR) #连接服务器
    while True:
     data = input('>>').strip()
     if not data:
     break
     tcpCliSock.send(data.encode('utf-8')) #发送消息
     data = tcpCliSock.recv(BUFSIZ) #读取消息
     if not data:
     break
     print(data.decode('utf-8'))
    tcpCliSock.close() #关闭客户端

    miss

    zho***[email protected]

    7年前 (2019年07月27日)
  3. #0

    sean

    140***[email protected]

    42

    用socket实现的一个简易聊天室功能,支持@用户私聊。

    服务端代码:

    import socket
    import threading
    # 客户端地址 名称
    addr_name = {}
    # 所有客户端
    all_clients = []
    # 名称 客户端
    name_client = {}
    server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    host = socket.gethostname()
    port = 9999
    server.bind((host, port))
    server.listen(5)
    lock = threading.Lock()
    print("开启聊天室")
    def handle_sock(sock, addr):
     while True:
     try:
     data = sock.recv(1024)
     msg = data.decode("utf-8")
     print("send msg")
     from_name = addr_name[str(addr)]
     if msg.startswith('@'):
     index = msg.index(' ')
     # 私聊人 
     to_name = msg[1:index]
     # 接收者客户端 
     to_sock = name_client[to_name]
     # 发送的消息 
     to_msg = msg[index:]
     send_one(to_sock, addr, from_name + ":" + to_msg)
     else:
     # 群发消息 
     send_all(all_clients, addr, from_name + ":" + msg)
     except ConnectionResetError:
     exit_name = addr_name[str(addr)]
     exit_client = name_client[exit_name]
     all_clients.remove(exit_client)
     msg = exit_name + " 退出了群聊" 
     send_all(all_clients, addr, msg)
     break
    def send_all(socks, addr, msg):
     for sock in socks:
     sock.send(msg.encode("utf-8"))
    def send_one(sock, addr, msg):
     sock.send(msg.encode("utf-8"))
    while True:
     sock, addr = server.accept()
     name = sock.recv(1024).decode("utf-8")
     addr_name[str(addr)] = name
     name_client[name] = sock
     all_clients.append(sock)
     hello = name + "加入了聊天室" 
     send_all(all_clients, addr, hello)
     client_thread = threading.Thread(target=handle_sock, args=(sock, addr))
     client_thread.start()

    客户端代码:

    import socket
    import threading
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    host = socket.gethostname()
    port = 9999s.connect((host, port))
    name = "cc"
    s.send(name.encode("utf-8"))
    def receive_handle(sock, addr):
     while True:
     data = sock.recv(1024)
     print(data.decode("utf-8"))
    # 开启线程监听接收消息
    receive_thread = threading.Thread(target=receive_handle, args=(s, '1'))
    receive_thread.start()
    while True:
     re_data = input()
     s.send(re_data.encode("utf-8"))

    sean

    140***[email protected]

    5年前 (2021年06月24日)

点我分享笔记

  • 昵称 (必填)
  • 邮箱 (必填)
  • 引用地址

AltStyle によって変換されたページ (->オリジナル) /