软件开发架构

  1. cs架构
  2. bs架构
  3. # 本质bs也是cs

纯手撸web框架

  1. # HTTP协议
  2. """
  3. 网络协议
  4. HTTP协议 数据传输是明文
  5. HTTPS协议 数据传输是密文
  6. websocket协议 数据传输是密文
  7. 四大特性
  8. 1.基于请求响应
  9. 2.基于TCP、IP作用于应用层之上的协议
  10. 3.无状态
  11. 4.短/无链接
  12. 数据格式
  13. 请求首行
  14. 请求头
  15. 请求体
  16. 响应状态码
  17. 1XX
  18. 2XX 200
  19. 3XX
  20. 4XX 403 404
  21. 5XX 500
  22. """
  23. # 如何做到后缀的不同返回不同的内容
  24. # 拿到用户输入的后缀 做判断
  25. #coding:utf-8
  26. import socket
  27. server = socket.socket() # TCP 三次握手四次挥手 osi七层
  28. server.bind(('127.0.0.1',8080)) # IP协议 以太网协议 arp协议...
  29. server.listen(5) # 池 ...
  30. while True:
  31. conn, addr = server.accept()
  32. data = conn.recv(1024)
  33. # print(data) # 二进制数据
  34. data = data.decode('utf-8') # 字符串
  35. # 获取字符串中特定的内容 正则 如果字符串有规律也可以考虑用切割
  36. conn.send(b'HTTP/1.1 200 OK\r\n\r\n')
  37. current_path = data.split(' ')[1]
  38. # print(current_path)
  39. if current_path == '/index':
  40. conn.send(b'index heiheihei')
  41. # with open(r'templates/01 myhtml.html', 'rb') as f:
  42. # conn.send(f.read())
  43. elif current_path == '/login':
  44. conn.send(b'login')
  45. else:
  46. # 你直接忽略favicon.ico
  47. conn.send(b'hello web')
  48. conn.close()
  49. # 不足之处
  50. 1.代码重复(服务端代码所有人都要重复写)
  51. 2.手动处理http格式的数据 并且只能拿到url后缀 其他数据获取繁琐(数据格式一样处理的代码其实也大致一样 重复写)
  52. 3.并发的问题

借助于wsgiref模块

  1. """
  2. urls.py 路由与视图函数对应关系
  3. views.py 视图函数(后端业务逻辑)
  4. templates文件夹 专门用来存储html文件
  5. """
  6. # 按照功能的不同拆分之后 后续添加功能只需要在urls.py书写对应关系然后取views.py书写业务逻辑即可
  7. from wsgiref.simple_server import make_server
  8. from urls import urls
  9. from views import *
  10. def run(env, response):
  11. """
  12. :param env:请求相关的所有数据
  13. :param response:响应相关的所有数据
  14. :return: 返回给浏览器的数据
  15. """
  16. # print(env) # 大字典 wsgiref模块帮你处理好http格式的数据 封装成了字典让你更加方便的操作
  17. # 从env中取
  18. response('200 OK', []) # 响应首行 响应头
  19. current_path = env.get('PATH_INFO')
  20. # if current_path == '/index':
  21. # return [b'index']
  22. # elif current_path == '/login':
  23. # return [b'login']
  24. # return [b'404 error']
  25. # 定义一个变量 存储匹配到的函数名
  26. func = None
  27. for url in urls: # url (),()
  28. if current_path == url[0]:
  29. # 将url对应的函数名赋值给func
  30. func = url[1]
  31. break # 匹配到一个之后 应该立刻结束for循环
  32. # 判断func是否有值
  33. if func:
  34. res = func(env)
  35. else:
  36. res = error(env)
  37. return [res.encode('utf-8')]
  38. if __name__ == '__main__':
  39. server = make_server('127.0.0.1',8080,run)
  40. """
  41. 会实时监听127.0.0.1:8080地址 只要有客户端来了
  42. 都会交给run函数处理(加括号触发run函数的运行)
  43. flask启动源码
  44. make_server('127.0.0.1',8080,obj)
  45. __call__
  46. """
  47. server.serve_forever() # 启动服务端

动静态网页

  1. """
  2. 静态网页
  3. 页面上的数据是直接写死的 万年不变
  4. 动态网页
  5. 数据是实时获取的
  6. eg:
  7. 1.后端获取当前时间展示到html页面上
  8. 2.数据是从数据库中获取的展示到html页面上
  9. """
  10. # 动态网页制作
  11. import datetime
  12. def get_time(env):
  13. current_time = datetime.datetime.now().strftime('%Y-%m-%d %X')
  14. # 如何将后端获取到的数据"传递"给html文件?
  15. with open(r'templates/03 mytime.html','r',encoding='utf-8') as f:
  16. data = f.read()
  17. # data就是一堆字符串
  18. data = data.replace('dwadasdsadsadasdas',current_time) # 在后端将html页面处理好之后再返回给前端
  19. return data
  20. # 将一个字典传递给html文件 并且可以在文件上方便快捷的操作字典数据
  21. from jinja2 import Template
  22. def get_dict(env):
  23. user_dic = {'username':'jason','age':18,'hobby':'read'}
  24. with open(r'templates/04 get_dict.html','r',encoding='utf-8') as f:
  25. data = f.read()
  26. tmp = Template(data)
  27. res = tmp.render(user=user_dic)
  28. # 给get_dict.html传递了一个值 页面上通过变量名user就能够拿到user_dict
  29. return res
  30. # 后端获取数据库中数据展示到前端页面

模版语法之Jinja2模块

pip3 install jinja2
"""模版语法是在后端起作用的"""

# 模版语法(非常贴近python语法)
{{ user }}
{{ user.get('username')}}
{{ user.age }}
{{ user['hobby'] }}


{% for user_dict in user_list %}
                        <tr>
                            <td>{{ user_dict.id}}</td>
                            <td>{{ user_dict.username}}</td>
                            <td>{{ user_dict.password}}</td>
                            <td>{{ user_dict.hobby}}</td>
                        </tr>
{% endfor%}

自定义简易版本web框架请求流程图

"""
wsgiref模块
    1.请求来的时候解析http格式的数据 封装成大字典
    2.响应走的时候给数据打包成符合http格式 再返回给浏览器

"""

python三大主流web框架

"""
django
    特点:大而全 自带的功能特别特别特别的多 类似于航空母舰
    不足之处:
        有时候过于笨重

flask
    特点:小而精  自带的功能特别特别特别的少 类似于游骑兵
    第三方的模块特别特别特别的多,如果将flask第三方的模块加起来完全可以盖过django
    并且也越来越像django
    不足之处:
        比较依赖于第三方的开发者

tornado
    特点:异步非阻塞 支持高并发
        牛逼到甚至可以开发游戏服务器
    不足之处:
        暂时你不会
"""
A:socket部分
B:路由与视图函数对应关系(路由匹配)
C:模版语法

django
  A用的是别人的        wsgiref模块
  B用的是自己的
  C用的是自己的(没有jinja2好用 但是也很方便)

flask
  A用的是别人的        werkzeug(内部还是wsgiref模块)
  B自己写的
  C用的别人的(jinja2)

tornado
    A,B,C都是自己写的

注意事项

# 如何让你的计算机能够正常的启动django项目
  1.计算机的名称不能有中文
  2.一个pycharm窗口只开一个项目
  3.项目里面所有的文件也尽量不要出现中文
  4.python解释器尽量使用3.4~3.6之间的版本
      (如果你的项目报错 你点击最后一个报错信息
    去源码中把逗号删掉)

# django版本问题
  1.X 2.X 3.X(直接忽略)
  1.X和2.X本身差距也不大 我们讲解主要以1.X为例 会讲解2.X区别
  公司之前用的1.8 满满过渡到了1.11版本 有一些项目用的2.0

# django安装
  pip3 install django==1.11.11
  如果已经安装了其他版本 无需自己卸载
  直接重新装 会自动卸载安装新的

  如果报错 看看是不是timeout 如果是 那么只是网速波动
  重新安装即可

  验证是否安装成功的方式1
  终端输入django-admin看看有没有反应

django基本操作

# 命令行操作
    # 1.创建django项目
      """
      你可以先切换到对应的D盘 然后再创建
      """
      django-admin startproject mysite

        mysite文件夹
          manage.py
          mysite文件夹
            __init__.py
            settings.py
          urls.py
          wsgi.py
 # 2.启动django项目
    """
        一定要先切换到项目目录下    
        cd /mysite
    """
  python3 manage.py runserver
  # http://127.0.0.1:8000/

# 3.创建应用
"""
Next, start your first app by running python manage.py startapp [app_label].
"""
    python manage.py startapp app01
    应用名应该做到见名知意
      user
      order
      web
      ...
      但是我们教学统一就用app01/02/03/04

    有很多文件

# pycharm操作
    # 1 new project 选择左侧第二个django即可

  # 2 启动
          1.还是用命令行启动
        2.点击绿色小箭头即可

  # 3 创建应用
          1.pycharm提供的终端直接输入完整命令
        2.pycharm 
              tools 
                run manage.py task提示(前期不要用 给我背完整命令)
 # 4 修改端口号以及创建server    
        edit confi....

应用

"""
django是一款专门用来开发app的web框架

django框架就类似于是一所大学(空壳子)
app就类似于大学里面各个学院(具体功能的app)
    比如开发淘宝
        订单相关
        用户相关
        投诉相关
        创建不同的app对应不同的功能

    选课系统
        学生功能
        老师功能

一个app就是一个独立的功能模块
"""
***********************创建的应用一定要去配置文件中注册**********************
INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'app01.apps.App01Config',  # 全写
      'app01',             # 简写
]
# 创建出来的的应用第一步先去配置文件中注册 其他的先不要给我干
ps:你在用pycharm创建项目的时候 pycharm可以帮你创建一个app并且自动注册
***********************************************************************

主要文件介绍

-mysite项目文件夹
    --mysite文件夹
      ---settings.py        配置文件
    ---urls.py            路由与视图函数对应关系(路由层)
    ---wsgi.py            wsgiref模块(不考虑)
  --manage.py            django的入口文件
  --db.sqlite3            django自带的sqlite3数据库(小型数据库 功能不是很多还有bug)
  --app01文件夹
      ---admin.py            django后台管理
    ---apps.py            注册使用
    ---migrations文件夹  数据库迁移记录
    ---models.py        数据库相关的 模型类(orm)
      ---tests.py            测试文件
    ---views.py            视图函数(视图层)

命令行与pycharm创建的区别

# 1 命令行创建不会自动有templatew文件夹 需要你自己手动创建而pycharm会自动帮你创建并且还会自动在配置文件中配置对应的路径
# pycharm创建
TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [os.path.join(BASE_DIR, 'templates')]
]
# 命令行创建
TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
]
"""
也就意味着你在用命令创建django项目的时候不单单需要创建templates文件夹还需要去配置文件中配置路径
'DIRS': [os.path.join(BASE_DIR, 'templates')]
"""

django小白必会三板斧

"""
HttpResponse
    返回字符串类型的数据

render
    返回html文件的

redirect
    重定向
      return redirect('https://www.mzitu.com/')
    return redirect('/home/')
"""