|
6 | 6 |
|
7 | 7 | 1. **提升开发效率**。前后端分离以后,可以实现前后端代码的解耦,只要前后端沟通约定好应用所需接口以及接口参数,便可以开始并行开发,无需等待对方的开发工作结束。在这种情况下,前后端工程师都可以只专注于自己的开发工作,有助于打造出更好的团队。除此之外,在前后端分离的开发模式下,即使需求发生变更,只要接口与数据格式不变,后端开发人员就不需要修改代码,只要前端进行变动即可。
|
8 | 8 | 2. **增强代码的可维护性**。前后端分离后,应用的代码不再是前后端混合,只有在运行期才会有调用依赖关系,这样的话维护代码的工作将变得轻松愉快很多,再不会牵一发而动全身。当你的代码变得简明且整洁时,代码的可读性和可维护性都会有质的提升。
|
9 |
| -3. **支持多终端和服务化架构**。前后端分离后,同一套数据接口可以为不同的终端提供服务,更有助于打造多终端应用;此外,由于后端提供的接口之间可以通过HTTP进行调用,有助于打造服务化架构(包括微服务)。 |
| 9 | +3. **支持多终端和服务化架构**。前后端分离后,同一套数据接口可以为不同的终端提供服务,更有助于打造多终端应用;此外,由于后端提供的接口之间可以通过HTTP(S)进行调用,有助于打造服务化架构(包括微服务)。 |
| 10 | + |
| 11 | +接下来我们就用前后端分离的方式来改写之前的投票应用。 |
| 12 | + |
| 13 | +### 返回JSON格式的数据 |
| 14 | + |
| 15 | +刚才说过,在前后端分离的开发模式下,后端需要为前端提供数据接口,这些接口通常返回JSON格式的数据。在Django项目中,我们可以先将对象处理成字典,然后就可以利用Django封装的`JsonResponse`向浏览器返回JSON格式的数据,具体的做法如下所示。 |
| 16 | + |
| 17 | +```Python |
| 18 | +def show_subjects(request): |
| 19 | + queryset = Subject.objects.all() |
| 20 | + subjects = [] |
| 21 | + for subject in queryset: |
| 22 | + subjects.append({ |
| 23 | + 'no': subject.no, |
| 24 | + 'name': subject.name, |
| 25 | + 'intro': subject.intro, |
| 26 | + 'isHot': subject.is_hot |
| 27 | + }) |
| 28 | + return JsonResponse(subjects, safe=False) |
| 29 | +``` |
| 30 | + |
| 31 | +上面的代码中,我们通过循环遍历查询学科得到的`QuerySet`对象,将每个学科的数据处理成一个字典,在将字典保存在名为`subjects`的列表容器中,最后利用`JsonResponse`完成对列表的序列化,向浏览器返回JSON格式的数据。由于`JsonResponse`序列化的是一个列表而不是字典,所以需要指定`safe`参数的值为`False`才能完成对`subjects`的序列化,否则会产生`TypeError`异常。 |
| 32 | + |
| 33 | +可能大家已经发现了,自己写代码将一个对象转成字典是比较麻烦的,如果对象的属性很多而且某些属性又关联到一个比较复杂的对象时,情况会变得更加糟糕。为此我们可以使用一个名为bpmappers的三方库来简化将对象转成字典的操作,这个三方库本身也提供了对Django框架的支持。 |
| 34 | + |
| 35 | +安装三方库bpmappers。 |
| 36 | + |
| 37 | +```Shell |
| 38 | +pip install bpmappers |
| 39 | +``` |
| 40 | + |
| 41 | +编写映射器(实现对象到字典转换)。 |
| 42 | + |
| 43 | +```Python |
| 44 | +from bpmappers.djangomodel import ModelMapper |
| 45 | + |
| 46 | +from poll2.models import Subject |
| 47 | + |
| 48 | + |
| 49 | +class SubjectMapper(ModelMapper): |
| 50 | + |
| 51 | + class Meta: |
| 52 | + model = Subject |
| 53 | +``` |
| 54 | + |
| 55 | +修改视图函数。 |
| 56 | + |
| 57 | +```Python |
| 58 | +def show_subjects(request): |
| 59 | + queryset = Subject.objects.all() |
| 60 | + subjects = [] |
| 61 | + for subject in queryset: |
| 62 | + subjects.append(SubjectMapper(subject).as_dict()) |
| 63 | + return JsonResponse(subjects, safe=False) |
| 64 | +``` |
| 65 | + |
| 66 | +配置URL映射,然后访问该接口,可以得到如下所示的JSON格式数据。 |
| 67 | + |
| 68 | +```JSON |
| 69 | +[ |
| 70 | + { |
| 71 | + "no": 101, |
| 72 | + "name": "Python全栈+人工智能", |
| 73 | + "intro": "Python是一种计算机程序设计语言。是一种面向对象的动态类型语言,最初被设计用于编写自动化脚本(shell),随着版本的不断更新和语言新功能的添加,越来越多被用于独立的、大型项目的开发。", |
| 74 | + "create_date": "2017-08-01", |
| 75 | + "is_hot": true |
| 76 | + }, |
| 77 | + // 此处省略下面的内容 |
| 78 | +] |
| 79 | +``` |
| 80 | + |
| 81 | +如果不希望在JSON数据中显示学科的成立时间,我们可以在映射器中排除`create_date`属性;如果希望将是否为热门学科对应的键取名为`isHot`(默认的名字是`is_hot`),也可以通过修改映射器来做到。具体的做法如下所示: |
| 82 | + |
| 83 | +```Python |
| 84 | +from bpmappers import RawField |
| 85 | +from bpmappers.djangomodel import ModelMapper |
| 86 | + |
| 87 | +from poll2.models import Subject |
| 88 | + |
| 89 | + |
| 90 | +class SubjectMapper(ModelMapper): |
| 91 | + isHot = RawField('is_hot') |
| 92 | + |
| 93 | + class Meta: |
| 94 | + model = Subject |
| 95 | + exclude = ('create_date', 'is_hot') |
| 96 | +``` |
| 97 | + |
| 98 | +再次查看学科接口返回的JSON数据。 |
| 99 | + |
| 100 | +```JSON |
| 101 | +[ |
| 102 | + { |
| 103 | + "no": 101, |
| 104 | + "name": "Python全栈+人工智能", |
| 105 | + "intro": "Python是一种计算机程序设计语言。是一种面向对象的动态类型语言,最初被设计用于编写自动化脚本(shell),随着版本的不断更新和语言新功能的添加,越来越多被用于独立的、大型项目的开发。", |
| 106 | + "isHot": true |
| 107 | + }, |
| 108 | + // 此处省略下面的内容 |
| 109 | +] |
| 110 | +``` |
| 111 | + |
| 112 | +关于bpmappers详细的使用指南,请参考它的[官方文档](<https://bpmappers.readthedocs.io/en/stable/>),这个官方文档是用日语书写的,可以使用浏览器的翻译功能将它翻译成你熟悉的语言即可。 |
| 113 | + |
| 114 | +### 使用Vue.js渲染页面 |
| 115 | + |
| 116 | +关于Vue.js的知识,我们在第21天到第30天的内容中已经介绍过了,这里我们不再进行赘述。如果希望全面的了解和学习Vue.js,建议阅读它的[官方教程](<https://cn.vuejs.org/v2/guide/>)或者在[YouTube](<https://www.youtube.com/>)上搜索Vue.js的新手教程(Crash Course)进行学习。 |
| 117 | + |
| 118 | +重新改写subjects.html页面,使用Vue.js来渲染页面。 |
| 119 | + |
| 120 | +```HTML |
| 121 | +<!DOCTYPE html> |
| 122 | +<html lang="en"> |
| 123 | +<head> |
| 124 | + <meta charset="UTF-8"> |
| 125 | + <title>学科</title> |
| 126 | +</head> |
| 127 | +<body> |
| 128 | + <h1>所有学科</h1> |
| 129 | + <hr> |
| 130 | + <div id="app"> |
| 131 | + <div v-for="subject in subjects"> |
| 132 | + <h3> |
| 133 | + <a :href="getTeachersHref(subject.no)">{{ subject.name }}</a> |
| 134 | + <img v-if="subject.isHot" src="/static/images/hot.png" width="32"> |
| 135 | + </h3> |
| 136 | + <p>{{ subject.intro }}</p> |
| 137 | + </div> |
| 138 | + </div> |
| 139 | + <script src="https://cdn.bootcss.com/vue/2.6.10/vue.min.js"></script> |
| 140 | + <script> |
| 141 | + const app = new Vue({ |
| 142 | + el: '#app', |
| 143 | + data: { |
| 144 | + subjects: [] |
| 145 | + }, |
| 146 | + created() { |
| 147 | + fetch('/subjects/') |
| 148 | + .then(resp => resp.json()) |
| 149 | + .then(json => this.subjects = json) |
| 150 | + }, |
| 151 | + methods: { |
| 152 | + getTeachersHref(sno) { |
| 153 | + return `/static/teachers.html/?sno=${sno}` |
| 154 | + } |
| 155 | + } |
| 156 | + }) |
| 157 | + </script> |
| 158 | +</body> |
| 159 | +</html> |
| 160 | +``` |
| 161 | + |
| 162 | +前后端分离的开发需要将前端页面作为静态资源进行部署,项目实际上线的时候,我们会对整个Web应用进行动静分离,静态资源通过Nginx或Apache服务器进行部署,生成动态内容的Python程序部署在uWSGI或者Gunicorn服务器上,对动态内容的请求由Nginx或Apache路由到uWSGI或Gunicorn服务器上。 |
| 163 | + |
| 164 | +在开发阶段,我们通常会使用Django自带的测试服务器,如果要尝试前后端分离,可以先将静态页面放在之前创建的放静态资源的目录下,具体的做法可以参考[项目完整代码](<https://github.com/jackfrued/django1902_vue/>)。 |
0 commit comments