AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79582686
Accepted
chocalaca
chocalaca
Asked: 2025-04-20 02:03:31 +0800 CST2025-04-20 02:03:31 +0800 CST 2025-04-20 02:03:31 +0800 CST

如何在Mysql中正确设置约束?

  • 772

当用户的电子邮件确认后,mysql 时间戳值会更新,并且其值不应等于“1970-01-01 00:00:01”(默认值)。我正在尝试创建一个 mysql 约束来强制执行此操作。因此,如果用户有以下内容:
confirmed=1
email_confirmed_time='1970-01-01 00:00:01'
...则违反了该约束。但如果 confirmed=0且时间戳相同,则表示该消息被接受。

下面的表格命令有什么问题?如果我插入以下内容,它会告诉我CHK_emailConfirmedTime_when_confirmed存在违规:

insert into user (username, `password`, first_name, last_name, `role`, `group`, logged_in, confirmed, approved, email_confirmed_time) values 
    ("0", "0", "joe", "dill", "{admin, user, guest}", "{create, read, update, delete}", 0, 0, 0, DEFAULT)
;
CREATE TABLE IF NOT EXISTS user (
    `id` SERIAL PRIMARY KEY,
    `username` varchar(75) NOT NULL,
    `password` varchar(100) NOT NULL,
    `first_name` varchar(50) NOT NULL,
    `last_name` varchar(50) NOT NULL,
    `role` varchar(25) NOT NULL DEFAULT "{guest}",
    `group` varchar(50) NOT NULL DEFAULT "{read}",
    `logged_in` int NOT NULL DEFAULT 0,
    `confirmed` INTEGER NOT NULL DEFAULT 0,
    `approved` INTEGER NOT NULL DEFAULT 0,
    `email_confirmed_time` TIMESTAMP NOT NULL DEFAULT '1970-01-01 00:00:01',
    CONSTRAINT CHK_confirmed CHECK (`confirmed`=0 OR `confirmed`=1),
    CONSTRAINT CHK_approved CHECK (`approved`=0 OR `approved`=1),
    CONSTRAINT CHK_emailConfirmedTime_when_notConfirmed CHECK (`confirmed`=0 AND `email_confirmed_time`=TIMESTAMP('1970-01-01 00:00:01')),
    CONSTRAINT CHK_emailConfirmedTime_when_confirmed CHECK (`confirmed`=1 AND `email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'))
);
mysql
  • 2 2 个回答
  • 67 Views

2 个回答

  • Voted
  1. Samir Selia
    2025-04-20T02:25:23+08:002025-04-20T02:25:23+08:00

    CHECKMySQL 会针对插入的行独立评估每个已定义的约束。让我们在以下上下文中检查一下每个原始约束:

    1. CONSTRAINT CHK_confirmed CHECK (`confirmed`=0 OR `confirmed`=1)

    • 对于插入的行,confirmed为 0。

    • 条件0 = 0 OR 0 = 1计算结果为TRUE(因为0 = 0是真的)。

    • 因此,该约束得到满足。

    2. CONSTRAINT CHK_approved CHECK (`approved`=0 OR `approved`=1)

    • 对于插入的行,approved为 0。

    • 条件0 = 0 OR 0 = 1计算结果为TRUE。

    • 因此,该约束得到满足。

    3.CONSTRAINT CHK_emailConfirmedTime_when_notConfirmed CHECK (`confirmed`=0 AND `email_confirmed_time`=TIMESTAMP('1970-01-01 00:00:01'))

    • 对于插入的行,confirmed为 0,email_confirmed_time默认为“1970-01-01 00:00:01”。

    • 条件0 = 0 AND '1970-01-01 00:00:01' = TIMESTAMP('1970-01-01 00:00:01')计算结果为TRUE(两个部分都AND为真)。

    • 因此,这个约束也是满足的。

    4.CONSTRAINT CHK_emailConfirmedTime_when_confirmed CHECK (`confirmed`=1 AND `email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'))

    • 对于插入的行,confirmed为 0,email_confirmed_time为“1970-01-01 00:00:01”。

    • 条件0 = 1 AND '1970-01-01 00:00:01' > TIMESTAMP('1970-01-01 00:00:01')计算结果为FALSE(因为0 = 1为假,导致整个AND条件为假)。

    • 因此,该约束不满足。

    一个解决方法是将两个检查结合起来,如下所示

    CONSTRAINT CHK_emailConfirmedTime CHECK (NOT (`confirmed`=1 AND `email_confirmed_time` = TIMESTAMP('1970-01-01 00:00:01')))
    

    这确保了仅当两者都为真时才会违反约束。

    现在,当您插入一行带有confirmed = 0默认值的行时,中的email_confirmed_time条件将为假,从而使整个条件为假。然后,运算符将使约束计算为真,插入将成功。confirmed=1ANDANDNOT

    当confirmed为 1 且email_confirmed_time为默认值时,AND条件为真,NOT运算符将使约束计算结果为假,从而阻止插入或更新。

    我建议将默认值设置为email_confirmed_time而NULL不是1970-01-01 00:00:01,然后相应地更新约束。

    • 4
  2. Best Answer
    ValNik
    2025-04-20T03:09:41+08:002025-04-20T03:09:41+08:00

    检查条件应完全满足。
    因此,当confirmed=1AND时email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'),两个条件都应该成立。此条件禁止插入任何包含confirmed<>1and 的 行email_confirmed_time`<=TIMESTAMP('1970-01-01 00:00:01')

    可能将您的 2 个 CHECK 合并为一个

     CHECK( (`confirmed`=0 AND `email_confirmed_time`=TIMESTAMP('1970-01-01 00:00:01'))
         or (`confirmed`=1 AND `email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'))
          )
    

    并可能删除 CONSTRAINT CHK_confirmed CHECK ( confirmed=0 OR confirmed=1)

    CREATE TABLE IF NOT EXISTS `user` (
        `id` SERIAL PRIMARY KEY,
        `username` varchar(75) NOT NULL,
        `password` varchar(100) NOT NULL,
        `first_name` varchar(50) NOT NULL,
        `last_name` varchar(50) NOT NULL,
        `role` varchar(25) NOT NULL DEFAULT "{guest}",
        `group` varchar(50) NOT NULL DEFAULT "{read}",
        `logged_in` int NOT NULL DEFAULT 0,
        `confirmed` INTEGER NOT NULL DEFAULT 0,
        `approved` INTEGER NOT NULL DEFAULT 0,
        `email_confirmed_time` TIMESTAMP NOT NULL DEFAULT '1970-01-01 00:00:01',
        CONSTRAINT CHK_confirmed CHECK (`confirmed`=0 OR `confirmed`=1),
        CONSTRAINT CHK_approved CHECK (`approved`=0 OR `approved`=1),
        CONSTRAINT CHK_emailConfirmedTime_when_Confirmed_or_notConfirmed 
            CHECK( (`confirmed`=0 AND `email_confirmed_time`=TIMESTAMP('1970-01-01 00:00:01'))
                   or (`confirmed`=1 AND `email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'))
                 )
    );
    
    
    insert into `user` (username, `password`, first_name, last_name, `role`, `group`, logged_in
      , confirmed, approved, email_confirmed_time) values 
        ("0", "0", "joe", "dill", "{admin, user, guest}", "{create, read, update, delete}", 0
      , 0, 0, DEFAULT);
    ;
    
    select * from `user`;
    
    ID 用户名 密码 名 姓 角色 团体 登录 确认的 得到正式认可的 电子邮件确认时间
    1 0 0 乔 莳萝 {管理员、用户、访客} {创建、读取、更新、删除} 0 0 0 1970年1月1日 00:00:01
    insert into `user` (username, `password`, first_name, last_name, `role`, `group`, logged_in
        , confirmed, approved, email_confirmed_time) values 
    ("0", "0", "bill", "dill", "{admin, user, guest}", "{create, read, update, delete}", 0
        , 1, 0, DEFAULT)
    ;
    
    Check constraint 'CHK_emailConfirmedTime_when_Confirmed_or_notConfirmed' is violated.
    
    insert into `user` (username, `password`, first_name, last_name, `role`, `group`, logged_in
        , confirmed, approved, email_confirmed_time) values 
    ("0", "0", "john", "dill", "{admin, user, guest}", "{create, read, update, delete}", 0
        , 0, 0, TIMESTAMP('1970-01-01 00:00:02'))
    ;
    
    Check constraint 'CHK_emailConfirmedTime_when_Confirmed_or_notConfirmed' is violated.
    
    insert into `user` (username, `password`, first_name, last_name, `role`, `group`, logged_in
        , confirmed, approved, email_confirmed_time) values 
    ("0", "0", "mike", "dill", "{admin, user, guest}", "{create, read, update, delete}", 0
        , 1, 0, TIMESTAMP('1970-01-01 00:00:02'))
    ;
    
    select id,first_name,logged_in, confirmed, approved, email_confirmed_time
    from `user`;
    
    ID 名 登录 确认的 得到正式认可的 电子邮件确认时间
    1 乔 0 0 0 1970年1月1日 00:00:01
    2 麦克风 0 1 0 1970年1月1日 00:00:02

    小提琴

    • 3

相关问题

  • mySQL 时间范围之间 - 开始时间可以大于结束时间

  • MySQL:获取连续记录的数量(日期)?

  • Wordpress - 媒体库 - SQL 语法

  • docker-compose:无法将应用程序服务连接到 mysql 数据库,收到“错误:用户‘root’@‘localhost’访问被拒绝(使用密码:YES)

  • MySQL 运行自动脚本

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    重新格式化数字,在固定位置插入分隔符

    • 6 个回答
  • Marko Smith

    为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会?

    • 2 个回答
  • Marko Smith

    VScode 自动卸载扩展的问题(Material 主题)

    • 2 个回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Martin Hope
    Fantastic Mr Fox msvc std::vector 实现中仅不接受可复制类型 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant 使用 chrono 查找下一个工作日 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor 构造函数的成员初始化程序可以包含另一个成员的初始化吗? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský 为什么 C++20 概念会导致循环约束错误,而老式的 SFINAE 不会? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul C++20 是否进行了更改,允许从已知绑定数组“type(&)[N]”转换为未知绑定数组“type(&)[]”? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann 为什么 {2,3,10} 和 {x,3,10} (x=2) 的顺序不同? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve