SweetAlert前端插件
Django自带的序列化组件
serializers序列化组件可以把我们用ORM产生的QuerySet对象转换成json格式数据。
from django.core import serializers
def index(request):
book_queryset = models.Book.objects.all()
res = serializers.serialize('json', book_queryset)
return HttpResponse(res)
更多教程请访问http://www.manongzj.com
批量数据操作
如果我们想要使用ORM去循环插入10万条数据,每次添加数据都执行一次create(),这样会频繁走数据库操作,效率极低,比如:
for i in range(100000):
models.Book.objects.create(title=f'第{i}本书')
这样操作需要等待很久,所以我们可以换一个方法:先用类创建多个对象,在用bulk_create(),这样只要走一次数据库操作就可以添加多个数据了:
obj_list = [] # 存放对象
for i in range(100000):
obj = models.Book(title=f'第{i}本书') # 实例化多个数据对象
obj_list.append(obj) # 对象追加到列表种
models.Book.objects.bulk_create(obj_list) # 一次性全部添加
分页器与推导流程
网站不可能将所有的数据全部展示到一页,应该考虑使用分页,每页只展示部分数据。那么分页该如何实现呢?
推导流程
1.首先需要知道ORM中的all()方法返回的结果集是支持正数的索引切片的。
# 取第一个到第10个的结果
book_queryset = models.Books.objects.all()[0:10]
2.在用户点击分页的页数时肯定是要向后端请求数据的,比如第5页就给前端返回第41到第50的结果(每页展示10条数据的情况),所以后端需要用一个变量接收前端传来的页数。
前端:发送第五页的请求,可以用a标签发送GET请求,并携带数据page=5。
<a href='?page=5'>5</a>
后端:用变量接收
current_page = request.GET.get('page')
3.既然需要分页,那么每页肯定都有最多的展示条数,这里我们设置每页10条,返回指定页数的数据。
def index(request):
current_page = request.GET.get('page')
try: # 异常处理,防止current_page值为空时报错
current_page = int(current_page)
except:
current_page = 1
start = (current_page - 1) * 10 # 数据起始位置
end = current_page * 10 # 数据结束位置
book_queryset = models.Books.objects.all()[start:end]
return render(request, 'index.html', {'book_queryset': book_queryset})
4.前端接收后端数据:
<div class="text-center">
{% for book_obj in book_queryset %}
<p>{{ book_obj.name }}</p>
{% endfor %}
</div>
5.这时候我们只需要在浏览器地址后面输入?page=10,就可以获取第10页的数据。
6.添加Bootstrap提供的分页器
<div class="text-center">
{% for book_obj in book_queryset %}
<p>{{ book_obj.name }}</p>
{% endfor %}
<nav aria-label="Page navigation">
<ul class="pagination">
<li>
<a href="#" aria-label="Previous">
<span aria-hidden="true">«</span>
</a>
</li>
<li><a href="#">1</a></li>
<li><a href="#">2</a></li>
<li><a href="#">3</a></li>
<li><a href="#">4</a></li>
<li><a href="#">5</a></li>
<li>
<a href="#" aria-label="Next">
<span aria-hidden="true">»</span>
</a>
</li>
</ul>
</nav>
</div>
7.由于前端不好写动态的分页器,所以我们用后端编写html标签,编写页数时还需要用到divmod()获取所有数据需要的页数,比如99条数据要10页,100条数据要10页,101条数据要11页。
后端:
def index(request):
# 获取当前页数
current_page = request.GET.get('page')
try: # 异常处理,防止current_page值为空时报错
current_page = int(current_page)
except:
current_page = 1
data_queryset = models.Books.objects.all()
start = (current_page - 1) * 10 # 数据起始位置
end = current_page * 10 # 数据结束位置
book_queryset = data_queryset[start:end]
data_count = data_queryset.count()
# 接收整数和余数
page_count, m = divmod(data_count, 10)
# 余数不为0,则要把整数部分加一
if m != 0:
page_count += 1
html = []
# 让当前页数左边显示5个页码,右边显示五个页码
for i in range(current_page - 5, current_page + 5):
if i == current_page: # 当前页数高亮显示
html.append(f"<li class='active'><a href='?page={i}'>{i}</a></li>")
else: # 当前页数普通显示
html.append(f"<li><a href='?page={i}'>{i}</a></li>")
return render(request, 'index.html', {'book_queryset': book_queryset, 'html': html})
前端:
<div class="text-center">
{% for book_obj in book_queryset %}
<p>{{ book_obj.name }}</p>
{% endfor %}
<nav aria-label="Page navigation">
<ul class="pagination">
<li>
<a href="#" aria-label="Previous">
<span aria-hidden="true">«</span>
</a>
</li>
{% for h in html %}
{{ h|safe }}
{% endfor %}
<li>
<a href="#" aria-label="Next">
<span aria-hidden="true">»</span>
</a>
</li>
</ul>
</nav>
</div>
8.现在可以点击分页器到指定页面了,但是出现了新问题,当前页码小于6时,分页器有零或负数,当前页码过大时,分页器会超出。
这个问题加个变量就可以了。
temp_page = current_page
# 页数过小
if current_page < 6:
temp_page = 6
# 页数过大
if current_page > page_count - 4:
temp_page = page_count - 4
for i in range(temp_page - 5, temp_page + 5):
if i == current_page: # 当前页数高亮显示
html.append(f"<li class='active'><a href='?page={i}'>{i}</a></li>")
else: # 当前页数普通显示
html.append(f"<li><a href='?page={i}'>{i}</a></li>")
究极大法
上面是自定义分页器开发流程的基本思路,我们不需要掌握代码的编写,只需要掌握基本用法即可,原文博客:自定义分页器 - JasonJi - 博客园 (cnblogs.com)
自定义分页器封装代码
点击查看代码
前端使用:
<div class="container">
<div class="row">
<div class="col-md-8 col-md-offset-2">
{% for book in page_queryset %}
<p>{{ book.title }}</p>
{% endfor %}
{{ page_obj.page_html|safe }}
</div>
</div>
</div>
后端使用:
def get_book(request):
book_list = models.Book.objects.all()
current_page = request.GET.get("page", 1)
all_count = book_list.count()
page_obj = Pagination(current_page=current_page, all_count=all_count, per_page_num=10)
page_queryset = book_list[page_obj.start:page_obj.end]
return render(request, 'booklist.html', locals())
Forms组件之创建
Forms组件功能:数据校验、标签渲染、展示信息。
- 数据校验:数据是否符合规范(长度、格式等)
- 标签渲染:快速生成输入标签等
- 信息展示:展示错误的提示信息,并保留原输入内容
基本使用
from django import forms
# 创建表单类
class MyForm(forms.Form):
# 用户名至少三个字符最多八个字符
username = forms.CharField(min_length=3, max_length=8)
# 年龄最小不能小于0 最大不能超过150
age = forms.IntegerField(min_value=0, max_value=150)
# 邮箱必须符合邮箱格式(@关键符号)
email = forms.EmailField()
Forms组件之数据校验
创建好表单类之后,在视图函数中使用:
将数据传入并实例化对象,需要字典类型,字典的键名称与表单类中自定义的名称一致:
form_obj = MyForm({
'username': 'abc',
'age': 999,
'email': '12qq'
})
查看数据是否合法(全部合法结果才是True):
form_obj.isvalid()
查看不符合条件的数据及原因:
form_obj.errors
查看符合条件的数据:
form_obj.cleaned_data
补充
1.forms类中所有的字段数据默认都是必填的,不能少,如果想忽略某些字段,可以添加 required=False。
email = forms.EmailField(required=False)
2.forms类中额外传入的字段数据不会做任何的校验,直接忽略。
Forms组件之渲染标签
后端返回给前端form对象,前端可以使用这个对象创建标签:
def index(request):
form_obj = MyForm()
return render(request, 'index.html', locals())
创建方式一:封装程度高,扩展性较差,主要用于快速生成页面测试功能
1.每一个输入框占一行
<form action="" method="post">
{{ form_obj.as_p }}
<input type="submit">
</form>
2.所有输入框占一行
<form action="" method="post">
{{ form_obj.as_table }}
<input type="submit">
</form>
3.输入框以无序列表形式展示
<form action="" method="post">
{{ form_obj.as_ul }}
<input type="submit">
</form>
创建方式二:封装程度低,扩展性较好,但是字段比较多的情况下不方便。
form对象.字段名.label:文本提示
form对象.字段名:输入标签
<form action="" method="post">
{{ form_obj.username.label }}
{{ form_obj.username }}
{{ form_obj.age.label }}
{{ form_obj.age }}
{{ form_obj.email.label }}
{{ form_obj.email }}
<input type="submit">
</form>
创建方式三:创建方式二使用for循环创建
<form action="" method="post">
{% for form in form_obj %}
<p>
{{ form.label }}
{{ form }}
</p>
{% endfor %}
<input type="submit">
</form>
补充
1.forms组件只负责渲染获取用户数据的标签,form表单标签和提交按钮需要自己写。
2.渲染标签中文提示,可以在创建Form类中,创建字段时用参数 label指定,不指定默认使用字段名(首字母大写)。
username = forms.CharField(min_length=3, max_length=8, label='用户名')
Forms组件之信息展示
在你点击提交表单信息后,它会提醒你错误信息:
如果不想要这种提示方式,form表单可以取消浏览器自动添加校验功能的操作:添加属性novalidate。
<form action="" method="post" novalidate>
</form>
这时候前端的校验功能没了,我们可以在后端进行校验:
def index(request):
form_obj = MyForm()
if request.method == 'POST':
# request.POST可以看成字典类型
form_obj = MyForm(request.POST)
# 校验数据
if form_obj.is_valid():
return HttpResponse('数据正常!')
return render(request, 'index.html', locals())
前端使用form.errors.0获取错误信息
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}
{{ form }}
<span style="color: red">{{ form.errors.0 }}</span>
</p>
{% endfor %}
<input type="submit">
</form>
错误信息是可以自定义的,在Form类中创建字段时定义:
# 用户名至少三个字符最多八个字符
username = forms.CharField(min_length=3, max_length=8, label='用户名',
error_messages={
'min_length': '用户名最短3位',
'max_length': '用户名最长8位',
'required': '用户名必填'
})