当前位置: 首页 > news >正文

Django5+DRF序列化

概述

本教程将介绍如何创建一个简单的粘贴板代码高亮 Web API。在此过程中,它将介绍构成 REST 框架的各种组件,让你全面了解所有组件是如何组合在一起的。

本教程相当深入,因此在开始学习之前,你可能需要先吃一块饼干,再喝一杯你最喜欢的啤酒。如果你只想快速了解概况,那就去看看快速入门文档吧。

注意:本文代码基于入门教程的代码继续,如果有不明白的地方,建议先看入门教程的代码。

安装依赖

# 之前已经安装的
pip install django
pip install djangorestframework# 需要新安装的
pip install pygments

创建新的应用

完成后,我们就可以创建一个应用程序,用来创建一个简单的 Web API。

python manage.py startapp snippets

注册新的应用:

INSTALLED_APPS = ['django.contrib.admin','django.contrib.auth','django.contrib.contenttypes','django.contrib.sessions','django.contrib.messages','django.contrib.staticfiles','quickstart','snippets','rest_framework',
]

创建模型

在本教程中,我们将首先创建一个用于存储代码片段的简单片段模型。继续编辑 snippets/models.py 文件。注:良好的编程实践包括注释。虽然您可以在本教程代码的存储库版本中找到注释,但我们在此省略了它们,以专注于代码本身。

from django.db import models
from pygments.lexers import get_all_lexers
from pygments.styles import get_all_stylesLEXERS = [item for item in get_all_lexers() if item[1]]
# 编程语言
LANGUAGE_CHOICES = sorted([(item[1][0], item[0]) for item in LEXERS])
# 代码样式
STYLE_CHOICES = sorted([(item, item) for item in get_all_styles()])class Snippet(models.Model):"""代码片段"""created = models.DateTimeField(verbose_name="创建时间", auto_now_add=True)title = models.CharField(verbose_name="标题", max_length=100, blank=True, default='')code = models.TextField(verbose_name="代码")linenos = models.BooleanField(verbose_name="开启行号", default=False)language = models.CharField(verbose_name="编程语言", choices=LANGUAGE_CHOICES, default='python', max_length=100)style = models.CharField(verbose_name="代码样式", choices=STYLE_CHOICES, default='friendly', max_length=100)class Meta:# 根据创建时间升序ordering = ['created']

我们还需要为片段模型创建初始迁移,并首次同步数据库。

python manage.py makemigrations snippets
python manage.py migrate snippets

创建序列化类

要开始使用 Web API,我们首先需要提供一种将片段实例序列化和反序列化为 json 等表示形式的方法。我们可以通过声明与 Django 的表单非常相似的序列化器来实现这一点。在片段目录中创建一个名为 serializers.py 的文件,并添加以下内容。

from rest_framework import serializers
from snippets.models import Snippet, LANGUAGE_CHOICES, STYLE_CHOICESclass SnippetSerializer(serializers.Serializer):"""代码片段序列化类"""# 主键id = serializers.IntegerField(read_only=True)# 标题title = serializers.CharField(required=False, allow_blank=True, max_length=100)# 代码code = serializers.CharField(style={'base_template': 'textarea.html'})# 是否开启行号linenos = serializers.BooleanField(required=False)# 编程语言language = serializers.ChoiceField(choices=LANGUAGE_CHOICES, default='python')# 代码样式style = serializers.ChoiceField(choices=STYLE_CHOICES, default='friendly')def create(self, validated_data):"""根据验证数据创建并返回一个新的 `Snippet` 实例。"""return Snippet.objects.create(**validated_data)def update(self, instance, validated_data):"""根据验证数据更新并返回现有的 `Snippet` 实例。"""instance.title = validated_data.get('title', instance.title)instance.code = validated_data.get('code', instance.code)instance.linenos = validated_data.get('linenos', instance.linenos)instance.language = validated_data.get('language', instance.language)instance.style = validated_data.get('style', instance.style)instance.save()return instance

序列化器类的第一部分定义了序列化/反序列化的字段。create() 和 update() 方法定义了在调用 serializer.save() 时如何创建或修改完整的实例。

序列化器类与 Django 表单类非常相似,包括各种字段的类似验证标志,如 required、max_length 和 default。

字段标志还可以控制序列化器在某些情况下的显示方式,例如渲染为 HTML 时。上面的 {‘base_template’: ‘textarea.html’} 标志相当于在 Django 表单类上使用 widget=widgets.Textarea。这对于控制如何显示可浏览 API 尤为有用,我们将在本教程稍后部分看到这一点。

实际上,我们还可以通过使用 ModelSerializer 类来节省时间,稍后我们将看到这一点,但现在我们将保持序列化器定义的明确性。

序列化类的用法

在进一步了解之前,我们先熟悉一下如何使用新的序列化器类。让我们进入 Django shell。

python manage.py shell

好了,在我们完成一些导入后,让我们创建几个代码片段来使用。

from snippets.models import Snippet
from snippets.serializers import SnippetSerializer
from rest_framework.renderers import JSONRenderer
from rest_framework.parsers import JSONParsersnippet = Snippet(code='foo = "bar"\n')
snippet.save()snippet = Snippet(code='print("hello, world")\n')
snippet.save()

现在,我们有了几个片段实例可以使用。让我们看看如何将其中一个实例序列化。

serializer = SnippetSerializer(snippet)
serializer.data
# {'id': 2, 'title': '', 'code': 'print("hello, world")\n', 'linenos': False, 'language': 'python', 'style': 'friendly'}

至此,我们已将模型实例转化为 Python 本地数据类型。为了最终完成序列化过程,我们将数据渲染为 json。

content = JSONRenderer().render(serializer.data)
content
# b'{"id": 2, "title": "", "code": "print(\\"hello, world\\")\\n", "linenos": false, "language": "python", "style": "friendly"}'

反序列化与此类似。首先,我们将数据流解析为 Python 原生数据类型…

import iostream = io.BytesIO(content)
data = JSONParser().parse(stream)

…然后我们将这些本地数据类型还原成一个完全填充的对象实例。

serializer = SnippetSerializer(data=data)
serializer.is_valid()
# True
serializer.validated_data
# OrderedDict([('title', ''), ('code', 'print("hello, world")\n'), ('linenos', False), ('language', 'python'), ('style', 'friendly')])
serializer.save()
# <Snippet: Snippet object>

请注意,API 与使用表单是多么相似。当我们开始编写使用序列化器的视图时,这种相似性会变得更加明显。

我们还可以序列化查询集而不是模型实例。为此,我们只需在序列化器参数中添加 many=True 标志即可。

serializer = SnippetSerializer(Snippet.objects.all(), many=True)
serializer.data
# [OrderedDict([('id', 1), ('title', ''), ('code', 'foo = "bar"\n'), ('linenos', False), ('language', 'python'), ('style', 'friendly')]), OrderedDict([('id', 2), ('title', ''), ('code', 'print("hello, world")\n'), ('linenos', False), ('language', 'python'), ('style', 'friendly')]), OrderedDict([('id', 3), ('title', ''), ('code', 'print("hello, world")'), ('linenos', False), ('language', 'python'), ('style', 'friendly')])]

使用模型序列化类

我们的 SnippetSerializer 类正在复制 Snippet 模型中包含的大量信息。如果我们能让代码更简洁一些就更好了。

就像 Django 提供表单类和 ModelForm 类一样,REST 框架也包括序列化器类和模型序列化器类。

让我们看看如何使用 ModelSerializer 类重构序列化器。再次打开文件 snippets/serializers.py,用以下代码替换 SnippetSerializer 类。

class SnippetSerializer(serializers.ModelSerializer):class Meta:model = Snippetfields = ['id', 'title', 'code', 'linenos', 'language', 'style']

序列化器有一个很好的特性,那就是你可以通过打印序列化器实例的表示来检查它的所有字段。使用 python manage.py shell 打开 Django shell,然后尝试以下操作:

from snippets.serializers import SnippetSerializer
serializer = SnippetSerializer()
print(repr(serializer))
# SnippetSerializer():
#    id = IntegerField(label='ID', read_only=True)
#    title = CharField(allow_blank=True, max_length=100, required=False)
#    code = CharField(style={'base_template': 'textarea.html'})
#    linenos = BooleanField(required=False)
#    language = ChoiceField(choices=[('Clipper', 'FoxPro'), ('Cucumber', 'Gherkin'), ('RobotFramework', 'RobotFramework'), ('abap', 'ABAP'), ('ada', 'Ada')...
#    style = ChoiceField(choices=[('autumn', 'autumn'), ('borland', 'borland'), ('bw', 'bw'), ('colorful', 'colorful')...

请务必记住,ModelSerializer 类并没有什么特别神奇的功能,它们只是创建序列化类的快捷方式:

  • 自动确定的字段集。
  • create() 和 update() 方法的简单默认实现。

完整代码如下:

from rest_framework import serializers
from snippets.models import Snippetclass SnippetSerializer(serializers.ModelSerializer):class Meta:model = Snippetfields = ['id', 'title', 'code', 'linenos', 'language', 'style']

使用我们的序列化器编写常规 Django 视图

让我们看看如何使用新的 Serializer 类编写 API 视图。目前,我们不会使用 REST 框架的任何其他功能,我们只会把视图写成普通的 Django 视图。

编辑 snippets/views.py 文件,添加以下内容。

from django.http import HttpResponse, JsonResponse
from django.views.decorators.csrf import csrf_exempt
from rest_framework.parsers import JSONParser
from snippets.models import Snippet
from snippets.serializers import SnippetSerializer

我们 API 的根将是一个视图,它支持列出所有现有片段或创建新片段。

@csrf_exempt
def snippet_list(request):"""List all code snippets, or create a new snippet."""if request.method == 'GET':snippets = Snippet.objects.all()serializer = SnippetSerializer(snippets, many=True)return JsonResponse(serializer.data, safe=False)elif request.method == 'POST':data = JSONParser().parse(request)serializer = SnippetSerializer(data=data)if serializer.is_valid():serializer.save()return JsonResponse(serializer.data, status=201)return JsonResponse(serializer.errors, status=400)

请注意,由于我们希望从没有 CSRF 标记的客户端向该视图进行 POST,因此需要将视图标记为 csrf_exempt。这并不是你通常想要做的事情,REST 框架视图实际上使用了比这更合理的行为,但对于我们现在的目的来说,这已经足够了。

我们还需要一个与单个片段相对应的视图,用于检索、更新或删除片段。

@csrf_exempt
def snippet_detail(request, pk):"""Retrieve, update or delete a code snippet."""try:snippet = Snippet.objects.get(pk=pk)except Snippet.DoesNotExist:return HttpResponse(status=404)if request.method == 'GET':serializer = SnippetSerializer(snippet)return JsonResponse(serializer.data)elif request.method == 'PUT':data = JSONParser().parse(request)serializer = SnippetSerializer(snippet, data=data)if serializer.is_valid():serializer.save()return JsonResponse(serializer.data)return JsonResponse(serializer.errors, status=400)elif request.method == 'DELETE':snippet.delete()return HttpResponse(status=204)

完整代码如下:

from django.http import HttpResponse, JsonResponse
from django.views.decorators.csrf import csrf_exempt
from rest_framework.parsers import JSONParser
from snippets.models import Snippet
from snippets.serializers import SnippetSerializer@csrf_exempt
def snippet_list(request):"""列出所有代码片段,或创建一个新片段。"""# 查询所有的代码if request.method == 'GET':snippets = Snippet.objects.all()serializer = SnippetSerializer(snippets, many=True)return JsonResponse(serializer.data, safe=False)# 新增代码elif request.method == 'POST':data = JSONParser().parse(request)serializer = SnippetSerializer(data=data)if serializer.is_valid():serializer.save()return JsonResponse(serializer.data, status=201)return JsonResponse(serializer.errors, status=400)@csrf_exempt
def snippet_detail(request, pk):"""检索、更新或删除代码片段。"""try:snippet = Snippet.objects.get(pk=pk)except Snippet.DoesNotExist:return HttpResponse(status=404)# 根据ID获取代码if request.method == 'GET':serializer = SnippetSerializer(snippet)return JsonResponse(serializer.data)# 根据ID更新代码elif request.method == 'PUT':data = JSONParser().parse(request)serializer = SnippetSerializer(snippet, data=data)if serializer.is_valid():serializer.save()return JsonResponse(serializer.data)return JsonResponse(serializer.errors, status=400)# 根据ID删除代码elif request.method == 'DELETE':snippet.delete()return HttpResponse(status=204)

配置路由

最后,我们需要将这些视图连接起来。创建 snippets/urls.py 文件:

from django.urls import path
from snippets import viewsurlpatterns = [path('', views.snippet_list),path('<int:pk>/', views.snippet_detail),
]

我们还需要对 tutorial/urls.py 文件中的 urlconf 根目录进行布线,以包含我们的片段应用程序的 URL。

from django.urls import include, path
from rest_framework import routersfrom quickstart import views# 创建子路由
router = routers.DefaultRouter()
router.register(r'users', views.UserViewSet)
router.register(r'groups', views.GroupViewSet)# 使用自动 URL 路由为我们的应用程序接口布线。
# 此外,我们还提供了可浏览 API 的登录 URL。
urlpatterns = [path('', include(router.urls)),path('api-auth/', include('rest_framework.urls', namespace='rest_framework')),path('snippets/', include('snippets.urls')),
]urlpatterns += router.urls

值得注意的是,有几种边缘情况我们目前还没有处理好。如果我们发送了畸形的 json,或者请求使用了视图无法处理的方法,那么我们最终会收到 500 "服务器错误 "的响应。不过,现在这样也可以。

测试我们的接口

新增代码片段:

(venv) PS D:\tmp\drf_demo>  http http://127.0.0.1:8000/snippets/ code=abc
HTTP/1.1 201 Created
Content-Length: 98
Content-Type: application/json
Cross-Origin-Opener-Policy: same-origin
Date: Sun, 07 Jan 2024 04:30:19 GMT
Referrer-Policy: same-origin
Server: WSGIServer/0.2 CPython/3.12.0
X-Content-Type-Options: nosniff
X-Frame-Options: DENY{"code": "abc","id": 1,"language": "python","linenos": false,"style": "friendly","title": ""
}

请求所有代码片段:

(venv) PS D:\tmp\drf_demo> http http://127.0.0.1:8000/snippets/
HTTP/1.1 200 OK
Content-Length: 100
Content-Type: application/json
Cross-Origin-Opener-Policy: same-origin
Date: Sun, 07 Jan 2024 04:30:35 GMT
Referrer-Policy: same-origin
Server: WSGIServer/0.2 CPython/3.12.0
X-Content-Type-Options: nosniff
X-Frame-Options: DENY[{"code": "abc","id": 1,"language": "python","linenos": false,"style": "friendly","title": ""}
](venv) PS D:\tmp\drf_demo>

总结

到目前为止,我们做得还不错,我们已经有了一个序列化 API,感觉与 Django 的表单 API 和一些常规的 Django 视图非常相似。

目前,我们的 API 视图除了提供 json 响应外,并没有做什么特别的事情,而且我们还想清理一些错误处理的边缘情况,但这已经是一个正常运行的 Web API 了。

我们将在本教程的第二部分看看如何开始改进。

相关文章:

Django5+DRF序列化

概述 本教程将介绍如何创建一个简单的粘贴板代码高亮 Web API。在此过程中&#xff0c;它将介绍构成 REST 框架的各种组件&#xff0c;让你全面了解所有组件是如何组合在一起的。 本教程相当深入&#xff0c;因此在开始学习之前&#xff0c;你可能需要先吃一块饼干&#xff0…...

什么是编译程序和解释程序

一、编译程序 1、编译器接收源代码作为输入&#xff0c;它会一次性地将整个源代码程序转换成目标代码&#xff08;通常是机器语言或汇编语言&#xff09;&#xff0c;这个过程包括词法分析、语法分析、语义分析、优化以及最终的目标代码生成。2、编译后的目标代码是一个独立的…...

文档审阅批注的合并和对比

#创作灵感# 最近在改论文&#xff0c;Feedback返回的时候&#xff0c;把之前的批注都删了&#xff0c;这就增加了工作量&#xff0c;看起来不方便&#xff0c;所以就需要将删掉的批注全部复原。 那在原来的文档重新在修改一遍&#xff0c;工作量还是很大的&#xff0c;所以这里…...

广义零样本学习综述的笔记

1 Title A Review of Generalized Zero-Shot Learning Methods&#xff08;Farhad Pourpanah; Moloud Abdar; Yuxuan Luo; Xinlei Zhou; Ran Wang; Chee Peng Lim&#xff09;【IEEE Transactions on Pattern Analysis and Machine Intelligence 2022】 2 conclusion Generali…...

java每日一题——输出9x9乘法表(答案及编程思路)

前言&#xff1a; 打好基础&#xff0c;daydayup! 题目&#xff1a;输出下图9x9乘法表 编程思路&#xff1a;java只能输出行&#xff0c;不能输出列&#xff0c;所以考虑好每一行输出的内容即可 public class demo {public static void main(String[] args) {for (int i 1; i…...

Android 车联网——基础简介(一)

传统的车载功能单一,无太多娱乐性,而随着智能化时代的发展,车载系统也被赋予了在系统中预装 Android 应用的能力,基于Android平台的车载信息娱乐系统 —— Android AutoMotive 应运而生。 一、AutoMotive简介 Android Automotive OS 车载操作系统,是一个基本 Android 平台…...

自动驾驶货车编队行驶系统功能规范

货车编队行驶功能规范 Truck Platooning Functional Specification 目录 1 概述... 7 1.1 目的... 7 1.2 范围... 7 1.3 术语及缩写... 7 1.4 参考法规标准... 8 2 功能规范... 9 2.1 功能描述... 9 2.1.1 功能用途…...

javafx

JavaFX JavaFX简介 JavaFX是一个用于创建富客户端应用程序的图形用户界面&#xff08;GUI&#xff09;框架。它是Java平台的一部分&#xff0c;从Java 8开始成为Java的标准库。 JavaFX提供了丰富的图形和多媒体功能&#xff0c;使开发人员能够创建具有吸引力和交互性的应用程…...

玩转贝启科技BQ3588C开源鸿蒙系统开发板 —— 编译构建及此过程中的踩坑填坑(3)

接前一篇文章&#xff1a;玩转贝启科技BQ3588C开源鸿蒙系统开发板 —— 编译构建及此过程中的踩坑填坑&#xff08;2&#xff09; 上一篇文章结束时在等待提示的各依赖包下载安装后的编译结果&#xff0c;但是很遗憾&#xff0c;编译并没有最终完成&#xff0c;既未成功也没有失…...

SQL ORDER BY 关键字

ORDER BY 关键字用于对结果集进行排序。 SQL ORDER BY 关键字 ORDER BY 关键字用于对结果集按照一个列或者多个列进行排序。 ORDER BY 关键字默认按照升序对记录进行排序。如果需要按照降序对记录进行排序&#xff0c;您可以使用 DESC 关键字。 SQL ORDER BY 语法 SELECT …...

多线程-生产者消费者模型

一、基本信息 1、场景介绍&#xff1a;厨师和吃货的例子&#xff0c;吃货吃桌子上的面条&#xff0c;吃完让厨师做&#xff0c;厨师做完面条放桌子上&#xff0c;让吃货吃&#xff0c;厨师如果发现桌子上有面条&#xff0c;就不做&#xff0c;吃货发现桌子上没有面条就不吃。 …...

解压命令之一 gzip

文章目录 解压命令之一 gzip更多信息 解压命令之一 gzip gzip用于对后缀为gz文件进行解压&#xff1a; $ gzip -d data.gz这个命令将解压examplefile.gz&#xff0c;并且在当前目录下生成一个名为data的解压后的文件。 但特别需要留意的是&#xff0c;这个操作会删除源文件&…...

力扣:438. 找到字符串中所有字母异位词 题解

Problem: 438. 找到字符串中所有字母异位词 438. 找到字符串中所有字母异位词 预备知识解题思路复杂度Code其它细节推荐博客或题目博客题目滑动窗口哈希表 预备知识 此题用到了双指针算法中的滑动窗口思想&#xff0c;以及哈希表的运用。c中是unordered_map。如果对此不了解的u…...

QT 高DPI解决方案

一、根据DPI实现动态调整控件大小&#xff08;三种方式&#xff09; 1、QT支持高DPI&#xff08;针对整个进程中所有的UI&#xff09; // main函数中 QApplication::setAttribute(Qt::AA_EnableHighDpiScaling)tips&#xff1a;&#xff08;1&#xff09;如果不想全局设置&am…...

SLB、DMZ、Nginx、Ingress、Gateway、Kibana和Grafana

SLB、DMZ、Nginx、Ingress、Gateway、Kibana和Grafana虽然有一些相似之处&#xff0c;但是它们的功能和适用场景还是有所不同。 SLB主要用于将大流量的请求分配到多个服务器上进行处理&#xff0c;从而提高系统的可伸缩性和可靠性。它适用于需要处理大流量的应用&#xff0c;如…...

【已解决】Invalid bound statement (not found)

报错讯息 org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): com.casey.mapper.SysRoleMapper.getUserRoleCode at org.apache.ibatis.binding.MapperMethod S q l C o m m a n d . < i n i t > ( M a p p e r M e t h o d . j a v a :…...

汽车信息安全--芯片厂、OEM安全启动汇总(1)

目录 1.芯驰E3安全启动 2.STM32 X-CUBE-SBSFU 3.小米澎湃OS安全启动 4.小结 我在前篇文章里详细记录了车规MCU信息安全设计过程关于网络安全架构的思考过程,从芯片原厂、供应商、OEM等角度思考如何建立起完备的信任链; 不过这思考过程仅仅只是一家之言,因此我又对比了国…...

气膜建筑:舒适、智能、可持续

气膜建筑之所以能够拥有广阔的发展空间&#xff0c;源于其融合了诸多优势特点&#xff0c;使其成为未来建筑领域的前沿趋势。 气膜建筑注重环境可持续性和能源效率。在材料和设计上&#xff0c;它采用可回收材料、提高热保温效果&#xff0c;并积极利用太阳能等可再生能源&…...

【C语言】一种状态超时阻塞循环查询的办法

【C语言】一种状态超时阻塞循环查询的办法 文章目录 【C语言】一种状态超时阻塞循环查询的办法1.方法12.方法21.方法1 static void wait_notify_async(notify_type_t notify_type) {static rt_tick_t exit_tick;exit_tick = rt_time_get_msec();lb_int32 notify_success = RT_F…...

【leetcode】力扣热门之回文链表【简单难度】

题目描述 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为回文链表。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 用例 输入&#xff1a;head [1,2,2,1] 输出&#xff1a;true 输入&#xff1a;head [1,2] 输出&#xff1a;f…...

【MySQL】ALL函数的巧用 以及 排序(order by)巧用 sum(条件表达式) 语法

力扣题 1、题目地址 578. 查询回答率最高的问题 2、模拟表 SurveyLog 表&#xff1a; Column NameTypeidintactionENUMquestion_idintanswer_idintq_numinttimestampint 这张表可能包含重复项。action 是一个 ENUM(category) 数据&#xff0c;可以是 “show”、“answer”…...

Debezium发布历史49

原文地址&#xff1a; https://debezium.io/blog/2019/02/19/reliable-microservices-data-exchange-with-the-outbox-pattern/ 欢迎关注留言&#xff0c;我是收集整理小能手&#xff0c;工具翻译&#xff0c;仅供参考&#xff0c;笔芯笔芯. 使用发件箱模式进行可靠的微服务数…...

数据结构——队列(Queue)

目录 1.队列的介绍 2.队列工程 2.1 队列的定义 2.1.1 数组实现队列 2.1.2 单链表实现队列 2.2 队列的函数接口 2.2.1 队列的初始化 2.2.2 队列的数据插入&#xff08;入队&#xff09; 2.2.3 队列的数据删除&#xff08;出队&#xff09; 2.2.4 取队头数据 2.2.5 取队…...

uniapp微信小程序投票系统实战 (SpringBoot2+vue3.2+element plus ) -后端架构搭建

锋哥原创的uniapp微信小程序投票系统实战&#xff1a; uniapp微信小程序投票系统实战课程 (SpringBoot2vue3.2element plus ) ( 火爆连载更新中... )_哔哩哔哩_bilibiliuniapp微信小程序投票系统实战课程 (SpringBoot2vue3.2element plus ) ( 火爆连载更新中... )共计21条视频…...

两种方式实现mysql截取年月日

select date_format(now(), %Y-%m-%d) select substring(now(), 1, 10)...

WPF 使用矢量字体图标

矢量字体图标 在WPF项目中经常需要显示图标&#xff0c;但是项目改动后&#xff0c;有时候需要替换和修改图标&#xff0c;这样非常麻烦且消耗开发和美工的时间。为了快速开发项目&#xff0c;节省项目时间&#xff0c;使用图标矢量字体图标是一个非常不错的选择。 矢量字体图标…...

编程语言的语法糖,你了解多少?

什么是语法糖 语法糖是一种编程语言的特性&#xff0c;通常是一些简单的语法结构或函数调用&#xff0c;它可以通过隐藏底层的复杂性&#xff0c;并提供更高级别的抽象&#xff0c;从而使代码更加简洁、易读和易于理解&#xff0c;但它并不会改变代码的执行方式。 为什么需要语…...

MySQL中FLUSH TABLES命令语法

在MySQL中&#xff0c;FLUSH TABLES 命令的作用是刷新表。当你使用 FLUSH TABLES 命令的具体选项时&#xff08;例如 FLUSH TABLES WITH READ LOCK&#xff09;&#xff0c;该选项必须是在 FLUSH 语句中唯一指定的命令。即&#xff0c;在一个 FLUSH 语句中&#xff0c;你只能使…...

如何在小米4A刷OpenWRT系统并通过cpolar实现公网访问本地路由器

文章目录 前言1. 安装Python和需要的库2. 使用 OpenWRTInvasion 破解路由器3. 备份当前分区并刷入新的Breed4. 安装cpolar内网穿透4.1 注册账号4.2 下载cpolar客户端4.3 登录cpolar web ui管理界面4.4 创建公网地址 5. 固定公网地址访问 前言 OpenWRT是一个高度模块化、高度自…...

Spring学习之——事务控制

Spring中的事务控制 说明&#xff1a; JavaEE体系进行分层开发&#xff0c;事务处理位于业务层&#xff0c;Spring提供了分层设计业务层的事务处理解决方案。 Spring框架为我们提供了一组事务控制的接口。具体在后面的小节介绍。这组接口是在spring-tx.RELEASE.jar中。 spri…...

建设高流量网站/长春网站开发

文章目录一、eBPF编程接口1.1 系统调用bpf(2)1.2 eBPF的系统调用命令1.3 eBPF的程序类型1.4 eBPF使用的数据结构&#xff08;Map映射&#xff09;1.5 eBPF辅助函数/BPF API的作用&#xff1f;有哪些&#xff1f;1.6 eBPF并发控制1.7 eBPF的sysfs接口1.8 BTF、CO-RE是BPF工具的未…...

公司网站用模板做/西安百度代运营

1.避免漏测 我们肯定都遇到过这样一种情况&#xff0c;有时你在做某事的时候&#xff0c;突然想起来一件事来&#xff0c;但没过几分钟你就又忘记了&#xff0c;后面你总是觉得好像要做什么&#xff0c;但就是想不起来是什么&#xff0c;这时最好的解决方法就是写下来&#xff…...

java怎么自学/seo是指什么职位

1.已有类为partial&#xff0c;部分类&#xff08;public partial class User&#xff09; 2.设计类为partial&#xff0c;添加嵌套类&#xff08;private class MetaUser&#xff09;&#xff0c;嵌套类设计已有类字段 3.设计类添加[MetaDataType(typeof(MetaUser))] public p…...

朝阳网站建设 国展/网络推广seo怎么做

目录 题目 思路 考点 Code 题目 公司用一个字符串来表示员工的出勤信息 absent:缺勤 late:迟到 leaveearly:早退 present:正常上班 现需根据员工出勤信息,判断本次是否能获得出勤奖,能获得出勤奖的条件如下: 缺勤不超过一次; 没有连续的迟到/早退; 任意连续7次考勤…...

wordpress 评论时间/网络服务商怎么咨询

上代码&#xff1a; :http-request"(file)>imgUploadLogin(file, 上传的参数)"这是upload组件的上传文件成功时的钩子。 文件上传一、 methods:{imgUploadLogin(file, name) {//name就是上传的参数let content file.file;let data new FormData();data.appen…...

十大高端网站建设/比较靠谱的电商培训机构

一次WIFI渗透小米4A千兆路由器提权开telnet 到朋友家里&#xff0c;手机信号太差了&#xff0c;于是想连个wifi网络。苦于没有Wi-Fi密码&#xff0c;于是这篇文章便有了着落。 1. 网络嗅探 打开手机 WIFI 发现周围的 WIFI 热点挺多的&#xff0c;用模拟器下载WIFI万能钥匙开始…...