DjangoORM字段参数、常用字段类型及参数、模型和表单验证器详解

由于项目原因必须使用DjangoORM模型,所以今天整理了一下关于DjangoORM模型里的详细内容。包含字段参数、常用字段类型及参数、模型和表单验证器。


一、通用字段参数
这些参数可以应用于多种字段类型:

(1)null:如果为 True,则数据库中该字段可以存储 NULL 值。默认为 False。

(2)blank: 布尔值。如果为 True,则表单验证允许该字段为空。默认为 False。关键区别:blank 控制表单验证(是否允许在表单中留空),而 null 控制数据库存储(是否允许存储 NULL 值)。 它们是独立的设置。通常,对于字符串类型(CharFieldTextField 等),如果允许为空,应该同时设置 blank=Truenull=True

(3)choices: 一个由二元组 (实际存储值, 显示值) 组成的序列,用于提供字段的可选项。例如:STATUS_CHOICES = [(0, '草稿'), (1, '发布')]。在表单中会显示为下拉框。

(4)default: 字段的默认值。

(5)help_text: 在表单中显示的帮助文本,用于提供字段的说明。

(6)primary_key: 布尔值。如果为 True,则该字段为主键。每个模型只能有一个字段设置 primary_key=True。 通常 Django 会自动添加 AutoFieldBigAutoField 作为主键,除非你需要自定义主键字段名或使用复合主键(不推荐)。

(7)unique: 布尔值。如果为 True,则该字段的值在数据库中必须是唯一的。

(8)verbose_name: 字段的人类可读的名称,用于在管理后台等地方显示。如果不指定,Django 会使用字段名并把下划线转换为空格。

(9)db_column: 用于指定数据库字段的名称。通常不需要手动指定,Django 会根据字段名自动生成。只有在需要与现有数据库集成或有特殊的命名规范时才需要使用。

(10)db_index: 布尔值。如果为 True,则在数据库中为此字段创建索引,可以加快查询速度。索引会提高查询速度,但会降低写入速度,并且会占用额外的存储空间。因此,应该只为经常用于查询的字段创建索引。

二、常用字段类型及参数

(1)AutoField: 一个自动递增的整数类型字段,通常用作主键。在 Django 3.2 之前,默认主键类型是 AutoField,之后是 BigAutoField。强烈建议不要手动指定主键,除非有特殊需求。

BigAutoField: 64 位整数类型字段,类似于 AutoField,但支持更大的数值范围。Django 3.2 之后成为默认主键类型。

(2)CharField: 用于存储短字符串。

max_length: 字符串的最大长度,必须指定。

(3)TextField: 用于存储大量文本。

没有 max_length 限制。

IntegerField、BigIntegerField、SmallIntegerField、PositiveIntegerField、PositiveSmallIntegerField: 用于存储各种大小的整数。

(4)FloatField: 用于存储浮点数。对应 SQL 的 FLOATDOUBLE。精度可能存在问题,对于需要精确计算的场景,建议使用 DecimalField

(5)DecimalField: 用于存储固定精度的十进制数。

max_digits: 数字的最大位数(包括整数部分和小数部分)。

decimal_places: 小数位数。

(6)BooleanField: 用于存储布尔值(TrueFalse)。对应 SQL 的 TINYINT(1)BOOLEAN

(7)NullBooleanField:已完全弃用,应该使用 BooleanField(null=True)

(8)DateField: 用于存储日期。

auto_now: 每次保存对象时自动更新为当前日期。通常用于记录修改时间。

auto_now_add: 对象第一次创建时自动设置为当前日期。通常用于记录创建时间。

auto_nowauto_now_add 互斥,不能同时使用。

(9)DateTimeField: 用于存储日期和时间。

auto_now: 每次保存对象时自动更新为当前日期和时间。

auto_now_add: 对象第一次创建时自动设置为当前日期和时间。

auto_nowauto_now_add 互斥,不能同时使用。

(10)TimeField: 用于存储时间。

(11)DurationField: 用于存储时间间隔。

(12)EmailField: 用于存储电子邮件地址。会对输入进行基本的电子邮件格式验证。

(13)FileField: 用于存储文件。

upload_to: 指定文件上传的目录。强烈推荐使用可调用对象(函数或方法),以便更灵活地组织文件存储路径,例如按日期或用户ID等。

max_length: 文件路径的最大长度。

(14)ImageField: 用于存储图像文件。继承自 FileField。需要安装 Pillow 库 (pip install Pillow)。

(15)URLField: 用于存储 URL。会对输入进行基本的 URL 格式验证。

(16)UUIDField: 用于存储 UUID。

(17)GenericIPAddressField: 用于存储 IPv4 或 IPv6 地址。

protocol: 指定协议('both''IPv4''IPv6')。

unpack_ipv4: 如果为 True,则将 IPv4 地址解包为 IPv6 格式。

(18)ForeignKey: 用于定义多对一关系。

to: 关联的模型。

on_delete: 非常重要! 指定删除关联对象时的行为:

models.CASCADE: 级联删除。

models.PROTECT: 阻止删除。

models.SET_NULL: 设置为 NULL,必须配合 null=True 使用。

models.SET_DEFAULT: 设置为默认值,必须配合 default 参数使用。

models.DO_NOTHING: 什么也不做(不推荐,容易造成数据不一致)。

models.SET(value): 设置为指定的值 valuevalue 也可以是一个可调用对象。

related_name: 反向关联的名称。强烈推荐显式指定,提高代码可读性。

related_query_name: 用于反向查询时的查询名称。

db_constraint: 是否在数据库中创建外键约束,默认为 True。通常保持默认值。

(19)ManyToManyField: 用于定义多对多关系。

through: 指定中间模型。当需要在关系中存储额外数据(例如关系创建时间)时,必须使用 through 指定中间模型。

db_table: 指定数据库表的名称。通常不需要指定,Django 会自动生成。

parent_link: 仅用于多表继承的情况,表示该字段是父模型的链接。并非所有 ManyToManyField 都会用到。

(20)OneToOneField: 用于定义一对一关系。本质上就是一个 unique=TrueForeignKey

再次强调的重要点:

理解 blanknull 的区别及其组合使用方式。

on_delete 参数至关重要,根据业务逻辑谨慎选择。

强烈建议为 ForeignKeyManyToManyField 指定 related_name

NullBooleanField 已完全弃用。

Django 3.2 之后,默认主键类型是 BigAutoField

auto_nowauto_now_add 互斥。

对于浮点数精确计算,使用 DecimalField

结合上面通用字段参数和常用字段类型及参数编写几个关于电商平台的表:

import uuid
from django.db import models
from django.core.exceptions import ValidationError
from django.db.models import F
from django.utils.translation import gettext_lazy as _

class Category(models.Model):
    """商品分类"""
    name = models.CharField(_("分类名称"), max_length=50, unique=True)
    description = models.TextField(_("分类描述"), blank=True, null=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品分类")
        verbose_name_plural = _("商品分类") # 添加复数形式

class Tag(models.Model):
    """商品标签"""
    name = models.CharField(_("标签名称"), max_length=20, unique=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品标签")
        verbose_name_plural = _("商品标签")

class Product(models.Model):
    """商品信息"""
    STATUS_CHOICES = [
        (0, _('下架')),
        (1, _('上架')),
    ]

    id = models.UUIDField(_("商品ID"), primary_key=True, default=uuid.uuid4, editable=False)
    name = models.CharField(_("商品名称"), max_length=100, db_index=True)
    description = models.TextField(_("商品描述"), blank=True, null=True)
    price = models.DecimalField(_("商品价格"), max_digits=14, decimal_places=2)
    stock = models.PositiveIntegerField(_("商品库存"), default=0)
    created_at = models.DateTimeField(_("创建时间"), auto_now_add=True)
    updated_at = models.DateTimeField(_("更新时间"), auto_now=True)
    status = models.IntegerField(_("商品状态"), choices=STATUS_CHOICES, default=1)
    category = models.ForeignKey(Category, on_delete=models.PROTECT, related_name='products', verbose_name=_("商品分类"))
    tags = models.ManyToManyField(Tag, related_name='products', blank=True, verbose_name=_("商品标签"))
    image = models.ImageField(_("商品图片"), upload_to='products/', blank=True, null=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品")
        verbose_name_plural = _("商品")
        ordering = ['-created_at']

class Order(models.Model):
    """订单信息"""
    STATUS_CHOICES = [
        (0, _('待付款')),
        (1, _('已付款')),
        (2, _('已发货')),
        (3, _('已完成')),
        (4, _('已取消')),
    ]
    order_number = models.CharField(_("订单编号"), max_length=20, unique=True)
    created_at = models.DateTimeField(_("下单时间"), auto_now_add=True)
    status = models.IntegerField(_("订单状态"), choices=STATUS_CHOICES, default=0)

    def __str__(self):
        return self.order_number

    class Meta:
        verbose_name = _("订单")
        verbose_name_plural = _("订单")

class OrderItem(models.Model):
    """订单条目"""
    order = models.ForeignKey(Order, on_delete=models.CASCADE, related_name='items', verbose_name=_("所属订单"))
    product = models.ForeignKey(Product, on_delete=models.PROTECT, related_name='order_items', verbose_name=_("商品"))
    quantity = models.PositiveIntegerField(_("购买数量"))
    price = models.DecimalField(_("购买时价格"), max_digits=14, decimal_places=2)

    def clean(self):
        if self.quantity > self.product.stock:
            raise ValidationError({'quantity': _('库存不足')}) # 翻译错误信息

    def save(self, *args, **kwargs):
        # 使用 F() 表达式原子性地更新库存,避免并发问题
        if self.pk is None: #仅在创建新OrderItem时更新库存
            Product.objects.filter(pk=self.product.pk).update(stock=F('stock') - self.quantity)
        super().save(*args, **kwargs)

    class Meta:
        verbose_name = _("订单条目")
        verbose_name_plural = _("订单条目")

# 移除信号处理,使用 F() 表达式更高效和安全
# @receiver(pre_save, sender=OrderItem)
# def check_stock(sender, instance, **kwargs):
#     if instance.quantity > instance.product.stock:
#         raise ValidationError({'quantity': '库存不足'})
# 关于库存管理的更详细解释:
# 使用 F() 表达式是解决并发环境下库存管理的最佳实践。假设有两个用户同时购买同一件商品,如果使用 Python 代码先读取库存,然后减去购买数量再保存,就可能出现以下情况:
# 用户 A 读取库存,假设为 10。
# 用户 B 读取库存,此时库存仍然为 10。
# 用户 A 购买 5 件,库存更新为 5。
# 用户 B 购买 5 件,库存更新为 5。
# 结果是两个用户都购买了 5 件,总共购买了 10 件,而实际上库存只减少了 5 件,导致超卖。

三、模型和表单验证器

(1)数值验证器:

MaxValueValidator(limit_value): 验证值是否小于或等于 limit_value

MinValueValidator(limit_value): 验证值是否大于或等于 limit_value

DecimalValidator(max_digits, decimal_places): 验证十进制数的位数和精度。max_digits 是总位数(包括小数位),decimal_places 是小数位数。

IntegerValidator: 验证值是否为整数。

(2)字符串验证器:

EmailValidator(message=None, code=None, allowlist=None): 验证值是否为有效的电子邮件地址。allowlist 参数允许指定额外的域名白名单。

RegexValidator(regex, message=None, code=None, inverse_match=None, flags=0): 使用正则表达式 regex 验证值。inverse_match 参数用于反向匹配(即值 不 应该匹配正则表达式)。flags 参数用于设置正则表达式的标志(例如 re.IGNORECASE)。

URLValidator(schemes=None, message=None, code=None): 验证值是否为有效的 URL。schemes 参数允许指定允许的 URL 协议列表(例如 ['http', 'https'])。

MaxLengthValidator(limit_value): 验证字符串的最大长度。

MinLengthValidator(limit_value): 验证字符串的最小长度。

(3)文件验证器:

FileExtensionValidator(allowed_extensions, message=None, code=None): 验证上传文件的扩展名是否在 allowed_extensions 列表中。

(4)其他验证器和验证函数:

validate_email(value): 验证值是否为有效的电子邮件地址。(功能与 EmailValidator 类似,但可以直接作为函数使用。)

validate_slug(value): 验证值是否为有效的 slug(URL 中使用的短标签,通常只包含字母、数字、连字符和下划线)。

validate_unicode_slug(value): 验证值是否为有效的 Unicode slug。

validate_ipv4_address(value): 验证值是否为有效的 IPv4 地址。

validate_ipv6_address(value): 验证值是否为有效的 IPv6 地址。

validate_ipv46_address(value): 验证值是否为有效的 IPv4 或 IPv6 地址。

(5)使用方法:

这些验证器可以用于模型字段和表单字段。

(6)模型字段: 将验证器实例作为列表传递给字段的 validators 参数。

(7)表单字段: 同样,将验证器实例列表传递给字段的 validators 参数。

(8)自定义验证器:

除了内置验证器,你还可以创建自定义验证器。自定义验证器是一个可调用对象(例如函数或带有 __call__ 方法的类),它接收一个值,并在值无效时引发 django.core.exceptions.ValidationError 异常。

结合上面模型和表单验证器和上面的案例重新编写关于电商平台的表:

import uuid
from django.db import models, transaction
from django.db.models import F
from django.utils.translation import gettext_lazy as _
from django.core.validators import (
    MaxValueValidator,
    MinValueValidator,
    DecimalValidator,
    RegexValidator,
    URLValidator,
    MaxLengthValidator,
    MinLengthValidator,
    FileExtensionValidator,
    validate_email,
    validate_slug,
)
from django import forms

class Category(models.Model):
    """商品分类"""
    name = models.CharField(_("分类名称"), max_length=50, unique=True)
    description = models.TextField(_("分类描述"), blank=True, null=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品分类")
        verbose_name_plural = _("商品分类")

class Tag(models.Model):
    """商品标签"""
    name = models.CharField(_("标签名称"), max_length=20, unique=True)

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品标签")
        verbose_name_plural = _("商品标签")

class Product(models.Model):
    """商品信息"""
    STATUS_CHOICES = [
        (0, _('下架')),
        (1, _('上架')),
    ]

    id = models.UUIDField(_("商品ID"), primary_key=True, default=uuid.uuid4, editable=False)
    name = models.CharField(_("商品名称"), max_length=100, db_index=True, validators=[MinLengthValidator(2)])
    description = models.TextField(_("商品描述"), blank=True, null=True)
    price = models.DecimalField(_("商品价格"), max_digits=14, decimal_places=2, validators=[MinValueValidator(0)])
    stock = models.PositiveIntegerField(_("商品库存"), default=0) # 移除不必要的MaxValueValidator,或根据实际业务调整
    created_at = models.DateTimeField(_("创建时间"), auto_now_add=True)
    updated_at = models.DateTimeField(_("更新时间"), auto_now=True)
    status = models.IntegerField(_("商品状态"), choices=STATUS_CHOICES, default=1)
    category = models.ForeignKey(Category, on_delete=models.PROTECT, related_name='products', verbose_name=_("商品分类"))
    tags = models.ManyToManyField(Tag, related_name='products', blank=True, verbose_name=_("商品标签"))
    image = models.ImageField(_("商品图片"), upload_to='products/', blank=True, null=True, validators=[FileExtensionValidator(['jpg', 'jpeg', 'png', 'gif'])])

    def __str__(self):
        return self.name

    class Meta:
        verbose_name = _("商品")
        verbose_name_plural = _("商品")
        ordering = ['-created_at']

class Order(models.Model):
    """订单信息"""
    STATUS_CHOICES = [
        (0, _('待付款')),
        (1, _('已付款')),
        (2, _('已发货')),
        (3, _('已完成')),
        (4, _('已取消')),
    ]
    order_number = models.CharField(_("订单编号"), max_length=20, unique=True, db_index=True, validators=[RegexValidator(r'^[A-Za-z0-9-]+$', _('订单编号只能包含字母、数字和连字符'))])
    created_at = models.DateTimeField(_("下单时间"), auto_now_add=True)
    status = models.IntegerField(_("订单状态"), choices=STATUS_CHOICES, default=0)

    def __str__(self):
        return self.order_number

    class Meta:
        verbose_name = _("订单")
        verbose_name_plural = _("订单")

class OrderItem(models.Model):
    """订单条目"""
    order = models.ForeignKey(Order, on_delete=models.CASCADE, related_name='items', verbose_name=_("所属订单"))
    product = models.ForeignKey(Product, on_delete=models.PROTECT, related_name='order_items', verbose_name=_("商品"))
    quantity = models.PositiveIntegerField(_("购买数量"), validators=[MinValueValidator(1)])
    price = models.DecimalField(_("购买时价格"), max_digits=14, decimal_places=2, validators=[MinValueValidator(0)])

    def save(self, *args, **kwargs):
        with transaction.atomic():
            if self.pk is None:
                Product.objects.filter(pk=self.product.pk).update(stock=F('stock') - self.quantity)
            else:
                old_item = OrderItem.objects.get(pk=self.pk)
                stock_diff = old_item.quantity - self.quantity
                Product.objects.filter(pk=self.product.pk).update(stock=F('stock') + stock_diff)
            super().save(*args, **kwargs)

    def delete(self, *args, **kwargs):
        with transaction.atomic():
            Product.objects.filter(pk=self.product.pk).update(stock=F('stock') + self.quantity)
            super().delete(*args, **kwargs)

    class Meta:
        verbose_name = _("订单条目")
        verbose_name_plural = _("订单条目")

class OrderItemForm(forms.ModelForm):
    class Meta:
        model = OrderItem
        fields = '__all__'

    def clean_quantity(self):
        quantity = self.cleaned_data['quantity']
        product = self.cleaned_data['product']
        if quantity > product.stock:
            raise forms.ValidationError(_('库存不足'))
        return quantity

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/947322.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

《Vue3实战教程》37:Vue3生产部署

如果您有疑问,请观看视频教程《Vue3实战教程》 生产部署​ 开发环境 vs. 生产环境​ 在开发过程中,Vue 提供了许多功能来提升开发体验: 对常见错误和隐患的警告对组件 props / 自定义事件的校验响应性调试钩子开发工具集成 然而&#xff…

Ruby自动化:用Watir库获取YouTube视频链接

引言 Watir(Web Application Testing in Ruby)是一个强大的工具,它允许开发者使用Ruby语言来自动化控制浏览器。Watir最初被设计用于自动化Web应用测试,但其功能远不止于此。通过Watir,我们可以模拟用户行为&#xff…

[CTF/网络安全] 攻防世界 warmup 解题详析

查看页面源代码&#xff0c;发现source.php 得到一串代码&#xff0c;进行代码审计&#xff1a; <?phpclass emmm{public static function checkFile(&$page){$whitelist ["source">"source.php","hint">"hint.php"];…

solr9.7 单机安装教程

1.环境要求:jdk11以上 2.下载wget https://dlcdn.apache.org/solr/solr/9.7.0/solr-9.7.0.tgz 3.解压 4.修改solr.in.sh配置 5.启动命令 bin/solr start 6.创建core bin/solr create -c <core名称> 注意:用solr ui界面创建&#xff0c;会提示找不到solrconfig.xml和m…

应用架构模式-总体思路

采用引导式设计方法&#xff1a;以企业级架构为指导&#xff0c;形成较为齐全的规范指引。在实践中总结重要设计形成决策要点&#xff0c;一个决策要点对应一个设计模式。自底向上总结采用该设计模式的必备条件&#xff0c;将之转化通过简单需求分析就能得到的业务特点&#xf…

基于AI大模型的医院SOP优化:架构、实践与展望

一、引言 1.1 研究背景与意义 近年来,人工智能(AI)技术取得了迅猛发展,尤其是大模型的出现,为各个领域带来了革命性的变化。在医疗领域,AI 医疗大模型正逐渐崭露头角,展现出巨大的应用潜力。随着医疗数据的海量积累以及计算能力的大幅提升,AI 医疗大模型能够对复杂的…

AWS re:Invent 2024 - Dr. Werner Vogels 主题演讲

今年&#xff0c;我有幸亲临现场参加了所有的 keynote&#xff0c;每一场都让我感受到深深的震撼。无论是全新的功能发布&#xff0c;还是令人眼前一亮的新特性展示&#xff0c;每一场 keynote 都精彩纷呈&#xff0c;充满干货&#xff0c;值得反复学习和回味。 恰好&#xff…

UnionTech OS Server 20 网页无法访问yum源地址

统信yum地址 https://euler-packages.chinauos.com/server-euler/fuyu/1060/everything/sw_64/Packages/ 浏览器访问401报错无权限&#xff0c;查看linux uos环境下yum配置的用户名和密码 cat /etc/yum/vars/auth_* 然后自己组装生成Basic Authorization def generate_basic_…

自动化测试常考的面试题+答案汇总(持续更新)

Hi&#xff0c;大家好&#xff0c;。最近很多朋友都在说今年的互联网行情不好&#xff0c;面试很难&#xff0c;不知道怎么复习&#xff0c;我最近总结了一份在自动化测试面试中比较常见的面试题合集&#xff0c;希望对大家有帮助。 本文共 4800 字&#xff0c;预计阅读时间 1…

jvm结构介绍

JVM结构概述 Java虚拟机&#xff08;JVM&#xff09;是Java程序的运行环境&#xff0c;它负责将Java字节码转换为机器码并执行。JVM的结构主要包括类加载子系统、运行时数据区、执行引擎、本地接口以及垃圾收集器。 1. 类加载子系统&#xff08;Class Loader Subsystem&#xf…

ruoyi开发学习

将若依框架中的若依元素删掉 1.删除主目录中的“若依官网”&#xff1a; 在后端项目中&#xff0c;idea里借助mysql管理工具&#xff0c;找到sys_menu数据表&#xff0c;双击打开&#xff0c;找到4 若依官网&#xff0c;选中点击减号&#xff0c;绿色上箭头刷新&#xff0c;删…

计算机网络 (13)信道复用技术

前言 计算机网络中的信道复用技术是一种提高网络资源利用率的关键技术。它允许在一条物理信道上同时传输多个用户的信号&#xff0c;从而提高了信道的传输效率和带宽利用率。 一、信道复用技术的定义 信道复用&#xff08;Multiplexing&#xff09;就是在一条传输媒体上同时传输…

您的公司需要小型语言模型

当专用模型超越通用模型时 “越大越好”——这个原则在人工智能领域根深蒂固。每个月都有更大的模型诞生&#xff0c;参数越来越多。各家公司甚至为此建设价值100亿美元的AI数据中心。但这是唯一的方向吗&#xff1f; 在NeurIPS 2024大会上&#xff0c;OpenAI联合创始人伊利亚…

艾体宝产品丨加速开发:Redis 首款 VS Code 扩展上线!

Redis 宣布推出其首款专为 VS Code 设计的 Redis 扩展。这一扩展将 Redis 功能直接整合进您的集成开发环境&#xff08;IDE&#xff09;&#xff0c;旨在简化您的工作流程&#xff0c;提升工作效率。 我们一直致力于构建强大的开发者生态系统&#xff0c;并在您工作的每一步提…

数据挖掘——关联规则挖掘

数据挖掘——关联数据挖掘 关联数据挖掘关联规则关联规则挖掘问题&#xff1a;具体挖掘过程Apriori 产生关联规则 关联数据挖掘 关联分析用于发现隐藏在大型数据集中的令人感兴趣的联系&#xff0c;所发现的模式通常用关联规则或频繁项集的形式表示。 关联规则反映一个事物与…

12.30-1-5学习周报

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 文章链接摘要Abstract一、方法介绍1.HAT-CIR2.Horde3.DWGRNet 二、实验总结 文章链接 https://arxiv.org/pdf/2405.04101 摘要 本博客介绍了论文《Continual lea…

Synopsys软件基本使用方法

Synopsys软件基本使用方法 1 文件说明2 编译流程3 查看波形4 联合仿真 本文主要介绍Synopsys软件vcs、verdi的基本使用方法&#xff0c;相关文件可从 GitHub下载。 1 文件说明 创建verilog源文件add.v、mult.v、top.vmodule add (input signed [31:0] dina,input signed [3…

Linux-Redis哨兵搭建

环境资源准备 主机名IP端口号角色vm1192.168.64.156379/26379mastervm2192.168.64.166379/26379slavevm3192.168.64.176379/26379slave 6379为redis服务暴露端口号、26379为sentinel暴露端口号。 安装Redis # 包文件下载 wget https://github.com/redis/redis/archive/7.2.2…

【python】unittest单元测试

文章目录 基本使用不同启动方式的区别 基本使用 下面是根据文档写的一个demo&#xff0c;主要的内容基本都包含了&#xff0c;使用时导入自己的业务类测试类中的方法就行。 import unittest# 测试类不强制test开头&#xff0c;仅作为规范。但必须继承unittest.TestCase class…

基于SpringBoot的野生动物保护发展平台的设计与实现(源码+SQL+LW+部署讲解)

文章目录 摘 要1. 第1章 选题背景及研究意义1.1 选题背景1.2 研究意义1.3 论文结构安排 2. 第2章 相关开发技术2.1 前端技术2.2 后端技术2.3 数据库技术 3. 第3章 可行性及需求分析3.1 可行性分析3.2 系统需求分析 4. 第4章 系统概要设计4.1 系统功能模块设计4.2 数据库设计 5.…