sql >> Databáze >  >> RDS >> Mysql

PHP + MYSQL na Duplicate KEY stále zvyšují INDEX KEY

dobře, teď jsem si vzpomněl na tento problém. Byl jednou jeden člověk, který chtěl dělat vložky, ale každá vložka musela být v přírůstcích 100 pokud si dokážete představit, od @1000. A museli jsme to celé zabalit do uloženého proc, abychom měli jedno místo zranitelnosti. Váš problém se vynořil a jeho číslování se odhodilo asi o 1.

Tím, že to zabalíme, můžeme mít jeden bod, jak to udělat, se zámkem a zachovat hodnotu auto_inc pomocí ALTER TABLE

Druhý přístup, který jsem mu řekl, byl mít inkrementační tabulku, uzamknout 1 řádek, získat hodnotu v tomto řádku, použít ji, aktualizovat incTable o 100. odemknout.

Celou dobu jsme se smáli problémům s OCD. Myslím, že měl rád pouze násobky 10, idk

Upravit:

Schéma:

-- drop table ocd_nextnums;
create table ocd_nextnums
(   -- id table for nextnum, for the OCD impaired
    tableName varchar(100) not null,
    nextnum int not null
    -- won't bother with indexes, go for it if you want
)engine=INNODB; -- note engine type

insert ocd_nextnums(tableName,nextnum) values('thing',1);
insert ocd_nextnums(tableName,nextnum) values('some_other_table',1);

-- drop table thing;
create table thing
(   id int primary key, -- NOT an auto_increment, but is a PK
    email varchar(100) not null,
    version varchar(20) not null,
    lastupdate datetime not null,
    UNIQUE KEY (email)
)engine=MyIsam;

Uložený proces:

-- drop procedure putInThing;
delimiter $$
create procedure putInThing
(
    email_In varchar(100), version_In varchar(20)
)
BEGIN
    declare toUse int;
    declare theCount int;

    select count(*) into theCount from thing where email=email_In;
    select id into toUse from thing where email=email_In;   -- useful for result set @end
    IF theCount=1 THEN
        -- was there, do UPDATE
        update thing set version=version_In,lastupdate=now() where email=email_In;
    ELSE
        -- new row, do INSERT (please note the FOR UPDATE clause)
        select nextnum into toUse from ocd_nextnums where tableName='thing' FOR UPDATE;
        update ocd_nextnums set nextnum=nextnum+1 where tableName='thing';

        insert thing(id,email,version,lastupdate) values (toUse,email_In,version_In,now());
    end if;
    select toUse;   -- <------- that was your id
END
$$

Test:

call putInThing('[email protected]','111');
call putInThing('[email protected]','121');
call putInThing('[email protected]','107');
select * from thing;
+----+----------+---------+---------------------+
| id | email    | version | lastupdate          |
+----+----------+---------+---------------------+
|  1 | [email protected] | 111     | 2015-08-14 17:08:10 |
|  2 | [email protected] | 121     | 2015-08-14 17:08:54 |
|  3 | [email protected] | 107     | 2015-08-14 17:08:56 |
+----+----------+---------+---------------------+

call putInThing('[email protected]','101111007'); -- is an update
call putInThing('[email protected]','42'); -- is an update
call putInThing('[email protected]','10007'); -- is an update
call putInThing('[email protected]','1'); -- is an insert

select * from thing;
+----+----------------------+---------+---------------------+
| id | email                | version | lastupdate          |
+----+----------------------+---------+---------------------+
|  1 | [email protected]             | 111     | 2015-08-14 17:08:10 |
|  2 | [email protected]             | 121     | 2015-08-14 17:08:54 |
|  3 | [email protected]             | 10007   | 2015-08-14 17:22:09 |
|  4 | [email protected] | 1       | 2015-08-14 17:22:47 |
+----+----------------------+---------+---------------------+

Z části Mysql INNODB příručky :

Uvidíte, že to používám, pravděpodobně ne. Jen to ukázat. Jsem v pohodě s mezerami a spaním v noci. Proto jsem první tabulku pojmenoval, co jsem udělal :>



  1. Seskupování záznamů hodinu po hodině nebo den po dni a vyplnění mezer nulou nebo nulou

  2. Oracle:záleží na pořadí sloupců v indexu?

  3. Spring Boot Application se zasekne na Hikari-Pool-1 - Spouštění...

  4. Jak používat Relation::morphMap() pro různé třídy