框架: framework

DRP原则: Don’t repeat yourself
特指为解决一个开放性问题而设计的具有一定约束性的支撑结构,使用框架可以快速开发特定系统。去除重复部分

web应用的流程:

//浏览器发送一个HTTP请求
//服务器收到请求,生成一个HTML文档
//服务器把HTML文件作为HTTP响应的BODY发送给浏览器
//浏览器最终收到HTTP响应,从BODY中读取HTML文件并显示

对于所有的Web应用,本质是一个socket服务端,用户的浏览器是一个socket客户端。

import socket
def main():
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    sock.bind(('localhost',8089))  # url为 127.0.0.1:8089
    sock.listen(5) # 监听数量

    while True:
        connection, address = sock.accept()
        buf = connection.recv(1024)
        print(buf.decode('utf8'))  # 客户端返回的数据

        connection.sendall(bytes("HTTP/1.1 201 OK\r\n\r\n","utf8"))  # 有些浏览器可以不用这句,因为浏览器解析方式不同 在这之后就可以发送网页代码
        # connection.sendall(bytes("<h1 style='color:yellow'>Hello,World</h1>", "utf8"))  # <h1 style='color:yellow'>html和css共同完成文字效果,js完成动态效果
        with open('lesson1.html','rb') as f:
            data=f.read()
        connection.sendall(data)
        connection.close()

if __name__ == '__main__':
    main()
# 浏览器访问127.0.0.1:8089 连接服务器后返回的信息 
GET / HTTP/1.1
Host: localhost:8089
Connection: keep-alive
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.102 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Sec-Fetch-Site: none
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Sec-Fetch-Dest: document
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9
Cookie: Pycharm-a7943e9=3e8b6d40-197e-46ce-b197-ed89497751b3


GET /favicon.ico HTTP/1.1
Host: localhost:8089
Connection: keep-alive
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/85.0.4183.102 Safari/537.36
Accept: image/avif,image/webp,image/apng,image/*,*/*;q=0.8
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: no-cors
Sec-Fetch-Dest: image
Referer: http://localhost:8089/
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9
Cookie: Pycharm-a7943e9=3e8b6d40-197e-46ce-b197-ed89497751b3

最简单的Web应用就是先把HTML用文件保存好,用一个现成的HTTP服务器软件,接收用户请求,从中读取HTML文件并返回。

如果要动态生成HTML,就需要把所有步骤自己实现。不过,接受HTTP请求、解析HTTP请求、发送HTTP响应都是底层代码。正确的做法是底层代码由专门的服务器软件实现。用Python专注于生成HTML文档。所以,需要一个统一的接口,使得可以只用Python编写Web业务。

WSGI:Web Server Gateway Interface。

from wsgiref.simple_server import make_server
# wsgiref为python内置服务器

def application(environ, start_response):
	# environ为请求对象 封装了所有收到的请求信息 
    start_response('200 OK', [('Content-Type', 'text/html')])
    # start_response可以很方便地设置发送的响应头 
    return [b'<h1>Hello, web!</h1>'] # 响应体

# 封装socket对象以及准备过程(socket bind listen)
httpd = make_server('', 8080, application) #ip地址 端口 函数名

print('Serving HTTP on port 8080...')
# 开始监听HTTP请求:
httpd.serve_forever()
# 浏览器访问 http://127.0.0.1:8080/

整个application()函数本身没有涉及到任何解析HTTP的部分,即底层代码不需要自己编写,只负责在更高层次上考虑如何响应请求就可以了。application()函数必须由WSGI服务器来调用。

Python内置了一个WSGI服务器,这个模块叫wsgiref。
application()函数就是符合WSGI标准的一个HTTP处理函数,它接收两个参数:environ:一个包含所有HTTP请求信息的dict对象;start_response:一个发送HTTP响应的函数。

在application()函数中,调用:start_response(‘200 OK’, [(‘Content-Type’, ‘text/html’)]) 就发送了HTTP响应的Header,注意Header只能发送一次,也就是只能调用一次start_response()函数。start_response()函数接收两个参数,一个是HTTP响应码,一个是一组list表示的HTTP Header,每个Header用一个包含两个str的tuple表示。

b'Hello, web!'

有了WSGI,我们关心的就是如何从environ这个dict对象拿到HTTP请求信息,然后构造HTML,通过start_response()发送Header,最后返回Body。

根据分析请求对象里不同的路径信息返回不同的页面

from wsgiref.simple_server import make_server
# wsgiref为python内置服务器

def application(environ, start_response):
	# environ为请求对象 封装了所有的请求信息
	# print("environ:",environ)
	# print(environ["PATH_INFO"]) # 返回请求路径/book 可以通过浏览器输入的不同的路径返回不同的页面
	path = environ["PATH_INFO"]
	start_response('200 OK', [('Content-Type', 'text/html')])

	if path=="/book":
		return [ b'<h1>Hello, book!</h1>' ]
	elif path=="/web":
		return [ b'<h1>Hello, web!</h1>' ]
	else:
		return [ b'<h1>404</h1>' ]
	#start_response可以很方便地设置响应头

# 封装socket对象以及准备过程(socket bind listen)
httpd = make_server('', 8080, application) #ip地址 端口 函数名

print('Serving HTTP on port 8080...')
# 开始监听HTTP请求:
httpd.serve_forever()

解耦操作:将返回页面独立成为函数,使得联系不那么紧密

from wsgiref.simple_server import make_server
# wsgiref为python内置服务器

def f1(request):
	return [ b'<h1>Hello, book!</h1>' ]

def f2(request):
	return [ b'<h1>Hello, web!</h1>' ]

def application(environ, start_response):
	# environ为请求对象 封装了所有的请求信息
	# print("environ:",environ)
	# print(environ["PATH_INFO"]) # 返回请求路径/book 可以通过浏览器输入的不同的路径返回不同的页面
	path = environ["PATH_INFO"]
	start_response('200 OK', [('Content-Type', 'text/html')])

	if path=="/book":
		return f1(environ) # 使得f1f2想用请求信息的时候也可以拿到
	elif path=="/web":
		return f2(environ)
	else:
		return [ b'<h1>404</h1>' ]
	#start_response可以很方便地设置响应头

# 封装socket对象以及准备过程(socket bind listen)
httpd = make_server('', 8080, application) #ip地址 端口 函数名
print('Serving HTTP on port 8080...')
# 开始监听HTTP请求:
httpd.serve_forever()

优化 if else 语句

from wsgiref.simple_server import make_server
# wsgiref为python内置服务器

def f1(request):
	return [ b'<h1>Hello, book!</h1>' ]

def f2(request):
	return [ b'<h1>Hello, web!</h1>' ]

def routers():
	urlpatterns = (
		('/book',f1),
		('/web',f2),
	)
	return urlpatterns

def application(environ, start_response):
	# environ为请求对象 封装了所有的请求信息
	# print("environ:",environ)
	# print(environ["PATH_INFO"]) # 返回请求路径/book 可以通过浏览器输入的不同的路径返回不同的页面
	path = environ["PATH_INFO"]
	start_response('200 OK', [('Content-Type', 'text/html')])

	urlpatterns = routers()
	func = None
	for item in urlpatterns: # 使用循环遍历简化if else
		if item[0] == path:
			func = item[1]
			break
	if func:
		return func(environ)
	else:
		return ["<h1>404</h1>".encode("utf8")]

# 封装socket对象以及准备过程(socket bind listen)
httpd = make_server('', 8080, application) #ip地址 端口 函数名
print('Serving HTTP on port 8080...')
# 开始监听HTTP请求:
httpd.serve_forever()

在后端改变HTML发送给浏览器渲染

from wsgiref.simple_server import make_server
# wsgiref为python内置服务器
import time

def f1(request):
	return [b'<h1>Hello, book!</h1>']

def f2(request):
	return [b'<h1>Hello, web!</h1>']

def curtime(request):
	# print(time.time()) #1603349683.058512
	curttime=time.ctime(time.time())
	# print(curttime) #Thu Oct 22 14:54:43 2020
	return ['<h1>current_time:!curttime!</h1>'.replace("!curttime!",str(curttime)).encode("utf8")]

def routers():
	urlpatterns = (
		('/book',f1),
		('/web',f2),
		('/time',curtime),
	)
	return urlpatterns

def application(environ, start_response):
	# environ为请求对象 封装了所有的请求信息
	# print("environ:",environ)
	# print(environ["PATH_INFO"]) # 返回请求路径/book 可以通过浏览器输入的不同的路径返回不同的页面
	path = environ["PATH_INFO"]
	start_response('200 OK', [('Content-Type', 'text/html')])

	urlpatterns = routers()
	func = None
	for item in urlpatterns: # 使用循环遍历简化if else
		if item[0] == path:
			func = item[1]
			break
	if func:
		return func(environ)
	else:
		return ["<h1>404</h1>".encode("utf8")]

# 封装socket对象以及准备过程(socket bind listen)
httpd = make_server('', 8080, application) #ip地址 端口 函数名
print('Serving HTTP on port 8080...')
# 开始监听HTTP请求:
httpd.serve_forever()

将上述代码解耦,将不同功能的代码块存入对应名称的py文件中。

MVC模式
经典MVC模式。所谓MVC就是将web应用分为模型(M),控制器©,视图(V)三层;他们之间以一种松耦合的方式连接在一起。
模型M负责业务对象与数据库的对象(ORM),视图V负责与用户的交互(页面),控制器C接受用户的输入调用模型和视图完成用户的请求。

若函数中需要进行数据库相关操作则在模型M中进行。
视图存储传输给浏览器页面的请求体HTML文件
控制器C存储上述代码中的自定义函数

浏览器发送请求给控制器C,C对模型M中的数据库进行操作,且对视图view的前端代码操作后返回响应信息。

MTV模式
Django使用MTV模式。本质上与MVC模式没有什么差别,也是各组件之间保持松耦合关系
Model(模型):负责业务对象与数据库的对象(ORM)
Template(模版):负责如何把页面展示给用户 相当于MVC的view
View(视图):负责业务逻辑,并在适当的时候调用Model和Template 相当于MVC的controller

此外,Django还有一个url分发器,它的作用是将一个个URL的页面请求分发给不同的view处理,view再调用相应的Model和Template

具体流程如下:
在浏览器里输入url,通过url分发器匹配相应的视图函数,视图函数通过models从数据库中取数据,后把要展示的数据返回给模板,模板就是HTML文件返回给浏览器。

Django文件创建流程:

django
    #  安装:pip3 install django
    
    #  创建project
       # 打开文件夹,在路径栏中输入cmd
       django-admin startproject mysite
       ---mysite
		   ---mysite
		       ---settings.py(包含了项目的默认设置,包括数据库信息,调试标志以及其他一些工作的变量。)
		       ---url.py(负责把URL模式映射到应用程序VIEWS。)
		   ---- manage.py(Django项目里面的工具,通过它可以调用django shell和数据库等。)  
       
    #  创建APP   在mannage.py目录下执行
       python manage.py startapp app01
       ---mysite
		   ---mysite
		       ---settings.py
		       ---url.py
		   ---app01
		       ---views.py
		       ---models.py
		       ...
		   ---- manage.py
    #  使用模版
        render(req,"index.html")  # setting中有路径拼接,此时不需要进行绝对路径
        
    #  启动项目
        python manage.py runserver  127.0.0.1:8090
        访问:http://127.0.0.1:8080/
        可以在cmd下或者在pycharm的terminal下运行
    
    #   生成同步数据库的脚本:
        python manage.py makemigrations  
    #   同步数据库:  
        python manage.py migrate   

在url文件中进行url配置

from django.contrib import admin
from django.conf.urls import url
from app01 import views

urlpatterns = [
    url(r'^admin/', admin.site.urls),
    url(r'^cur_time',views.cur_time),# r表示原生字符串 ^匹配开头
    url(r'^userinfo', views.info),
]

里面涉及到的函数在app01 中的views文件中定义

from django.shortcuts import render,HttpResponse
import datetime

def cur_time(request):
	times=datetime.datetime.now()
	# HttpResponse 返回字符串
	# return HttpResponse("<h1>ok</h1>")
	# 渲染 解析后再实例化发送
	return render(request,"cur_time.html",{'abc':times}) # 模板规则,用变量times代替html文件中的{{abc}}

user_list=list()
def info(req):
	# print("method:::",req.method) # 在浏览器中输入网址回车时 采用的是get
	if req.method=="POST":
		name=req.POST.get("username",None)
		sex=req.POST.get("gender",None)
		email=req.POST.get("email",None)
		user={"name":name,"sex":sex,"email":email}
		user_list.append(user) # 但是放在列表中会断电后丢失数据 所以需要使用数据库
		print(user_list)
	return render(req,"index.html",{"user_list":user_list})
django.middleware.csrf.CsrfViewMiddleware
BASE_DIR1=os.path.dirname(os.path.dirname(os.path.abspath(__file__)))'DIRS':[os.path.join(BASE_DIR1,"templates")]

文件使用的模板放入项目下的templates文件夹内
cur_time.html

index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<!--action 处理数据的url 且默认当前的域名和端口-->
<form action="/userinfo/" method="post">
    <p>姓名<input type="text" name="username"></p>
    <p>性别<input type="text" name="gender"></p>
    <p>邮箱<input type="text" name="email"></p>
    <p><input type="submit" value="submit"></p>
</form>
    <hr>
    <h1>数据展示</h1>
    <table border="1px">
        <tr>
        <td>姓名</td>
        <td>性别</td>
        <td>邮箱</td>
        </tr>
        <!--后端render根据模板语言的语法规范渲染出来的 前端不做渲染-->
        {% for i in user_list %}
        <tr>
        <td>{{i.name}}</td>
        <td>{{i.sex}}</td>
        <td>{{i.email}}</td>
        </tr>
        {% endfor %}
    </table>
</body>
</html>

若要使用数据库,则需要用到models模块
首先需要在setting的INSTALLED_APPS中加入’app01’,

INSTALLED_APPS = 
    'app01',

在models中创建表

from django.db import models
class UserInfor(models.Model):
    username=models.CharField(max_length=64,null=True)
    sex=models.CharField(max_length=64,null=True)
    email=models.CharField(max_length=64,null=True)

在views中设置函数

from app01 import models
def info(req):
	if req.method=="POST":
		u = req.POST.get("username", None)
		s = req.POST.get("gender", None)
		e = req.POST.get("email", None)
		models.UserInfor.objects.create(
			username=u,
			sex=s,
			email=e
		)
	user_list=models.UserInfor.objects.all() # 取数据
	return render(req,"index.html",{"user_list":user_list})
{{i.name}}{{i.username}}