AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 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

Como definir restrições corretamente no Mysql?

  • 772

Quando o e-mail de um usuário é confirmado, um valor de TIMESTAMP do MySQL é atualizado e seu valor não deve ser igual a '1970-01-01 00:00:01' (PADRÃO). Estou tentando criar uma restrição do MySQL que imponha isso. Portanto, se o usuário tiver o seguinte:
confirmed=1
email_confirmed_time='1970-01-01 00:00:01'
... então a restrição é violada. Mas se confirmed=0 com o mesmo TIMESTAMP, então ela é aceita.

O que há de errado com o meu comando table abaixo? Ele me informa que há uma violação em CHK_emailConfirmedTime_when_confirmed se eu inserir o seguinte:

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 respostas
  • 67 Views

2 respostas

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

    O MySQL avalia cada uma das CHECKrestrições definidas independentemente para a linha que está sendo inserida. Vamos examinar cada uma das suas restrições originais neste contexto:

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

    • Para a linha inserida, confirmedé 0.

    • A condição 0 = 0 OR 0 = 1é avaliada como TRUE(porque 0 = 0é verdadeira).

    • Portanto, essa restrição é satisfeita.

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

    • Para a linha inserida, approvedé 0.

    • A condição 0 = 0 OR 0 = 1é avaliada como TRUE.

    • Portanto, essa restrição é satisfeita.

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

    • Para a linha inserida, confirmedé 0 e email_confirmed_timeo padrão é '1970-01-01 00:00:01'.

    • A condição 0 = 0 AND '1970-01-01 00:00:01' = TIMESTAMP('1970-01-01 00:00:01')é avaliada como TRUE(ambas as partes de ANDsão verdadeiras).

    • Portanto, essa restrição também é satisfeita.

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

    • Para a linha inserida, confirmedé 0 e email_confirmed_timeé '1970-01-01 00:00:01'.

    • A condição 0 = 1 AND '1970-01-01 00:00:01' > TIMESTAMP('1970-01-01 00:00:01')é avaliada como FALSE(porque 0 = 1é falsa, tornando toda a ANDcondição falsa).

    • Portanto, essa restrição não é satisfeita .

    Uma solução poderia ser combinar as duas verificações conforme abaixo

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

    Isso garante que a restrição seja violada somente quando ambas forem verdadeiras.

    Agora, ao inserir uma linha com confirmed = 0e o padrão email_confirmed_time, a condição confirmed=1dentro do ANDserá falsa, tornando toda a ANDcondição falsa. O NOToperador fará com que a restrição seja avaliada como verdadeira, e a inserção será bem-sucedida.

    Quando confirmedfor 1 e email_confirmed_timefor o valor padrão, a ANDcondição será verdadeira e o NOToperador fará com que a restrição seja avaliada como falsa, impedindo assim a inserção ou atualização.

    Eu recomendaria definir o valor padrão de email_confirmed_timepara NULLem vez de 1970-01-01 00:00:01e, então, atualizar as restrições adequadamente.

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

    As condições de verificação devem ser totalmente satisfeitas.
    Portanto, quando confirmed=1AND email_confirmed_time`>TIMESTAMP('1970-01-01 00:00:01'), ambas as condições devem ser verdadeiras. Essas condições proíbem a inserção de qualquer linha com confirmed<>1AND. email_confirmed_time`<=TIMESTAMP('1970-01-01 00:00:01')

    Possível juntar seus 2 CHEQUES em um

     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'))
          )
    

    E pode remover CONSTRAINT CHK_confirmed CHECK ( confirmed=0 OU 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`;
    
    eu ia nome de usuário senha primeiro nome sobrenome papel grupo logado_em confirmado aprovado hora_de_confirmação_de_email
    1 0 0 Joe aneto {admin, usuário, convidado} {criar, ler, atualizar, excluir} 0 0 0 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", "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`;
    
    eu ia primeiro nome logado_em confirmado aprovado hora_de_confirmação_de_email
    1 Joe 0 0 0 1970-01-01 00:00:01
    2 microfone 0 1 0 1970-01-01 00:00:02

    violino

    • 3

relate perguntas

  • mySQL Between Time Range - Onde a hora de início pode ser maior que a hora de término

  • MySQL: Obter contagem de registros consecutivos (data)?

  • Wordpress - Biblioteca de Mídia - Sintaxe SQL

  • docker-compose: Não é possível conectar o serviço de aplicativo ao banco de dados mysql, recebo "Erro: acesso negado para o usuário 'root'@'localhost' (usando a senha: SIM)

  • MySQL Executar um script automático

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST

Hot tag

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

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve