今日概要:
请求周期
路由系统
视图
类和函数(FBV和CBV)
参数 request
响应
HttpResponse/JsonResponse/render/redirect
return HttpResponse("...")
响应头
obj = HttpResponse("...")
obj['xxxxx'] = "值"
return obj
其他知识
虚拟环境
纯净版项目,内置app功能去掉。
多app,嵌套到apps目录。
pycharm创建django项目 + 虚拟环境
settings配置
django默认settings [先加载] 500
项目目录settings [后加载] 20
静态资源
TEMPLATES = [
{
'BACKEND': 'django.template.backends.django.DjangoTemplates',
'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',
],
},
},
]
优先去项目根目录 > 每个已注册的app的templates目录找。
如何选择:
扩展:修改内置app的模板也是同样的套路。
渲染完成后,生成了字符串,再返回给浏览器。
打开文件读取 里面的内容 返回浏览器
渲染完成 ,得到一个渲染完成的字符串
把渲染完成的字符串返回给浏览器
在django模板语法中提供了内置函数让我们来。
三种方式:
filter
- 数据处理,参数:1~2个
- 数据处理,if条件
simple_tag
参数无限制 & 返回文本
inclusion_tag
参数无限制 & HTML片段
需求来了:根据用户权限不同显示不同的菜单。
源码:
面向对象
class MyMd(object):
def __init__(self....):
pass
def __call__(self,....):
pass
django内部默认执行call方法,传入参数。
反射
class MyMd(object):
def __init__(self....):
pass
def __call__(self,....):
if hasattr(self,'process_request'):
response = self.process_request(request)
...
django内部默认执行call方法,传入参数。
class MiddlewareMixin:
def __init__(self, get_response=None):
self.get_response = get_response
def __call__(self, request):
response = None
if hasattr(self, 'process_request'):
response = self.process_request(request)
response = response or self.get_response(request)
if hasattr(self, 'process_response'):
response = self.process_response(request, response)
return response
class MyMd(MiddlewareMixin):
def process_request(self,request):
...
def process_response(self,request, response):
...
django内部默认执行call方法,传入参数。
疑问:prcess_request的执行时,是否已执行了路由匹配?
request.resolver_match
注意:process_view是在django中源码中写死了。
TemplateResponse
对象 or 对象中含有.render方法。orm,关系对象映射,本质翻译的。
实现:创建表、修改表、删除表。
在app中的models.py中按照规则编写类 ===> 表结构。
编写类
from django.db import models
class UserInfo(models.Model):
name = models.CharField(max_length=16)
age = models.IntegerField()
###定义完了表 一般表名字为类名小写再加上app得名字
注册app(在setting.py中注册app INSTALL_APPS)
INSTALLED_APPS = [
# 'django.contrib.admin',
# 'django.contrib.auth',
# 'django.contrib.contenttypes',
# 'django.contrib.sessions',
# 'django.contrib.messages',
'django.contrib.staticfiles',
'apps.app01.apps.App01Config',
'apps.app02.apps.App02Config',
]
命令,django根据models中类生成一个 对数据库操作的配置文件
=> migrations
python manage.py makemigrations##生成配置文件
命令,读取已经注册么给app中的migrations目录将配置文件 -> 转换成:生成表,修改表 SQL -> 连接数据库去运行。
python manage.py migrate### 把所有的app下面的migration都运行
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.sqlite3',
'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
}
}
常见问题:请不要再手动去修改数据的表结构 + 时刻保证 ORM和数据表是对应。
字段
CharField字符串 max_leght表示字符串长度
SmallIntegerField
IntegerField
BigIntegerField
DateField 年月日
DateTimeField 年月日时分表
BooleanField -> 其实数据库不支持真假,根据SmallIntegerField创造出来出来。 0 1
DecimalField -> 精确的小数
参数
name = models.CharField(verbose_name="姓名", max_length=16)
name = models.CharField(verbose_name="姓名", max_length=16, default="哈哈哈")
# 经常查询,速度快(MySQL,https://www.bilibili.com/video/BV15R4y1b7y9)
name = models.CharField(verbose_name="姓名", max_length=16, default="哈哈哈", null=True, blank=True, db_index=True)
email = models.CharField(verbose_name="姓名", max_length=16, default="哈哈哈", null=True, blank=True, unique=True)
# 在数据库存储时只能是:sh、bj (上海、北京一般用于页面显示中文)
code = models.CharField(verbose_name="姓名", max_length=16, choices=(("sh", "上海"), ("bj", "北京")),default="sh")
# 不用 max_length=16
count = models.IntegerField(verbose_name="数量", default=1, null=True, blank=True, unique=True)
code = models.IntegerField(verbose_name="性别",choices=((1, "男"), (2, "女")),default=1)
##数据库存储,1,2代表男女
register_date = models.DateField(verbose_name="注册时间", auto_now=True)###不用写 用户时间自动生成auto_now = True
amount = models.DecimalField(verbose_name="余额", max_digits=10, decimal_places=2)
示例:
from django.db import models
class UserInfo(models.Model):
name = models.CharField(verbose_name="姓名", max_length=16, db_index=True)
age = models.PositiveIntegerField(verbose_name="年龄")
email = models.CharField(verbose_name="邮箱", max_length=128, unique=True)
amount = models.DecimalField(verbose_name="余额", max_digits=10, decimal_places=2, default=0)
register_date = models.DateField(verbose_name="注册时间", auto_now=True)
class Goods(models.Model):
title = models.CharField(verbose_name="标题", max_length=32)
# detail = models.CharField(verbose_name="详细信息", max_length=255)
detail = models.TextField(verbose_name="详细信息")
price = models.PositiveIntegerField(verbose_name="价格")
count = models.PositiveBigIntegerField(verbose_name="库存", default=0)
###verbose 表示这个得中文含义
注意:让程序生成第三张表(多对多关系)ManyToManyField生成的表字段只能id/bid/gid
设计自己项目的业务时,理清楚表与表之间的关系。
强调:设计项目表结构:表名和字段都不要拼音。
实现:增删改查。
提交形式:zip包 -> markdown编写。
下节预告:
orm数据操作、cookie和session、缓存、刷票平台(表结构设计)、用户管理、用户+权限菜单