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


Posted in Python onAugust 25, 2017

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基于隐马尔可夫模型实现中文拼音输入
Apr 01 Python
EM算法的python实现的方法步骤
Jan 02 Python
快速了解Python中的装饰器
Jan 11 Python
Python实现简易版的Web服务器(推荐)
Jan 29 Python
python批量读取txt文件为DataFrame的方法
Apr 03 Python
如何使用Python的Requests包实现模拟登陆
Apr 27 Python
对numpy.append()里的axis的用法详解
Jun 28 Python
Pytorch 抽取vgg各层并进行定制化处理的方法
Aug 20 Python
python 画函数曲线示例
Dec 04 Python
pd.DataFrame统计各列数值多少的实例
Dec 05 Python
PyCharm无法识别PyQt5的2种解决方法,ModuleNotFoundError: No module named 'pyqt5'
Feb 17 Python
python实现时间序列自相关图(acf)、偏自相关图(pacf)教程
Jun 03 Python
详解Python实现多进程异步事件驱动引擎
Aug 25 #Python
python基础while循环及if判断的实例讲解
Aug 25 #Python
itchat和matplotlib的结合使用爬取微信信息的实例
Aug 25 #Python
用 Python 爬了爬自己的微信朋友(实例讲解)
Aug 25 #Python
详解python基础之while循环及if判断
Aug 24 #Python
用Python实现随机森林算法的示例
Aug 24 #Python
python利用urllib实现爬取京东网站商品图片的爬虫实例
Aug 24 #Python
You might like
《PHP边学边教》(02.Apache+PHP环境配置――下篇)
2006/12/13 PHP
php模板函数 正则实现代码
2012/10/15 PHP
CI框架源码阅读,系统常量文件constants.php的配置
2013/02/28 PHP
php+ajax无刷新上传图片实例代码
2015/11/17 PHP
thinkphp框架page类与bootstrap分页(美化)
2017/06/25 PHP
最新的10款jQuery内容滑块插件分享
2011/09/18 Javascript
js中的cookie的读写操作示例详解
2014/04/17 Javascript
node.js中的fs.rmdir方法使用说明
2014/12/16 Javascript
一道常被人轻视的web前端常见面试题(JS)
2016/02/15 Javascript
利用jquery实现实时更新歌词的方法
2017/01/06 Javascript
js实现图片旋转 js滚动鼠标中间对图片放大缩小
2017/07/05 Javascript
vue的diff算法知识点总结
2018/03/29 Javascript
vue.js实现格式化时间并每秒更新显示功能示例
2018/07/07 Javascript
vue2.0父子组件间传递数据的方法
2018/08/16 Javascript
vue实现element-ui对话框可拖拽功能
2018/08/17 Javascript
vue项目在安卓低版本机显示空白的原因分析(两种)
2018/09/04 Javascript
javascript实现考勤日历功能
2018/11/29 Javascript
AjaxFileUpload.js实现异步上传文件功能
2019/04/19 Javascript
微信小程序结合Storage实现搜索历史效果
2019/05/18 Javascript
javascript中call,apply,bind的区别详解
2020/12/11 Javascript
[03:07]DOTA2英雄基础教程 冰霜诅咒极寒幽魂
2013/12/06 DOTA
web.py 十分钟创建简易博客实现代码
2016/04/22 Python
python添加模块搜索路径方法
2017/09/11 Python
浅谈pandas中Dataframe的查询方法([], loc, iloc, at, iat, ix)
2018/04/10 Python
python 请求服务器的实现代码(http请求和https请求)
2018/05/25 Python
mac安装pytorch及系统的numpy更新方法
2018/07/26 Python
Python图像处理实现两幅图像合成一幅图像的方法【测试可用】
2019/01/04 Python
TensorFLow 不同大小图片的TFrecords存取实例
2020/01/20 Python
pycharm 对代码做静态检查操作
2020/06/09 Python
理肤泉美国官网:La Roche-Posay
2018/01/17 全球购物
静态成员和非静态成员的区别
2012/05/12 面试题
人事主管岗位职责
2014/01/30 职场文书
2014保险公司个人工作总结
2014/12/09 职场文书
中班下学期个人总结
2015/02/12 职场文书
建房合同协议书
2016/03/21 职场文书
浅谈Redis的keys命令到底有多慢
2021/10/05 Redis