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 / dba / Perguntas / 77121
Accepted
JasonDavis
JasonDavis
Asked: 2014-09-20 12:08:15 +0800 CST2014-09-20 12:08:15 +0800 CST 2014-09-20 12:08:15 +0800 CST

Atualize um campo de registro MySQL data_modificado APENAS quando outros campos foram alterados usando a instrução CASE não funcionando?

  • 772

Eu fiz minha pergunta no Stackoverflow, mas realmente não obtive nenhuma resposta, então estou tentando aqui agora.

Eu tenho a consulta MySQL que é executada em um script PHP abaixo.

O que ele faz é inserir um novo registro de tarefa do projeto se não existir um com esse número de ID.

Se existir um registro com o número do ID, ele ATUALIZA os campos que deve atualizar.

O problema que tenho é que preciso atualizar a date_modifiedcoluna apenas quando o valor de uma dessas colunas mudar... name, description, status, typeoupriority

public function addOrUpdateTaskRecord($taskId, $projectId, $name, $description, $status, $priority, $type, $date_entered, $date_modified, $sort_order, $heading){

    $sql = "
        INSERT INTO
            $this->tasksDbTableName(task_id, project_id, name, description, status, priority, type, date_entered, date_modified, sort_order, heading)
        VALUES
            ('$taskId', '$projectId', '$name', '$description', '$status', '$priority', '$type', UTC_TIMESTAMP(), UTC_TIMESTAMP(), '$sort_order', '$heading')
        ON DUPLICATE KEY UPDATE
            name='$name',
            description='$description',
            status='$status',
            priority='$priority',
            type='$type',
            date_modified=UTC_TIMESTAMP(),
            sort_order='$sort_order',
            heading='$heading'";

     return $this->db->query($sql);

}

Outro usuário aqui, Gordon Linoff, me mostrou que eu poderia mudar isso date_modified=UTC_TIMESTAMP(),e usar isso no lugar...

date_modified = (case when name <> values(name) or
                           description <> values(description) or
                           status <> values(status) or
                           type <> values(type) or
                           priority <> values(priority)
                      then UTC_TIMESTAMP()
                      else date_modified
                 end)

Isso parecia realmente promissor e não produz erros no MySQL. Ele ainda insere um novo registro quando necessário e atualiza quando é um registro existente, mas agora sempre atualiza o date_modifiedcampo mesmo quando outros campos não foram alterados em minha instrução CASE.

Eu realmente preciso de alguma ajuda para fazer essa funcionalidade avançada funcionar ou um método alternativo?

Alguma ajuda dos especialistas em banco de dados por aí?

Meu código mais recente....

public function addOrUpdateTaskRecord($taskId, $projectId, $name, $description, $status, $priority, $type, $date_entered, $date_modified, $sort_order, $heading){

    $sql = "
        INSERT INTO
            $this->tasksDbTableName(task_id, project_id, name, description, status, priority, type, date_entered, date_modified, sort_order, heading)
        VALUES
            ('$taskId', '$projectId', '$name', '$description', '$status', '$priority', '$type', UTC_TIMESTAMP(), UTC_TIMESTAMP(), '$sort_order', '$heading')
        ON DUPLICATE KEY UPDATE
            name='$name',
            description='$description',
            status='$status',
            priority='$priority',
            type='$type',
            date_modified = (case when name <> values(name) or
                       description <> values(description) or
                       status <> values(status) or
                       type <> values(type) or
                       priority <> values(priority)
                  then UTC_TIMESTAMP()
                  else date_modified
            end),
            sort_order='$sort_order',
            heading='$heading'";

     return $this->db->query($sql);

}

Minha tabela de banco de dados está assim...

CREATE TABLE IF NOT EXISTS apoll_web_projects_tasks (
            task_id bigint(20) NOT NULL default '0',
            project_id varchar(70) NOT NULL default '0',
            name varchar(255) NOT NULL,
            description varchar(36) default NULL,
            status varchar(20) DEFAULT NULL,
            priority varchar(20) DEFAULT NULL,
            type varchar(20) DEFAULT NULL,
            date_entered datetime DEFAULT NULL,
            date_modified datetime DEFAULT NULL,
            sort_order int(12) DEFAULT NULL,
            heading int(2) NOT NULL default '0',
            PRIMARY KEY  (task_id)
            ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
mysql case
  • 2 2 respostas
  • 3524 Views

2 respostas

  • Voted
  1. Best Answer
    JasonDavis
    2014-09-21T21:03:23+08:002014-09-21T21:03:23+08:00

    Abaixo está minha solução de trabalho final. Acontece que mover isso...

    date_modified = (CASE
                    WHEN name <> values(name)
                    OR description <> values(description)
                    OR status <> values(status)
                    OR type <> values(type)
                    OR priority <> values(priority)
                      THEN UTC_TIMESTAMP()
                      ELSE date_modified
                END),
    

    para o topo da lista, acima destes campos...

    name='$name',
    description='$description',
    status='$status',
    priority='$priority',
    type='$type',
    

    Então a solução final fica assim abaixo...

    $sql = "
        INSERT INTO
            $this->tasksDbTableName(task_id, project_id, name, description, status, priority, type, date_entered, date_modified, sort_order, heading)
        VALUES
            ('$taskId', '$projectId', '$name', '$description', '$status', '$priority', '$type', UTC_TIMESTAMP(), UTC_TIMESTAMP(), '$sort_order', '$heading')
        ON DUPLICATE KEY UPDATE
            date_modified = (CASE
                WHEN name <> values(name)
                OR description <> values(description)
                OR status <> values(status)
                OR type <> values(type)
                OR priority <> values(priority)
                  THEN UTC_TIMESTAMP()
                  ELSE date_modified
            END),
            name='$name',
            description='$description',
            status='$status',
            priority='$priority',
            type='$type',
            sort_order='$sort_order',
            heading='$heading'";
    

    Apenas movendo meu CASE Statementpara o topo acima dos outros campos definidos, ele começou a funcionar corretamente!

    Quase parece que esses campos de coluna do banco de dados name description status priority typeestavam sendo atualizados antes que minha instrução Case pudesse ser executada, o que resultou em eles sempre parecendo iguais aos campos do banco de dados e nunca diferentes, então minha instrução Case era quase inútil. Movendo-o para o topo, agora funciona 100% corretamente.

    Espero que isso ajude alguém com um problema semelhante algum dia, já que muitas pessoas em 3 sites tentaram consertá-lo sem sucesso. Apenas movê-lo para o topo fez o truque, então a Ordem neste caso é muito importante! ] Obrigado a todos que contribuíram com seu tempo para tentar encontrar uma solução, sempre aprecio a ajuda que recebo nos sites da rede StackExchange!

    • 3
  2. D8AGOD
    2014-09-20T13:06:23+08:002014-09-20T13:06:23+08:00

    Eu sugeriria retirar o sql in-line. Em seguida, crie um procedimento/gatilho que você possa aplicar os blocos adequados IFpara capturar os valores absolutos que deseja ATUALIZAR. Por exemplo:

    IF (date_modified = _this_ ) THEN
      SET _new_var_for_date_ := _this_desired_timestamp_;
    ELSEIF (date_modified = _this_ ) THEN
      SET _new_var_for_date := _this_desired_timestamp_;
    ...
    ELSE
      SET _new_var_for_date := date_modified
    END IF;
    

    Isso a.) eliminará seu sql em linha b.) permitirá verificações de integridade de dados c.) fornecerá a você uma instrução INSERT mais limpa de maneira programática.

    Por sua vez, isso permitirá que você percorra a lógica que precisa realizar para o INSERT / UPDATE;

    • 0

relate perguntas

  • Existem ferramentas de benchmarking do MySQL? [fechado]

  • Onde posso encontrar o log lento do mysql?

  • Como posso otimizar um mysqldump de um banco de dados grande?

  • Quando é o momento certo para usar o MariaDB em vez do MySQL e por quê?

  • Como um grupo pode rastrear alterações no esquema do banco de dados?

Sidebar

Stats

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

    conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host

    • 12 respostas
  • Marko Smith

    Como fazer a saída do sqlplus aparecer em uma linha?

    • 3 respostas
  • Marko Smith

    Selecione qual tem data máxima ou data mais recente

    • 3 respostas
  • Marko Smith

    Como faço para listar todos os esquemas no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Listar todas as colunas de uma tabela especificada

    • 5 respostas
  • Marko Smith

    Como usar o sqlplus para se conectar a um banco de dados Oracle localizado em outro host sem modificar meu próprio tnsnames.ora

    • 4 respostas
  • Marko Smith

    Como você mysqldump tabela (s) específica (s)?

    • 4 respostas
  • Marko Smith

    Listar os privilégios do banco de dados usando o psql

    • 10 respostas
  • Marko Smith

    Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Como faço para listar todos os bancos de dados e tabelas usando o psql?

    • 7 respostas
  • Martin Hope
    Jin conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host 2014-12-02 02:54:58 +0800 CST
  • Martin Hope
    Stéphane Como faço para listar todos os esquemas no PostgreSQL? 2013-04-16 11:19:16 +0800 CST
  • Martin Hope
    Mike Walsh Por que o log de transações continua crescendo ou fica sem espaço? 2012-12-05 18:11:22 +0800 CST
  • Martin Hope
    Stephane Rolland Listar todas as colunas de uma tabela especificada 2012-08-14 04:44:44 +0800 CST
  • Martin Hope
    haxney O MySQL pode realizar consultas razoavelmente em bilhões de linhas? 2012-07-03 11:36:13 +0800 CST
  • Martin Hope
    qazwsx Como posso monitorar o andamento de uma importação de um arquivo .sql grande? 2012-05-03 08:54:41 +0800 CST
  • Martin Hope
    markdorison Como você mysqldump tabela (s) específica (s)? 2011-12-17 12:39:37 +0800 CST
  • Martin Hope
    Jonas Como posso cronometrar consultas SQL usando psql? 2011-06-04 02:22:54 +0800 CST
  • Martin Hope
    Jonas Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL? 2011-05-28 00:33:05 +0800 CST
  • Martin Hope
    Jonas Como faço para listar todos os bancos de dados e tabelas usando o psql? 2011-02-18 00:45:49 +0800 CST

Hot tag

sql-server mysql postgresql sql-server-2014 sql-server-2016 oracle sql-server-2008 database-design query-performance sql-server-2017

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