基础与概念

  众所周知,对于所有的web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端

  web框架分两类:一类是包括socket和业务逻辑(tornado),另一类就是只负责业务逻辑

  对于第二类,没有socket就要使用其他的服务器程序,比如wsgi,它负责封装客户的请求信息,我们只要按它提供的方法获取数据

  首先我们明白了其本质就是socket,如果从socket开始开发应用程序那么效率就太低了,正确的做法是底层的socket处理代码由专门的服务器软件实现,而对于真实开发中的python web程序来说也是一般会分为两个部分:服务器程序和应用程序,服务器程序负责对socket服务器进行封装,并在请求到来时,对请求的各种数据进行整理,应用程序则负责具体的逻辑处理。

  WSGI(Web Server Gateway Interface)是一种规范,它定义了使用python编写的web app与web server之间接口格式,实现web app与web server间的解耦。

import socket

def handle_request(client):
#接收客户端信息并进行处理
buf = client.recv(1024)
client.send("HTTP/1.1 200 OK\r\n\r\n".encode())
client.send("Hello, Seven".encode()) def main():
# 实例socket对象
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 绑定端口
sock.bind(('localhost',8888))
# 设置监听客户端数量
sock.listen(5) while True:
# 获取客户端socket对象和端口
connection, address = sock.accept()
handle_request(connection)
connection.close() if __name__ == '__main__':
main()

python标准库提供的独立WSGI服务器称为wsgiref

from wsgiref.simple_server import make_server

def RunServer(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
return ['<h1>Hello, web!</h1>'.encode('utf-8')] #py3
# return '<h1>Hello, web!</h1>' py2 if __name__ == '__main__':
httpd = make_server('', 8888, RunServer)
print("Serving HTTP on port 8888...")
httpd.serve_forever()

自定义Web框架

一、框架

通过python标准库提供的wsgiref模块开发一个自己的web框架

from wsgiref.simple_server import make_server

def index():
return ['index'.encode("utf-8")] def login():
return ['login'.encode("utf-8")] def routers():
# 路由函数
urlpatterns = (
('/index/',index),
('/login/',login),
) return urlpatterns def RunServer(environ, start_response):
# 响应头
start_response('200 OK', [('Content-Type', 'text/html')])
# 获取客户端url信息
url = environ['PATH_INFO']
# 执行路由函数,获取路由元组
urlpatterns = routers()
func = None
for item in urlpatterns:
# 判断当前url,对应获取函数名
if item[0] == url:
func = item[1]
break
if func:
return func()
else:
return ['404 not found'.encode("utf-8")] if __name__ == '__main__':
httpd = make_server('', 8888, RunServer)
print("Serving HTTP on port 8888...")
httpd.serve_forever()

2、模板引擎

在上一步骤中,返回到浏览器给客户看的只是一个简单的字符串,而在真实的生活场景里,我们看到更多的是一个复杂的HTML规则字符串,所以我们一般将要返回给客户的HTML写在指定文件中,然后返回

from wsgiref.simple_server import make_server

def index():
# return 'index'
f = open('views/index.html')
data = [f.read().encode()]
return data def login():
# return 'login'
f = open('views/login.html')
data = [f.read().encode()]
return data def routers(): urlpatterns = (
('/index/', index),
('/login/', login),
) return urlpatterns def run_server(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
url = environ['PATH_INFO']
urlpatterns = routers()
func = None
for item in urlpatterns:
if item[0] == url:
func = item[1]
break
if func:
return func()
else:
return ['404 not found'.encode()] if __name__ == '__main__':
httpd = make_server('', 8888, run_server)
print("Serving HTTP on port 8888...")
httpd.serve_forever()

可能说,你对这还不满意,因为只是一个静态页面,根本就没有动态效果,好,那怎么给客户返回动态内容??

  • 自定义一套特殊的语法,进行替换
  • 使用开源工具jinja2,遵循其指定语法
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
<h1>{{name}}</h1>
{% for item in user_list %}
<li>{{item}}</li>
{% endfor %}
</body>
</html>

index.html

from wsgiref.simple_server import make_server
from jinja2 import Template def index():
# return 'index' # template = Template('Hello {{ name }}!')
# result = template.render(name='John Doe') f = open('views\index.html')
result = f.read()
template = Template(result)
data = template.render(name='John Doe', user_list=['alex', 'eric'])
return [data.encode('utf-8')] def login():
# return 'login'
f = open('views/login.html')
data = [f.read().encode("utf-8")]
return data def routers(): urlpatterns = (
('/index/', index),
('/login/', login),
) return urlpatterns def run_server(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
url = environ['PATH_INFO']
urlpatterns = routers()
func = None
for item in urlpatterns:
if item[0] == url:
func = item[1]
break
if func:
return func()
else:
return ['404 not found'.encode("utf-8")] if __name__ == '__main__':
httpd = make_server('', 8888, run_server)
print("Serving HTTP on port 8888...")
httpd.serve_forever()

  一个web框架应该包括路由系统和模板引擎等基本的东西,让我们看看一个高仿真自定义web框架:

from wsgiref.simple_server import make_server
import time
def new():
f = open('s1.html', 'r')
data = f.read()
f.close() # 模拟模板引擎处理数据,将html里文件的item替换成其他数据
# 这里仅仅是用字符串的替换来最简单的说明下原理
new_data = data.replace("{{item}}", str(time.time()))
return new_data def index():
f = open('index.html', 'r')
data = f.read()
f.close()
return data def home():
return 'home' URLS = { # 定义一个字典,简单模拟路由系统
"/new": new, # 一个url对应一个函数处理
"/index": index,
"/home": home,
} def application(environ, start_response):
start_response('200 OK', [('Content-Type', 'text/html')])
url = environ['PATH_INFO']
if url in URLS.keys(): # 不同的url执行不同的函数
func_name = URLS[url]
ret = func_name()
else:
ret = "404"
return [ret.encode('utf-8')] # 最后将结果返回 httpd = make_server('', 8000, application)
httpd.serve_forever()

python基础-第十三篇-13.1web框架本质的更多相关文章

  1. python基础-第十三篇-13.2Web框架之Tornado

    Tornado是非阻塞异步web frame,而且速度相当快,得力于其非阻塞的方式和对epoll的运用 Tornado每秒可以处理数以千计的链接,所以它可以有效的处理C10K问题 下载安装 pip3 ...

  2. Python基础【第一篇】

     一.Python简介 Python的创始人(Guido von Rossum 荷兰人),Guido希望有一种语言既能像C一样方便地调用操作系统的功能接口,也能像shell脚本一样,轻松地实现编程,A ...

  3. Python 学习 第十三篇:数据的读写-文件、DataFrame、json

    Python的文件是一个重要的对象,使用open()函数来打开文件,创建文件对象,进行文件的读写操作.当数据用于交换信息时,通常需要把数据保存为有格式的文本数据,可以保存为有特定的行分隔符和列分隔符的 ...

  4. Python 基础学习 总结篇

    Python 基础学习总结 先附上所有的章节: Python学习(一)安装.环境配置及IDE推荐 Python学习(二)Python 简介 Python学习(三)流程控制 Python学习(四)数据结 ...

  5. python基础-第六篇-6.2模块

    python之强大,就是因为它其提供的模块全面,模块的知识点不仅多,而且零散---一个字!错综复杂 没办法,二八原则抓重点咯!只要抓住那些以后常用开发的方法就可以了,哪些是常用的?往下看--找答案~ ...

  6. python基础知识第一篇(认识Python)

    开发语言: 高级语言:python java php c++ 生成的字节码 字节码转换为机器码 计算机识别运行 低级语言:C 汇编 生成的机器码 PHP语言:适用于网页,局限性 Python,Java ...

  7. python、第七篇:ORM框架SQLAlchemy

    一 介绍 SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作,简言之便是:将对象转换成SQL,然后使用数据API执行SQL并获取 ...

  8. Python基础第三篇

    一.collections系列 Counter是对字典类型的补充,用于追踪值的出现次数,具备字典的所有功能 + 自己的功能 1.计数器Counter import collections a='aba ...

  9. python基础-第十篇-10.1HTML基础

    htyper text markup language 即超文本标记语言 超文本:就是指页面内可以包含图片.链接,甚至音乐,程序等非文字元素 标记语言:标记(标签)构成的语言 网页==HTML文档,由 ...

随机推荐

  1. C语言 · 利息计算

    算法提高 利息计算   时间限制:1.0s   内存限制:512.0MB      编制程序完成下述任务:接受两个数,一个为用户一年期定期存款金额,一个为按照百分比格式表示的利率:程序计算一年期满 后 ...

  2. C语言 · 最大乘积

      算法提高 最大乘积   时间限制:1.0s   内存限制:512.0MB      问题描述 对于n个数,从中取出m个数,如何取使得这m个数的乘积最大呢? 输入格式 第一行一个数表示数据组数 每组 ...

  3. codeforces 429 On the Bench dp+排列组合 限制相邻元素,求合法序列数。

    限制相邻元素,求合法序列数. /** 题目:On the Bench 链接:http://codeforces.com/problemset/problem/840/C 题意:求相邻的元素相乘不为平方 ...

  4. jquery autocomplete s.toLowerCase(); 对象不支持此属性或方法

    今天发现了一个问题,自动提示删掉后再输入,会出现 s.toLowerCase(); 对象不支持此属性或方法的错误,后来格式化了jquery的autocomplete发现他是在matchSubset方法 ...

  5. EasyUI Tree checkbox node

    tree插件允许你创建checkbox tree,如果你点击节点的checkbox,被点击的节点信息得到下和上的继承.例如,点击tomato节点的checkbox,你可以看到vegetables节点现 ...

  6. 向服务器发送josn字符串,服务器端解析

    <script type="text/javascript"> $(function () { $("#btnsave").click(functi ...

  7. C++ 类的继承二(赋值兼容性原则)

    //赋值兼容性原则 #include<iostream> using namespace std; class PointA{ public: PointA(){ x = ; y = ; ...

  8. C++ 类的对象管理模型初讲

    //类的对象管理模型初讲 #include<iostream> using namespace std; class PointA{ private: int x;//占据4个字节大小的内 ...

  9. 【BZOJ】1657: [Usaco2006 Mar]Mooo 奶牛的歌声(单调栈)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1657 这一题一开始我想到了nlog^2n的做法...显然可做,但是麻烦.(就是二分+rmq) 然后我 ...

  10. 转载:Python十分钟入门

    Python十分钟入门:http://python.jobbole.com/23425/