简单来说,Web服务器是在运行在物理服务器上的一个程序,它永久地等待客户端(主要是浏览器,比如Chrome,Firefox等)发送请求。Web 服务器接受 Http Request,返回 Response,很多时候 Response 并不是静态文件,因此需要有一个应用程序根据 Request 生成相应的 Response。这里的应用程序主要用来处理相关业务逻辑,读取或者更新数据库,根据不同 Request 返回相应的 Response。两者之间的桥梁就是WSGI。

一直喜欢研究比较底层的技术,

之前就对python web框架web.py的运行机制比较迷惑, 大概学习了下之后发现flask框架以及Django框架都是基于python WSGI协议, python提供了一个简易的wsgi服务器实现--wsgiref, 在网站上找了两个例子运行了一下, 讲真, 第一次运行起来就比较懵逼, 尽管知道底层是依赖于socket, 但是深入一点就没有再研究了, 也看不懂。于是花了几天, 踏踏实实的看了源码, 一边百度一边理解,

终于学到了很多。有时候觉得自己让asp.net"惯坏"了, 因为微软闭源的关系, 自己掌握的基础知识并不全, 在很多的框架使用上, 仅仅会, 原理说个三三四四的, 还是差了很多, 果然开源就是好, 一言不合攻源码, 的确是学到了很多, python也是个很强大的语言, 这是我阅读其源码最大的感受。

对于本篇文章需要有的基础知识, 你可以从如下的链接中找到:

从零开始搭建论坛(1):Web服务器与Web框架

从零开始搭建论坛(2):Web服务器网关接口

讲真, 在没有读这两篇文章之前, 尽管我对web服务器,

web框架有了解, 但还是比较模糊, 这两篇文章写的很好。伯乐在线也是个不错的技术网站!

阅读完这两篇文章后, 那就有一定的基础了。先上代码:

# main.py
1 #coding: utf-8
2 importtime3 from resty importPathDispatcher4 from wsgiref.simple_server importmake_server5
6
7 _hello_resp = '''\8 9 
10 Hello {name}11 12 13 
 
Hello {name}!
14 15 ''' 
 
16
17
18 defhello_world(environ, start_response):19 #将响应状态和响应头交给WSGI server
20 #from wsgiref.handlers import SimpleHandler
21 start_response('200 OK', [('Content-type', 'text/html')])22 params = environ['params']23 resp = _hello_resp.format(name=params.get('name'))24 yield resp.encode('utf-8')25
26
27 _localtime_resp = '''\28 <?xml version="1.0"?>29 30 {t.tm_year}31 {t.tm_mon}32 {t.tm_mday}33 {t.tm_hour}34 {t.tm_min}35 {t.tm_sec}36 '''
37
38
39 deflocaltime(environ, start_response):40 #将响应状态和响应头交给WSGI server
41 start_response('200 OK', [('Content-type', 'application/xml')])42 resp = _localtime_resp.format(t=time.localtime)43 yield resp.encode('utf-8')44
45
46 #模拟客户端
47 if __name__ == '__main__':48 dispatcher =PathDispatcher()49 dispatcher.register('GET', '/hello', hello_world)50 dispatcher.register('GET', '/localtime', localtime)51
52 #启动一个简易的服务器
53 httpd = make_server('', 8080, dispatcher)54 print "Serving on port 8080..."
55 httpd.serve_forever() #开启循环机制复制代码
# resty.py
1 #coding: utf-8
2 importcgi3
4
5 defnotfound_404(environ, start_response):6 start_response('404 Not Found', [('Content-type', 'text/plain')])7 return ['Not Found']8
9
10 #使用了中间件
11 classPathDispatcher:12 def __init__(self):13 self.pathmap ={}14
15 #使此类的对象具有函数的能力, 对象能够接受传参, 就像函数调用一样, 此函数在handlers.py 中调用
16 def __call__(self, environ, start_response):17 path = environ['PATH_INFO']18 params = cgi.FieldStorage(environ['wsgi.input'], environ=environ)19
20 method = environ['REQUEST_METHOD'].lower()21 environ['params'] = {key: params.getvalue(key) for key in params} #字典推导式
22 handler =self.pathmap.get((method, path), notfound_404)23 returnhandler(environ, start_response)24
25 defregister(self, method, path, function):26 self.pathmap[method.lower(), path] =function27 return function复制代码

然后我分析下python自带的wsgi服务器主要文件的作用:

python 连接AD域创建账号 python怎么连接服务器_python 连接AD域创建账号

simple_server.py模拟了一个简单的web服务器, handlers.py是wsgi协议对http协议的封装处理函数。看下图吧:

python 连接AD域创建账号 python怎么连接服务器_Web_02

如上所示, 我大概归纳了一下不同py文件的作用。我之前对WSGI的作用比较模糊, 尽管知道WSGI就是连接web服务器与web应用程序之间的桥梁, 但是讲真!在客户端浏览器敲入换行后, python应用程序的具体执行了哪些重要的函数, 其调用顺序又是怎么来的。而且看着上面的代码,我问你一个问题:

__call__函数是啥时候调用的?在程序里面你看见了__call__的调用吗?尽管__call__函数是一个内置函数(我对其做了注释),

已经有了定义, 现在又有了实现代码, 那么调用程序呢? 原谅作为一个工科生的牛角尖, 看到程序里面有不明不白的调用实在憋屈-.-

看完下面的, 你应该就懂了

python 连接AD域创建账号 python怎么连接服务器_python 连接AD域创建账号_03

python 连接AD域创建账号 python怎么连接服务器_python_04

好好把<>看完, 同时要弄明白__call

__函数的作用。不过, 我先来解释下吧, 全程debug给你看看:

/编程环境: win7  // IDE: VS code   ///python version: 2.7.13

我分别在main.py, handlers, resty.py下了断点, 开始debug把....

python 连接AD域创建账号 python怎么连接服务器_python 连接AD域创建账号_05

python 连接AD域创建账号 python怎么连接服务器_python怎么连接web服务器_06

python 连接AD域创建账号 python怎么连接服务器_python_07

当调试控制台出现 "Serving on port 8080..." 表示此时等待客户端浏览器的访问了, 下面, 我们在浏览器中写入 http://127.0.0.1:8080/hello?name=Ryan

python 连接AD域创建账号 python怎么连接服务器_服务器_08

python 连接AD域创建账号 python怎么连接服务器_python_09

python 连接AD域创建账号 python怎么连接服务器_python怎么连接web服务器_10

python 连接AD域创建账号 python怎么连接服务器_服务器_11

python 连接AD域创建账号 python怎么连接服务器_服务器_12

图15注意下envron变量的值, 这就是一个dict类型的变量, 可以看到, 我们在浏览器中的 "?"后面的key-value都给保存进来了。传给了python应用程序。

当然, 要完成一个完整的url访问肯定不止这些函数模块的调用,

这就是主要的调用而已, 而且这已经很好的解释了我之前的问题了, 好了, 根据图自己去理解吧