标签:可行性 integer setup width 最大的 res 字符 erro 查看
模型负责业务对象与数据库的对象(ORM),视图负责与用户的交互(页面),控制器(C)接受用户的输入调用模型和视图完成用户的请求。
Model: 代表数据存取层,Django的MTV模式本质上与MVC模式没有什么差别,也是各组件之间为了保持松耦合关系,只是定义上有些许不同,Django的MTV分别代表:
Model(模型):负责业务对象与数据库的对象(ORM)
Template(模版):负责如何把页面展示给用户
View(视图):负责业务逻辑,并在适当的时候调用Model和Template
此外,Django还有一个url分发器,它的作用是将一个个URL的页面请求分发给不同的view处理,view再调用相应的Model和Template
# 多表查询示例表 class Book(models.Model): title = models.CharField(max_length=32) price = models.DecimalField(max_digits=8, decimal_places=2) publish_date = models.DateField(auto_now_add=True) # 外键关系 # 一对多:(外键) publish = models.ForeignKey(to=‘Publish‘) # 多对多: author = models.ManyToManyField(to=‘Author‘) # 这是一个虚拟字段, 信号字段 class Author(models.Model): name = models.CharField(max_length=32) age = models.IntegerField()
很显然,Django自动创建的多对多表,支持双下划线查询语法。
# 多对多表创建实例: class MyBook(models.Model): title = models.CharField(max_length=32) price = models.DecimalField(max_digits=8, decimal_places=2) publish_date = models.DateField(auto_now_add=True) class MyAuthor(models.Model): name = models.CharField(max_length=32) age = models.IntegerField() class MyBook2MyAuthor(models.Model): # 这里的book和author不要加_id,因为会Django识别ForenginKey,并给其自动加上_id book = models.ForeignKey(to=‘MyBook‘) author = models.ForeignKey(to=‘MyAuthor‘) #纯手动创建多对多,可以在该关联表中加入自己需要的字段 info = models.CharField(max_length=32)
# models中表关系映射创建完了,记得一定要对Django数据库进行迁移(保存)操作 数据库迁移方法一: cmd命令: python3 manage.py makemigrations migrate 数据库迁移方法二: pycharm中菜单栏Tools下有个Run manage.py Task...打开输入如下命令 makemigrations migrate
数据库迁移完之后我们看看创建的三张表(Navicat查看):
先创建一些数据:
提示一下:为了直接在Django页面中运行我们的测试表查询代码,快速显示结果,我们需要对Django进行配置一下:配置方法见我另一篇博客文中单表查询处:点我跳转
# 测试表查询py文件内:
import os if __name__ == "__main__": os.environ.setdefault("DJANGO_SETTINGS_MODULE", "created_by_pycharm.settings") import django django.setup() from app01 import models models.MyAuthor.objects.create(name=‘吴承恩‘, age=100) models.MyAuthor.objects.create(name=‘吴彦祖‘, age=30) models.MyAuthor.objects.create(name=‘刘德华‘, age=45) models.MyAuthor.objects.create(name=‘刘志鹏‘, age=18) models.MyAuthor.objects.create(name=‘曹雪芹‘, age=120) res1 = models.MyBook.objects.create(title=‘西游记‘, price=66.6) res2 = models.MyBook.objects.create(title=‘东游记‘, price=99.9) res3 = models.MyBook.objects.create(title=‘南游记‘, price=88.8) res4 = models.MyBook.objects.create(title=‘北游记‘, price=68.6) # 由于多对多这张表是我们自己创建的,所以在数据库中是真实存在的,它关联着MyAutho和 # MyBook这2张表,所以添加表关系的时候在任何一个MyBook或者是MyAuthor # 表对象点mybook2myauthor_set.add()的方法是行不通的: # models.MyBook.objects.filter(pk=1).first().mybook2myauthor.add(5) # 错误提示:AttributeError: ‘MyBook‘ object has no attribute ‘mybook2myauthor‘ # 因为外键建立在多对多那张关联表中,所以要添加关系需要在这张表中动手脚 models.MyBook2MyAuthor.objects.create(book_id=1, author_id=1) models.MyBook2MyAuthor.objects.create(book_id=1, author_id=2) models.MyBook2MyAuthor.objects.create(book_id=2, author_id=2) models.MyBook2MyAuthor.objects.create(book_id=2, author_id=3) models.MyBook2MyAuthor.objects.create(book_id=3, author_id=4) models.MyBook2MyAuthor.objects.create(book_id=4, author_id=5)
下面来验证手动创建的多对多表可否使用Django的ORM表查询和基于双下划线的反向查询:
# res1 = models.MyBook.objects.filter(pk=1).first().author.name # print(res1) # 报错:‘MyBook‘ object has no attribute ‘author‘ # res2 = models.MyBook.objects.filter(pk=1).first().mybook2myauthor_set.all().info # print(res2) # res4 = models.MyBook.objects.filter(pk=1).first().author_set.all().name # print(res4) # 报错 # res5 = models.MyBook.objects.filter(pk=1).values(‘author__name‘) # print(res5) # 报错 很显然用原来的orm方法貌似不行,同时用下划线试了也不行
很显然,外键的建立位置变了,肯定这样查不行,外键全部在mybook2myauthor这张表中,所以要想从一张表查到另外一张表,必须要先经过mybook2myauthor这张表。
至于如何查询,本文不作深入探讨,咱们继续下一种方法:半自动化建立多对多表。
class Books(models.Model): title = models.CharField(max_length=32) price = models.DecimalField(max_digits=8, decimal_places=2) publish_date = models.DateField(auto_now_add=True) author = models.ManyToManyField(to=‘Authors‘, through=‘Books2Authors‘, through_fields=(‘book‘, ‘author‘)) class Authors(models.Model): name = models.CharField(max_length=32) age = models.IntegerField() class Books2Authors(models.Model): book = models.ForeignKey(to=‘Books‘) author = models.ForeignKey(to=‘Authors‘) #纯手动创建多对多,可以在该关联表中加入自己需要的字段 info = models.CharField(max_length=32)
# 创建一些数据,添加表关系 models.Authors.objects.create(name=‘吴承恩‘, age=100) models.Authors.objects.create(name=‘吴彦祖‘, age=30) models.Authors.objects.create(name=‘刘德华‘, age=45) models.Authors.objects.create(name=‘刘志鹏‘, age=18) models.Authors.objects.create(name=‘曹雪芹‘, age=120) res1 = models.Books.objects.create(title=‘西游记‘, price=66.6) res2 = models.Books.objects.create(title=‘东游记‘, price=99.9) res3 = models.Books.objects.create(title=‘南游记‘, price=88.8) res4 = models.Books.objects.create(title=‘北游记‘, price=68.6) models.Books2Authors.objects.create(book_id=1, author_id=1) models.Books2Authors.objects.create(book_id=1, author_id=2) models.Books2Authors.objects.create(book_id=2, author_id=2) models.Books2Authors.objects.create(book_id=2, author_id=3) models.Books2Authors.objects.create(book_id=3, author_id=4) models.Books2Authors.objects.create(book_id=4, author_id=5)
# ORM查询可行性
res = models.Books.objects.filter(pk=1).first().author.all().values(‘name‘) res1 = models.Authors.objects.filter(pk=1).first().books_set.all().values(‘title‘) print(res) # <QuerySet [{‘name‘: ‘吴承恩‘}, {‘name‘: ‘吴彦祖‘}]> print(res1) # <QuerySet [{‘title‘: ‘西游记‘}]>
# 双下划线查询可行性: res = models.Books.objects.filter(pk=1).values_list(‘author__name‘) res1 = models.Authors.objects.filter(pk=1).values_list(‘books__title‘) print(res) # <QuerySet [(‘吴承恩‘,), (‘吴彦祖‘,)]> print(res1) # <QuerySet [(‘西游记‘,)]>
通过上面查询结果可知这种半自动创建多对多表的方法可以使用ORM查询方法也支持双下划线方法
* JSON 使用 JavaScript 语法来描述数据对象,但是 JSON 仍然独立于语言和平台。JSON 解析器和 JSON 库支持许多不同的编程语言。
合格的json对象(json只认双引的字符串格式):
["one", "two", "three"] { "one": 1, "two": 2, "three": 3 } {"names": ["张三", "李四"] } [ { "name": "张三"}, {"name": "李四"} ]
不合格的json对象:
{ name: "张三", ‘age‘: 32 } // 属性名必须使用双引号 [32, 64, 128, 0xFFF] // 不能使用十六进制值 { "name": "张三", "age": undefined } // 不能使用undefined { "name": "张三", "birthday": new Date(‘Fri, 26 Aug 2011 07:13:10 GMT‘), "getName": function() {return this.name;} // 不能使用函数和日期对象 }
JSON.parse(‘{"name":"Howker"}‘); JSON.parse(‘{name:"Stack"}‘) ; // 错误 JSON.parse(‘[18,undefined]‘) ; // 错误
JSON.stringify(): 用于将 JavaScript 值转换为 JSON 字符串。
JSON.stringify({"name":"Tonny"})
JSON 格式于2001年由 Douglas Crockford 提出,目的就是取代繁琐笨重的 XML 格式。
JSON 格式有两个显著的优点:书写简单,一目了然;符合 JavaScript 原生语法,可以由解释引擎直接处理,不用另外添加解析代码。所以,JSON迅速被接受,已经成为各大网站交换数据的标准格式,并被写入ECMAScript 5,成为标准的一部分。
XML和JSON都使用结构化方法来标记数据,下面来做一个简单的比较。
用XML表示中国部分省市数据如下:
<?xml version="1.0" encoding="utf-8"?> <country> <name>中国</name> <province> <name>黑龙江</name> <cities> <city>哈尔滨</city> <city>大庆</city> </cities> </province> <province> <name>广东</name> <cities> <city>广州</city> <city>深圳</city> <city>珠海</city> </cities> </province> <province> <name>台湾</name> <cities> <city>台北</city> <city>高雄</city> </cities> </province> <province> <name>新疆</name> <cities> <city>乌鲁木齐</city> </cities> </province> </country>
用JSON表示如下:
{ "name": "中国", "province": [{ "name": "黑龙江", "cities": { "city": ["哈尔滨", "大庆"] } }, { "name": "广东", "cities": { "city": ["广州", "深圳", "珠海"] } }, { "name": "台湾", "cities": { "city": ["台北", "高雄"] } }, { "name": "新疆", "cities": { "city": ["乌鲁木齐"] } }] }
由上面的两端代码可以看出,JSON 简单的语法格式和清晰的层次结构明显要比 XML 容易阅读,并且在数据交换方面,由于 JSON 所使用的字符要比 XML 少得多,可以大大得节约传输数据所占用得带宽。
AJAX(Asynchronous Javascript And XML)翻译成中文就是“异步的Javascript和XML”。即使用Javascript语言与服务器进行异步交互,传输的数据为XML(当然,传输的数据不只是XML)。
AJAX 不是新的编程语言,而是一种使用现有标准的新方法。
AJAX 最大的优点是在不重新加载整个页面的情况下,可以与服务器交换数据并更新部分网页内容。(这一特点给用户的感受是在不知不觉中完成请求和响应过程)
AJAX 不需要任何浏览器插件,但需要用户允许JavaScript在浏览器上执行。
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>图书管理系统</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.min.js"></script> <link rel="stylesheet" href="/static/bootstrap-3.3.7-dist/css/bootstrap.min.css"> <script src="/static/bootstrap-3.3.7-dist/js/bootstrap.min.js"></script> <link rel="stylesheet" href="/static/layui/css/layui.css"> <script src="/static/layui/layui.js"></script> <script src="https://cdn.bootcss.com/sweetalert/2.1.2/sweetalert.min.js"></script> </head> <body> <button class="btn btn-success" id="b1">AJAX测试</button> <script> {#绑定按钮点击触发ajax提交数据#} $(‘#b1‘).on(‘click‘, function () { $.ajax({ {#固定语法格式#} url: ‘‘, {#不写默认朝当前页面发请求#} type: ‘post‘, {#请求方式#} data : {‘name‘: ‘sgt‘, ‘pwd‘: ‘123‘}, {#发送的请求数据#} {#这里的data就是回调函数success获取到的后端响应返回的数据#} success:function (data) { swal({ title: ‘AJAX‘, text: ‘模拟这里显示了返回的数据‘, icon: ‘success‘, button: ‘OK‘, }) } }) }) </script> </body> </html>
我们知道ajax默认的传输数据格式是urlencoded,前面我们说过,ajax可以用json作为数据格式传输
现在强调一点:前后端传输数据必须要求数据是什么格式就应该用对应的传输格式,一一对应去传输,否则Django解析数据出问题,以至于我们在前后端拿不到想要的数据。
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>图书管理系统</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.min.js"></script> <link rel="stylesheet" href="/static/bootstrap-3.3.7-dist/css/bootstrap.min.css"> <script src="/static/bootstrap-3.3.7-dist/js/bootstrap.min.js"></script> <link rel="stylesheet" href="/static/layui/css/layui.css"> <script src="/static/layui/layui.js"></script> <script src="https://cdn.bootcss.com/sweetalert/2.1.2/sweetalert.min.js"></script> </head> <body> <input type="file" id="f1"> <br> <button id="s1" class="btn btn-default">提交</button> <script> //需要知道,ajax无法自己传文件,需要借助于内置对象FormData $(‘#s1‘).on(‘click‘, function () { //ajax借助FormData内置对象传文件 let formdata = new FormData(); //FormData不仅能传文件,也能传普通键值 formdata.append(‘name‘, ‘sgt‘); <!--添加普通键值--> //获取input框存放文件 file_data = $(‘#f1‘)[0].files[0]; //$(‘#f1‘)[0]是将jQuery对象转换成js对象,然后通过js对象点files来获取到文件对象,然后去索引0才能拿到文件对象 formdata.append(‘myfile‘, file_data); //使用ajax发送文件 $.ajax({ url: ‘‘, type: ‘post‘, data: formdata, <!--传输数据为formdata--> processData: false, <!--告诉浏览器不要处理数据--> contentType: false, <!--不要用任何编码,就用我formdata自带的格式,因为Django能自动识别formdata--> //回调函数获取提交后后端返回的数据 success:function (data) { swal({ <!--用SweetAlert显示结果--> title: ‘传输文件‘, text: data, icon: ‘success‘, button: ‘OK‘, }) } }) }) </script> </body> </html>
后端
def ttt(request): if request.method == ‘POST‘: print(request.POST) # 拿到ajax传来的formdata普通键值数据 print(request.FILES) # 拿到ajax传来的formdata文件 return HttpResponse(‘我从后端来,文件[%s]已收到‘ % request.FILES.get(‘myfile‘)) return render(request, ‘ttt.html‘)
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>图书管理系统</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.min.js"></script> <link rel="stylesheet" href="/static/bootstrap-3.3.7-dist/css/bootstrap.min.css"> <script src="/static/bootstrap-3.3.7-dist/js/bootstrap.min.js"></script> <link rel="stylesheet" href="/static/layui/css/layui.css"> <script src="/static/layui/layui.js"></script> <script src="https://cdn.bootcss.com/sweetalert/2.1.2/sweetalert.min.js"></script> </head> <body> <div class="container"> <div class="row"> <div class="col-sm-4 col-sm-offset-4"> 用户名: <input type="text" class="user form-control"> <span class="hide" style="color: red">用户已存在</span> </div> </div> </div> <p></p> <script> //input事件来实时监控input框输入的数据,如果用change,则在input框失去焦点时候触发 $(‘.user‘).on(‘input‘,function () { $.ajax({ url: ‘‘, type: ‘post‘, data: {‘username‘: $(this).val()}, success:function (data) { {#res = JSON.parse(data);#} if (data[‘flag‘]){ $(‘div>span‘).removeClass(‘hide‘)} else { $(‘div>span‘).addClass(‘hide‘) } } }) }) </script>
</body> </html>
后端:
def ttt(request): if request.method == ‘POST‘: res = {‘flag‘: False} # 获取到ajax发送过来的username username = request.POST.get(‘username‘) # 通过username进行数据库查询 user_obj = models.User.objects.filter(name=username).first() # 判断用户名是否存在与数据库中 if user_obj: res[‘flag‘] = True return JsonResponse(res) return render(request, ‘ttt.html‘)
结果演示:
import random user_list = [‘用户[{}]‘.format(i) for i in range(100)] data = [] for j in user_list: data.append(models.MyUser(name=j, password=‘123‘, gender=str(random.choice([1, 2, 3])))) models.MyUser.objects.bulk_create(data)
class Pagination(object): def __init__(self,current_page,all_count,per_page_num=2,pager_count=11): """ 封装分页相关数据 :param current_page: 当前页 :param all_count: 数据库中的数据总条数 :param per_page_num: 每页显示的数据条数 :param pager_count: 最多显示的页码个数 用法: queryset = model.objects.all() page_obj = Pagination(current_page,all_count) page_data = queryset[page_obj.start:page_obj.end] 获取数据用page_data而不再使用原始的queryset 获取前端分页样式用page_obj.page_html """ try: current_page = int(current_page) except Exception as e: current_page = 1 if current_page <1: current_page = 1 self.current_page = current_page self.all_count = all_count self.per_page_num = per_page_num # 总页码 all_pager, tmp = divmod(all_count, per_page_num) if tmp: all_pager += 1 self.all_pager = all_pager self.pager_count = pager_count self.pager_count_half = int((pager_count - 1) / 2) @property def start(self): return (self.current_page - 1) * self.per_page_num @property def end(self): return self.current_page * self.per_page_num def page_html(self): # 如果总页码 < 11个: if self.all_pager <= self.pager_count: pager_start = 1 pager_end = self.all_pager + 1 # 总页码 > 11 else: # 当前页如果<=页面上最多显示11/2个页码 if self.current_page <= self.pager_count_half: pager_start = 1 pager_end = self.pager_count + 1 # 当前页大于5 else: # 页码翻到最后 if (self.current_page + self.pager_count_half) > self.all_pager: pager_end = self.all_pager + 1 pager_start = self.all_pager - self.pager_count + 1 else: pager_start = self.current_page - self.pager_count_half pager_end = self.current_page + self.pager_count_half + 1 page_html_list = [] # 添加前面的nav和ul标签 page_html_list.append(‘‘‘ <nav aria-label=‘Page navigation>‘ <ul class=‘pagination‘> ‘‘‘) first_page = ‘<li><a href="?page=%s">首页</a></li>‘ % (1) page_html_list.append(first_page) if self.current_page <= 1: prev_page = ‘<li class="disabled"><a href="#">上一页</a></li>‘ else: prev_page = ‘<li><a href="?page=%s">上一页</a></li>‘ % (self.current_page - 1,) page_html_list.append(prev_page) for i in range(pager_start, pager_end): if i == self.current_page: temp = ‘<li class="active"><a href="?page=%s">%s</a></li>‘ % (i, i,) else: temp = ‘<li><a href="?page=%s">%s</a></li>‘ % (i, i,) page_html_list.append(temp) if self.current_page >= self.all_pager: next_page = ‘<li class="disabled"><a href="#">下一页</a></li>‘ else: next_page = ‘<li><a href="?page=%s">下一页</a></li>‘ % (self.current_page + 1,) page_html_list.append(next_page) last_page = ‘<li><a href="?page=%s">尾页</a></li>‘ % (self.all_pager,) page_html_list.append(last_page) # 尾部添加标签 page_html_list.append(‘‘‘ </nav> </ul> ‘‘‘) return ‘‘.join(page_html_list)
l = [models.User(name=‘这是第%s记录‘ % i) for i in range(1000)] models.User.objects.bulk_create(l)
然后操作分页器的实现:
class Pagination(object): def __init__(self, current_page, all_count, per_page_num=20, pager_count=11): """ 封装分页相关数据 :param current_page: 当前页 :param all_count: 数据库中的数据总条数 :param per_page_num: 每页显示的数据条数 :param pager_count: 最多显示的页码个数 用法: queryset = model.objects.all() page_obj = Pagination(current_page,all_count) page_data = queryset[page_obj.start:page_obj.end] 获取数据用page_data而不再使用原始的queryset 获取前端分页样式用page_obj.page_html """ try: current_page = int(current_page) except Exception as e: current_page = 1 if current_page < 1: current_page = 1 self.current_page = current_page self.all_count = all_count self.per_page_num = per_page_num # 总页码 all_pager, tmp = divmod(all_count, per_page_num) if tmp: all_pager += 1 self.all_pager = all_pager self.pager_count = pager_count self.pager_count_half = int((pager_count - 1) / 2) @property def start(self): return (self.current_page - 1) * self.per_page_num @property def end(self): return self.current_page * self.per_page_num def page_html(self): # 如果总页码 < 11个: if self.all_pager <= self.pager_count: pager_start = 1 pager_end = self.all_pager + 1 # 总页码 > 11 else: # 当前页如果<=页面上最多显示11/2个页码 if self.current_page <= self.pager_count_half: pager_start = 1 pager_end = self.pager_count + 1 # 当前页大于5 else: # 页码翻到最后 if (self.current_page + self.pager_count_half) > self.all_pager: pager_end = self.all_pager + 1 pager_start = self.all_pager - self.pager_count + 1 else: pager_start = self.current_page - self.pager_count_half pager_end = self.current_page + self.pager_count_half + 1 page_html_list = [] # 添加前面的nav和ul标签 page_html_list.append(‘‘‘ <nav aria-label=‘Page navigation>‘ <ul class=‘pagination‘> ‘‘‘) first_page = ‘<li><a href="?page=%s">首页</a></li>‘ % (1) page_html_list.append(first_page) if self.current_page <= 1: prev_page = ‘<li class="disabled"><a href="#">上一页</a></li>‘ else: prev_page = ‘<li><a href="?page=%s">上一页</a></li>‘ % (self.current_page - 1,) page_html_list.append(prev_page) for i in range(pager_start, pager_end): if i == self.current_page: temp = ‘<li class="active"><a href="?page=%s">%s</a></li>‘ % (i, i,) else: temp = ‘<li><a href="?page=%s">%s</a></li>‘ % (i, i,) page_html_list.append(temp) if self.current_page >= self.all_pager: next_page = ‘<li class="disabled"><a href="#">下一页</a></li>‘ else: next_page = ‘<li><a href="?page=%s">下一页</a></li>‘ % (self.current_page + 1,) page_html_list.append(next_page) last_page = ‘<li><a href="?page=%s">尾页</a></li>‘ % (self.all_pager,) page_html_list.append(last_page) # 尾部添加标签 page_html_list.append(‘‘‘ </nav> </ul> ‘‘‘) return ‘‘.join(page_html_list)
③:视图文件views.py中:
def ttt(request): user_list = models.User.objects.all() all_count = user_list.count() current_page = request.GET.get(‘page‘, 1) page_obj = my_page.Pagination(current_page=current_page, all_count=all_count) page_queryset = user_list[page_obj.start:page_obj.end] return render(request, ‘ttt.html‘, locals())
④:html文件中:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>图书管理系统</title> <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.min.js"></script> <link rel="stylesheet" href="/static/bootstrap-3.3.7-dist/css/bootstrap.min.css"> <script src="/static/bootstrap-3.3.7-dist/js/bootstrap.min.js"></script> <link rel="stylesheet" href="/static/layui/css/layui.css"> <script src="/static/layui/layui.js"></script> <script src="https://cdn.bootcss.com/sweetalert/2.1.2/sweetalert.min.js"></script> </head> <body> <div class="container"> <div class="row"> <div class="col-md-8 col-md-offset-2"> <table class="text-center table-bordered table table-hover table-striped"> <thead> <tr> <th>ID</th> <th>name</th> <th>password</th> </tr> </thead> <tbody> {% for user in page_queryset %} <tr> <th>{{ user.pk }}</th> <th>{{ user.name }}</th> <th>{{ user.password }}</th> </tr> {% endfor %} </tbody> </table> {{ page_obj.page_html|safe }} </div> </div> </div> </body> </html>
演示结果:
2019年6月14日 Web框架之Django_07 进阶操作(MTV与MVC、多对多表三种创建方式、前后端传输数据编码格式contentType、ajax、自定义分页器)
标签:可行性 integer setup width 最大的 res 字符 erro 查看
原文地址:https://www.cnblogs.com/suguangti/p/11023360.html