HTTP协议介绍
现今广泛使用的一个版本——HTTP 1.1
HTTP请求方法
HTTP/1.1协议中共定义了八种方法(动作)来以不同方式操作指定的资源:
GET
向指定的资源发出“显示”请求。使用GET方法应该只用在读取数据,而不应当被用于产生“副作用”的操作中,例如在Web Application中。其中一个原因是GET可能会被网络蜘蛛等随意访问。
HEAD
与GET方法一样,都是向服务器发出指定资源的请求。只不过服务器将不传回资源的本文部分。它的好处在于,使用这个方法可以在不必传输全部内容的情况下,就可以获取其中“关于该资源的信息”(元信息或称元数据)。
POST
向指定资源提交数据,请求服务器进行处理(例如提交表单或者上传文件)。数据被包含在请求本文中。这个请求可能会创建新的资源或修改现有资源,或二者皆有。
PUT
向指定资源位置上传其最新内容。
DELETE
请求服务器删除Request-URI所标识的资源。
TRACE
回显服务器收到的请求,主要用于测试或诊断。
OPTIONS
这个方法可使服务器传回该资源所支持的所有HTTP请求方法。用'*'来代替资源名称,向Web服务器发送OPTIONS请求,可以测试服务器功能是否正常运作。
CONNECT
HTTP/1.1协议中预留给能够将连接改为管道方式的代理服务器。通常用于SSL加密服务器的链接(经由非加密的HTTP代理服务器)。
注意事项:
方法名称是区分大小写的。当某个请求所针对的资源不支持对应的请求方法的时候,服务器应当返回状态码405(Method Not Allowed),当服务器不认识或者不支持对应的请求方法的时候,应当返回状态码501(Not Implemented)。
HTTP服务器至少应该实现GET和HEAD方法,其他方法都是可选的。当然,所有的方法支持的实现都应当匹配下述的方法各自的语义定义。此外,除了上述方法,特定的HTTP服务器还能够扩展自定义的方法。例如PATCH(由 RFC 5789 指定的方法)用于将局部修改应用到资源HTTP状态码
状态代码的第一个数字代表当前响应的类型:
1xx消息——请求已被服务器接收,继续处理
2xx成功——请求已成功被服务器接收、理解、并接受
3xx重定向——需要后续操作才能完成这一请求
4xx请求错误——请求含有词法错误或者无法被执行
5xx服务器错误——服务器在处理某个正确请求时发生错误URL
超文本传输协议(HTTP)的统一资源定位符将从因特网获取信息的五个基本元素包括在一个简单的地址中:
传送协议。
层级URL标记符号(为[//],固定不变)
访问资源需要的凭证信息(可省略)
服务器。(通常为域名,有时为IP地址)
端口号。(以数字方式表示,若为HTTP的默认值“:80”可省略)
路径。(以“/”字符区别路径中的每一个目录名称)
查询。(GET模式的窗体参数,以“?”字符为起点,每个参数以“&”隔开,再以“=”分开参数名称与数据,通常以UTF8的URL编码,避开字符冲突的问题)
片段。以“#”字符为起点
以http://www.luffycity.com:80/news/index.html?id=250&page=1 为例, 其中:
http,是协议;
www.luffycity.com,是服务器;
80,是服务器上的网络端口号;
/news/index.html,是路径;
?id=250&page=1,是查询。
大多数网页浏览器不要求用户输入网页中“http://”的部分,因为绝大多数网页内容是超文本传输协议文件。同样,“80”是超文本传输协议文件的常用端口号,因此一般也不必写明。一般来说用户只要键入统一资源定位符的一部分(www.luffycity.com:80/news/index.html?id=250&page=1)就可以了。
由于超文本传输协议允许服务器将浏览器重定向到另一个网页地址,因此许多服务器允许用户省略网页地址中的部分,比如 www。从技术上来说这样省略后的网页地址实际上是一个不同的网页地址,浏览器本身无法决定这个新地址是否通,服务器必须完成重定向的任务。HTTP请求格式

HTTP响应格式

半成品自定义WEB框架
import socket
# 创建一个socket对象
sk = socket.socket()
# 绑定IP和端口
sk.bind(('127.0.0.1', 8000))
# 监听
sk.listen()
# 等待连接
while 1:
# 接受连接
conn, addr = sk.accept()
# 接受数据
data = conn.recv(1024)
print(data)
# 发送数据
conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
conn.send(b'ok')
# 断开连接
conn.close()服务端接收到的消息
b'GET /favicon.ico HTTP/1.1\r\nHost: 127.0.0.1:8000\r\nConnection: keep-alive\r\nUser-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/68.0.3440.106 Safari/537.36\r\nAccept: image/webp,image/apng,image/*,*/*;q=0.8\r\nReferer: http://127.0.0.1:8000/\r\nAccept-Encoding: gzip, deflate, br\r\nAccept-Language: zh-CN,zh;q=0.9\r\n\r\n'
根据不同请求路径返回不同页面
import socket
sk = socket.socket()
sk.bind(('127.0.0.1',8888))
sk.listen()
while 1:
conn,addr = sk.accept()
data = conn.recv(9000)
# 把收到的字节类型的数据转换成字符
data_str = str(data,encoding='utf8')
# 切分字符串,得到请求行
first_line = data_str.split('\r\n')[0]
# 对请求行安装空格分割
url = first_line.split(' ')[1]
print(url)
if url == '/test01/':
msg = b'shutdown'
elif url == '/test02/':
msg = b'start up'
else:
msg = b'not found'
conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
conn.send(msg)
conn.close()根据不同请求路径返回不同页面函数版
import socket
sk = socket.socket()
sk.bind(('127.0.0.1',8888))
sk.listen()
# 定义函数
def test01(url):
data = 'ni fangwen de shi: {}'.format(url)
# return bytes(data,encoding='utf8')
return data.encode('utf-8')
def test02(url):
data = 'ni fangwen de shi: {}'.format(url)
# return bytes(data,encoding='utf8')
return data.encode('utf-8')
while 1:
# 接受连接
conn, addr = sk.accept()
# 接受数据
data = conn.recv(1024)
# 获取路径
url = data.decode('utf-8').split()[1]
# 发送数据
conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
if url == '/test01/':
msg = test01(url)
elif url == '/test02/':
msg = test02(url)
else:
msg = b'404 not found'
conn.send(msg)
# 断开连接
conn.close()根据不同请求路径返回不同页面函数进阶版
import socket
sk = socket.socket()
sk.bind(('127.0.0.1',8888))
sk.listen()
# 定义函数
def test01(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
def test02(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
#url和函数的对应关系
url_func = [
('/test01/',test01),
('/test02/',test02),
]
while 1:
#建立连接,接收消息
conn,addr = sk.accept()
data = conn.recv(9000)
#对用户发来的消息做处理
# 把收到的字节类型的数据转换成字符
data_str = str(data,encoding='utf8')
# 切分字符串,得到请求行
first_line = data_str.split('\r\n')[0]
# 对请求行安装空格分割
url = first_line.split()[1]
#业务逻辑处理部分
# 使用func变量保存要执行的函数
for i in url_func:
if i[0] == url:
func = i[1]
break
else:
func = None
#执行的函数
if func:
msg = func(url)
else:
msg = b'404 not found'
#回复响应部分
conn.send(b'HTTP/1.1 200 OK\r\nContent-Type: text/html; charset=utf-8\r\n\r\n')
conn.send(msg)
conn.close()
#回应给浏览器页面的数据类型和字符集
Content-Type: text/html; charset=UTF-8返回HTML页面
import socket
sk = socket.socket()
sk.bind(('127.0.0.1',8888))
sk.listen()
# 定义函数
def test01(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
def test02(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
def login(url):
#02.html页面就是一个普通的html页面即可
with open('02.html','rb')as f:
data = f.read()
return data
#url和函数的对应关系
url_func = [
('/test01/',test01),
('/test02/',test02),
('/login/',login),
]
while 1:
conn,addr = sk.accept()
data = conn.recv(9000)
# 把收到的字节类型的数据转换成字符
data_str = str(data,encoding='utf8')
# 切分字符串,得到请求行
first_line = data_str.split('\r\n')[0]
# 对请求行安装空格分割
url = first_line.split(' ')[1]
# 使用func变量保存要执行的函数
for i in url_func:
if i[0] == url:
func = i[1]
break
else:
func = None
#执行的函数
if func:
msg = func(url)
else:
msg = b'404 not found'
conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
conn.send(msg)
conn.close()返回动态HTML页面
import socket
import time
sk = socket.socket()
sk.bind(('127.0.0.1',8888))
sk.listen()
def test01(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
def test02(url):
data = 'ni fangwen de shi: {}'.format(url)
return data.encode('utf-8')
def timer(url):
with open('time.html','r',encoding='utf8')as f:
data = f.read()
now = time.strftime('%Y-%m-%d %H:%M:%S')
data = data.replace('@@xx@@',now)
return data.encode('utf-8')
#url和函数的对应关系
url_func = [
('/test01/',test01),
('/test02/',test02),
('/timer/',timer),
]
while 1:
conn,addr = sk.accept()
data = conn.recv(9000)
# 把收到的字节类型的数据转换成字符
data_str = str(data,encoding='utf8')
# 切分字符串,得到请求行
first_line = data_str.split('\r\n')[0]
# 对请求行安装空格分割
url = first_line.split(' ')[1]
# 使用func变量保存要执行的函数
for i in url_func:
if i[0] == url:
func = i[1]
break
else:
func = None
#执行的函数
if func:
msg = func(url)
else:
msg = b'404 not found'
conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
conn.send(msg)
conn.close()
time.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<a>时间:@@xx@@</a>
</body>
</html>服务器程序和应用程序
真实开发中的python web程序来说,一般会分为两部分:服务器程序和应用程序。
服务器程序负责对socket服务端进行封装,并在请求到来时,对请求的各种数据进行整理。
应用程序则负责具体的逻辑处理。为了方便应用程序的开发,就出现了众多的Web框架,例如:Django、Flask、web.py 等。不同的框架有不同的开发方式,但是无论如何,开发出的应用程序都要和服务器程序配合,才能为用户提供服务
常用的WSGI服务器有uwsgi、Gunicorn。而Python标准库提供的独立WSGI服务器叫wsgiref,Django开发环境用的就是这个模块来做服务器。安装Django LTS版本
CMD命令行安装django
pip install Django==1.11.20
pip install Django==1.11.20 -i https:///simple/
-i 指定pip的源
== 指定django版本(不指定默认安装最新版的)
pycharm安装django

创建django项目
cmd窗口创建
django-admin startproject 项目名
pycharm创建

目录介绍
mysite/
├── manage.py # 管理文件
└── mysite # 项目目录
├── __init__.py
├── settings.py # 配置
├── urls.py # 路由 --> URL和函数的对应关系
└── wsgi.py # runserver命令就使用wsgiref模块做简单的web server命令行启动django(默认启动端口8000)
切换到项目的根目录下 manage.py
python manage.py runserver # http://127.0.0.1:8000/
python manage.py runserver 80 # http://127.0.0.1:80/
python manage.py runserver 0.0.0.0:80
pycharm启动

改变pycharm启动django端口


可以用浏览器访问

django配置文件的使用
settings.py文件
TEMPLATES = [
{
'BACKEND': 'django.template.backends.django.DjangoTemplates',
#templates文件夹位置(cmd)创建的没有,需要手动创建和修改配置文件,html文件位置
'DIRS': [os.path.join(BASE_DIR, 'templates')],
'APP_DIRS': True,
'OPTIONS': {
'context_processors': [
'django.template.context_processors.debug',
'django.template.context_processors.request',
'django.contrib.auth.context_processors.auth',
'django.contrib.messages.context_processors.messages',
],
},
},
]
*允许所有地址访问,不写默认只能本机访问
ALLOWED_HOSTS = [*]
urls.py 在最后添加
from django.shortcuts import HttpResponse, render
#方法1,HttpResponse方法
def index(request):
"""业务逻辑"""
return HttpResponse('<h1>测试页面</h1>')
#方法2,render方法(把time.html文件放到项目的templates目录中)
#def index(request):
# return render(request, 'time.html')
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^index/', index), #r'^index/',index相当于/index/,函数名
]
启动Django报错
Django 启动时报错 UnicodeEncodeError ...
报这个错误通常是因为计算机名为中文,改成英文的计算机名重启下电脑就可以了
















