使用Python实现简单的服务器功能

发布时间 - 2026-01-11 02:58:07    点击率:

socket接口是实际上是操作系统提供的系统调用。socket的使用并不局限于Python语言,你可以用C或者Java来写出同样的socket服务器,而所有语言使用socket的方式都类似(Apache就是使用C实现的服务器)

Web框架就是提前写好了服务器。不能跨语言的使用框架。框架的好处在于帮你处理了一些细节,从而实现快速开发,但同时受到python本身性能的限制。我们已经看到,许多成功的网站都是利用动态语言(比如Python, Ruby或者PHP,比如twitter和facebook)快速开发,在网站成功之后,将代码转换成诸如C和JAVA这样一些效率比较高的语言,从而让服务器能更有效率的面对每天亿万次的请求。

TCP/IP和socket

socket是进程间通信的一种方法 ,它是基于网络传输协议的上层接口。socket有许多种类型,比如基于TCP协议或者UDP协议(两种网络传输协议)。其中又以TCP socket最为常用。TCP socket与双向管道(duplex PIPE)有些类似,一个进程向socket的一端写入或读取文本流,而另一个进程可以从socket的另一端读取或写入,比较特别是,这两个建立socket通信的进程可以分别属于两台不同的计算机,只需要通过网络链路和通信协议连同即可

TCP协议,就是规定了一些通信的守则,以便在网络环境下能够有效实现上述进程间通信过程。双向管道(duplex PIPE)存活于同一台电脑中,所以不必区分两个进程的所在计算机的地址,而socket必须包含有地址信息,以便实现网络通信。

一个socket包含四个地址信息: 两台计算机的IP地址和两个进程所使用的端口(port)。IP地址用于定位计算机,而port用于定位进程 (一台计算机上可以有多个进程分别使用不同的端口)。

TCP socket

我们可以让某台计算机作为服务器。服务器开放自己的端口,被动等待其他计算机连接我们这个port。当其他计算机作为客户,主动使用socket连接到服务器的时候,服务器就开始为客户提供服务。其实也就是二者建立了双向联通。

在Python中,我们使用标准库中的socket包来进行底层的socket编程。

首先是服务器端,我们使用bind()方法来赋予socket以固定的地址和端口,并使用listen()方法来被动的监听该端口。当有客户尝试用connect()方法连接的时候,服务器使用accept()接受连接,从而建立一个连接的socket:

server.py

import socket
# Address
HOST = '127.0.0.1'
PORT = 8000
# Configure socket
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind((HOST, PORT))
#socket.socket()创建一个socket对象,并说明socket使用的是IPv4(AF_INET,IP version 4)
#和TCP协议(SOCK_STREAM)。
# passively wait, 3: maximum number of connections in the queue
s.listen(3)
# accept and establish connection
conn, addr = s.accept()
# receive message
request = conn.recv(1024)
print ('request is: ',request)
print ('Connected by', addr)
# send message
reply = 'Yes'
""" 
好像3.* socket.send 传递的数据必须是bytes。不能是str。发送和接收数据时做下编码转换就可以了。修改如下:s.sendall(request.encode())
"""
conn.sendall(reply.encode())
# close connection
conn.close()

client.py

import socket
# Address
HOST = '127.0.0.1'
PORT = 8000
request = 'can you hear me?'
# configure socket
s    = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((HOST, PORT))
""" 

好像3.* socket.send 传递的数据必须是bytes。不能是str。发送和接收数据时做下编码转换就可以了。修改如下:s.sendall(request.encode())

"""
# send message
s.sendall(request.encode())
# receive message
reply  = s.recv(1024)
print ('reply is: ',reply)
# close connection
s.close()

我们对socket的两端都可以调用recv()方法来接收信息,调用sendall()方法来发送信息。这样,我们就可以在分处于两台计算机的两个进程间进行通信了。当通信结束的时候,我们使用close()方法来关闭socket连接。

基于TCP socket的HTTP服务器

使用TCP socket来为两台远程计算机建立连接,就可以通信了。
然而socket传输自由度太高,从而带来很多安全和兼容的问题。我们往往利用一些应用层的协议(比如HTTP协议)来规定socket使用规则,以及所传输信息的格式。

HTTP协议利用请求-回应(request-response)的方式来使用TCP socket。

客户端向服务器发一段文本作为request,服务器端在接收到request之后,向客户端发送一段文本作为response。

在完成了这样一次request-response交易之后,TCP socket被废弃。下次的request将建立新的socket。

request和response本质上说是两个文本,只是HTTP协议对这两个文本都有一定的格式要求。

Httpserver.py

import socket
# Address
HOST = ''
PORT = 8000
# Prepare HTTP response
text_content = '''HTTP/1.x 200 OK 
Content-Type: text/html
<head>
<title>WOW</title>
</head>
<html>
<p>Wow, Python Server</p>
<IMG src="test.jpg"/>
</html>
'''
# Read picture, put into HTTP format
f = open('test.jpg','rb')
pic_content = '''
HTTP/1.x 200 OK 
Content-Type: image/jpg
'''
pic_content = pic_content.bytes()+ f.read()
f.close()
# Configure socket
s  = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind((HOST, PORT))
# infinite loop, server forever
while True:
  # 3: maximum number of requests waiting
  s.listen(3)
  conn, addr = s.accept()
  request  = conn.recv(1024)
  method  = request.split(' ')[0]
  src      = request.split(' ')[1]
  # deal with GET method
  if method == 'GET':
    # ULR  
    if src == '/test.jpg':
      content = pic_content
    else: content = text_content
    print ('Connected by', addr)
    print ('Request is:', request)
    conn.sendall(content.bytes())
  # close connection
  conn.close()

解读

服务器会根据request向客户传输的两条信息text_content和pic_content中的一条,作为response文本。

整个response分为起始行(start line), 头信息(head)和主体(body)三部分。

起始行就是第一行:HTTP/1.x 200 OK 它实际上又由空格分为三个片段,HTTP/1.x表示所使用的HTTP版本,200表示状态(status code),200是HTTP协议规定的,表示服务器正常接收并处理请求,OK是供人来阅读的status code。

头信息跟随起始行,它和主体之间有一个空行。这里的text_content或者pic_content都只有一行的头信息,text_content用来表示主体信息的类型为html文本:Content-Type: text/html

主体信息为html或者jpg文件的内容。(注意,对于jpg文件,我们使用'rb'模式打开,是为了与windows兼容。因为在windows下,jpg被认为是二进制(binary)文件,在UNIX系统下,则不需要区分文本文件和二进制文件。)

HttpClient.py

客户端程序,你可以自己写。当然一般用浏览器作为客户端。

request由客户端程序发给服务器。尽管request也可以像response那样分为三部分,但是request的格式与response的格式并不相同。request由客户发送给服务器,比如下面是一个request:

GET /test.jpg HTTP/1.x
Accept: text/*

起始行可以分为三部分,第一部分为请求方法(request method),第二部分是URL,第三部分为HTTP版本。request method可以有GET, PUT, POST, DELETE, HEAD。最常用的为GET和POST。GET是请求服务器发送资源给客户,POST是请求服务器接收客户送来的数据。当我们打开一个网页时,我们通常是使用GET方法;当我们填写表格并提交时,我们通常使用POST方法。第二部分为URL,它通常指向一个资源(服务器上的资源或者其它地方的资源)。像现在这样,就是指向当前服务器的当前目录的test.jpg。

按照HTTP协议的规定,服务器需要根据请求执行一定的操作。正如我们在服务器程序中看到的,我们的Python程序先检查了request的方法,随后根据URL的不同,来生成不同的response(text_content或者pic_content)。随后,这个response被发送回给客户端。

Test

终端运行上面的Httpserver程序,作为服务器端,再打开一个浏览器作为客户端。(如果有时间,你也完全可以用Python写一个客户端。原理与上面的TCP socket的客户端程序相类似。)
在浏览器的地址栏输入:127.0.0.1:8000

使用浏览器的调试功能 F12

GET / HTTP/1.1
Host: 127.0.0.1:8000
User-Agent: Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:14.0) Gecko/20100101 Firefox/14.0.1
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us,en;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive

浏览器接收到text_content之后,发现正文的html文本中有,知道需要获得text.jpg文件来补充为图片,立即发出了第二个请求:

GET /test.jpg HTTP/1.1
Host: 127.0.0.1:8000
User-Agent: Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:14.0) Gecko/20100101 Firefox/14.0.1
Accept: image/png,image/*;q=0.8,*/*;q=0.5
Accept-Language: en-us,en;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive
Referer: http://127.0.0.1:8000/

小结

1) 在我们上面的服务器程序中,我们用while循环来让服务器一直工作下去。实际上使用多线程的知识,将while循环中的内容改为多进程或者多线程工作。

2) 服务器程序还不完善,还可以让Python程序调用Python的其他功能,以实现更复杂的功能。比如说制作一个时间服务器,让服务器向客户返回日期和时间。你还可以使用Python自带的数据库,来实现一个完整的LAMP服务器。

3) socket包是比较底层的包。Python标准库中还有高层的包,比如SocketServer,SimpleHTTPServer,CGIHTTPServer,cgi。这些都包都是在帮助我们更容易的使用socket。这些包就很容易明白了。利用这些高层的包,你可以写一个相当成熟的服务器。

4) 在经历了所有的麻烦之后,发现框架是那么的方便,所以决定去使用框架。当然也可以参与到框架开发的热情。

总结

以上所述是小编给大家介绍的使用Python实现简单的服务器功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


# python  # 实现服务器  # Python多进程与服务器并发原理及用法实例分析  # 用Python实现一个简单的能够上传下载的HTTP服务器  # Python实现的简单文件传输服务器和客户端  # 400多行Python代码实现了一个FTP服务器  # python实现简单的TCP代理服务器  # python实现的文件同步服务器实例  # 用Python实现一个简单的多线程TCP服务器的教程  # python搭建简易服务器分析与实现  # python实现FTP服务器服务的方法  # Python实现的服务器示例小结【单进程、多进程、多线程、非阻塞式】  # 客户端  # 方法来  # 两台  # 就可以  # 你可以  # 可以用  # 这两个  # 当我们  # 小编  # 多线程  # 信了  # 自己的  # 库中  # 的是  # 都是  # 是一个  # 传输协议  # 是在  # 好了  # 还可以 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: EditPlus 正则表达式 实战(3)  C++时间戳转换成日期时间的步骤和示例代码  微信小程序制作网站有哪些,微信小程序需要做网站吗?  微信小程序 require机制详解及实例代码  Laravel安装步骤详细教程_Laravel环境搭建指南  如何在万网开始建站?分步指南解析  Laravel Eloquent:优雅地将关联模型字段扁平化到主模型中  android nfc常用标签读取总结  如何用AWS免费套餐快速搭建高效网站?  Laravel怎么使用Intervention Image库处理图片上传和缩放  如何用ChatGPT准备面试 模拟面试问答与职场话术练习教程  Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程  如何在宝塔面板创建新站点?  如何快速生成ASP一键建站模板并优化安全性?  网站制作价目表怎么做,珍爱网婚介费用多少?  如何在IIS中配置站点IP、端口及主机头?  英语简历制作免费网站推荐,如何将简历翻译成英文?  微信推文制作网站有哪些,怎么做微信推文,急?  INTERNET浏览器怎样恢复关闭标签页_INTERNET浏览器标签恢复快捷键与方法【指南】  如何快速搭建自助建站会员专属系统?  怎样使用JSON进行数据交换_它有什么限制  javascript日期怎么处理_如何格式化输出  Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】  专业企业网站设计制作公司,如何理解商贸企业的统一配送和分销网络建设?  nginx修改上传文件大小限制的方法  Linux系统命令中tree命令详解  在Oracle关闭情况下如何修改spfile的参数  如何快速搭建高效WAP手机网站吸引移动用户?  Laravel如何使用.env文件管理环境变量?(最佳实践)  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  如何在阿里云虚拟主机上快速搭建个人网站?  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  手机网站制作与建设方案,手机网站如何建设?  移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?  Laravel如何集成Inertia.js与Vue/React?(安装配置)  如何在宝塔面板中创建新站点?  公司门户网站制作流程,华为官网怎么做?  如何获取PHP WAP自助建站系统源码?  浅谈Javascript中的Label语句  什么是JavaScript解构赋值_解构赋值有哪些实用技巧  浅谈javascript alert和confirm的美化  JS中对数组元素进行增删改移的方法总结  Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比  php静态变量怎么调试_php静态变量作用域调试技巧【解答】  谷歌Google入口永久地址_Google搜索引擎官网首页永久入口  消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工  Laravel如何使用Service Provider注册服务_Laravel服务提供者配置与加载  如何在不使用负向后查找的情况下匹配特定条件前的换行符  如何在阿里云购买域名并搭建网站?  使用豆包 AI 辅助进行简单网页 HTML 结构设计