从Socket到Django

socket

web服务器本质上可以认为是一段代码,可以不断的处理http协议的网络请求,而http协议可以使用socket实现,并且http协议是一个无状态的协议,即浏览器发起请求,服务器接收请求,然后给浏览器回复数据,然后断开连接;那么可以用socket来实现一个最简单的web服务器,

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import socket


def handle_request(client):
    client.recv(1024)
    client.send("HTTP/1.1 200 OK\r\n\r\n".encode())
    client.send("Hello,This is WebService".encode())
    pass


def main():
    # 创建一个socket服务端对象
    sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    # 监听8000端口
    sock.bind(('localhost', 8000))
    # 最多排队5个
    sock.listen(5)

    while True:
        connection, address = sock.accept()
        # 处理请求
        handle_request(connection)
        connection.close()


if __name__ == '__main__':
    main()

这就是利用socket实现的最简单的web服务器,绑定的端口是8000,并且当有浏览器访问该服务器的时候,会向浏览器写入http请求状态和Hello,This is WebService 这句话

image

WSGI接口

我们可以直接使用socket来写web服务器,并且了解了一个Web应用的本质是:

  1. 浏览器发送一个HTTP请求;
  2. 服务器接收到请求,生成一个HTML文档
  3. 服务器把HTML文档作为HTTP响应的Body发送给浏览器
  4. 浏览器收到HTTP响应,从HTTP Body中取出HTML文档并显示

而作为一个动态的服务器就需要完成上述的所有步骤,但是自己从Socket来实现一个个太慢了,那么如果我们把底层代码交给其他人去封装,自己写生成HTML文档的部分,这就需要一个统一的接口,来专心用Python去写Web业务,这个接口就是WSGI(Web Server Gateway Interface),而python中内置了一个WSGI服务器,这个模块是wsgiref,它是用纯Python编写的WSGI服务器参考实现,所谓参考实现指的是该实现完全符合WSGI标准,大事不考虑任何运行效率

from wsgiref.simple_server import make_server


def application(environ, start_response):
    start_response('200 OK', [('Content-Type', 'text/html')])
    return [b'<h1>Hello, web!</h1>']


if __name__ == '__main__':
    httpd = make_server('', 8000, application)
    print("Serving Http on port 8000")
    httpd.serve_forever()

其中application方法就是符合WSGI标准的一个HTTP处理函数,在当前例子中我们通过浏览器访问localhost的8000端口,则会出现Hello,web!的字样

image

进一步封装

目前我们已经搭建出了一个web框架,但是我们的web框架无论访问的url是什么都会显示Hello,web页面,正常的网站是需要根据url的不同显示不同的页面的,可以稍微更改一下,在application中获取访问的url,根据不同的url来调用不同的方法

# -*- coding: utf-8 -*-
from wsgiref.simple_server import make_server


def index():
    return [b'<h1>Index</h1>']


def login():
    return [b'login']


def application(environ, start_response):
    start_response('200 OK', [('Content-Type', 'text/html')])
    current_url = environ['PATH_INFO']
    if current_url == '/index':
        return index()
    elif current_url == '/login':
        return login()
    else:
        return [b'<h1>404</h1>']


if __name__ == '__main__':
    httpd = make_server('', 8000, application)
    print("Serving Http on port 8000")
    httpd.serve_forever()

image
image
image

可以看到,我们目前就可以实现根据不同的url来访问不同页面的目的了

封装

url

根据我们目前的封装,随着项目的扩大URL如果进一步增加的话,那么将会写很多很多的if判断,所以,我们需要把可变的部分,即 url与方法的映射 封装起来

# -*- coding: utf-8 -*-
from wsgiref.simple_server import make_server


def index():
    return [b'<h1>Index</h1>']


def login():
    return [b'login']


URLS = {
    '/index': index,
    '/login': login
}


def application(environ, start_response):
    start_response('200 OK', [('Content-Type', 'text/html')])
    current_url = environ['PATH_INFO']
    func = None
    if current_url in URLS:
        func = URLS[current_url]

    if func:
        return func()
    else:
        return [b'<h1>404</h1>']


if __name__ == '__main__':
    httpd = make_server('', 8000, application)
    print("Serving Http on port 8000")
    httpd.serve_forever()

定义了一个URLS的字典,作为路径与方法的映射,这样在application中的代码就不许要再进行改变而可以使用了

封装html

目前所有的页面都是在方法中直接返回html字节,可以把这些html统一放入到一个文件夹中
[站外图片上传中...(image-92778b-1516206345685)]
而在指定的方法中去返回这个文件中的数据

def index():
    f = open('templates/index.html','rb')
    data = f.read()
    f.close()
    return [data]

封装view

每一个页面对应着一个方法,随着我们页面的增加,方法也会变得多起来,这个时候可以统一把所有的方法放到一个view的py文件中

def index():
    f = open('templates/index.html','rb')
    data = f.read()
    f.close()
    return [data]


def login():
    return [b'login']

而在主文件中去使用view这个文件

URLS = {
    '/index': view.index,
    '/login': view.login
}

动态数据

现在的HTML是静态文件,如果我们想使用动态网页,可以在html中去认为的规定一个字符串,在读取文件的时候去替换它

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>首页</title>
</head>
<body>
这里是首页,Welcome {{USER_NAME}}
</body>
</html>
def index():
    f = open('templates/index.html', 'rb')
    data = f.read()
    data = data.replace('{{USER_NAME}}'.encode(), '蓝鸥'.encode())
    f.close()
    return [data]
image

Django

Django 是一个web框架,可以认为它封装了我们之前写的一些逻辑

创建Django项目

可以先下载Django,然后使用命令来构建Django项目,也可以直接使用PyCharm来创建一个Django项目


image

在创建的时候选择Django,Location是项目的路径,Project Interpreter这里有两个选项

  • Existing interpreter : 使用本机的Python环境,那么需要本机中下载了Django,并且通过这种方式创建的所有Django项目都是同一个版本的,使用起来比较简单
  • New environment using : 使用Python虚拟环境,在一台电脑中,很有可能在创建不同项目的时候需要使用不同版本的库,如果统一使用计算机的python则会没有办法管理这些库,python虚拟环境则会将当前电脑的python复制一份到项目路径中,并且在该项目中下载的任何库都只属于当前项目,不会影响到其他的项目,通常会选择该选项

在MoreSetting中可以填写Application name,代表创建一个应用,可以认为是一个模块

目录结构

image

创建完项目后,目录结构如图

  • home : 该文件夹为创建的app,模块内的东西都写在该文件夹下
  • templates : 放html页面
  • tesDjango : 属于项目的全局配置,url映射则写到该文件夹下的urls路径
  • venv : python虚拟环境,所有项目中用到的库都会下载到该文件夹
  • manage.py : 操作django的命令都是通过该文件,可以使用help命令来查看所有的命令

helloworld

在templates下创建一个Html页面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
HelloWorld
</body>
</html>

在home/views中创建方法

def index(request):
    return render(request, 'index.html')

配置testDjango/urls

from django.contrib import admin
from django.urls import path
from home import views
urlpatterns = [
    path('admin/', admin.site.urls),
    path(r'index',views.index)
]
image
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容