MySQLテーブルの主キーをプレフィックス付きで自動インクリメントする方法


80

私はこのようなテーブルを持っています

table
id Varchar(45) NOT NULL AUTO_INCREMENT PRIMARY KEY,
name CHAR(30) NOT NULL,

'LHPL001','LHPL002','LHPL003'...などのようにidフィールドをインクリメントしたい。そのために何をする必要がありますか?可能な方法を教えてください。


これを行うには、別の列を追加するだけです。データベースに「異なる種類」のIDフィールドがあるのはまったく普通のことです。
Fattie

回答:


135

これが本当に必要な場合は、シーケンス用の個別のテーブル(気にしない場合)とトリガーを使用して目標を達成できます。

テーブル

CREATE TABLE table1_seq
(
  id INT NOT NULL AUTO_INCREMENT PRIMARY KEY
);
CREATE TABLE table1
(
  id VARCHAR(7) NOT NULL PRIMARY KEY DEFAULT '0', name VARCHAR(30)
);

今トリガー

DELIMITER $$
CREATE TRIGGER tg_table1_insert
BEFORE INSERT ON table1
FOR EACH ROW
BEGIN
  INSERT INTO table1_seq VALUES (NULL);
  SET NEW.id = CONCAT('LHPL', LPAD(LAST_INSERT_ID(), 3, '0'));
END$$
DELIMITER ;

次に、table1に行を挿入するだけです

INSERT INTO Table1 (name) 
VALUES ('Jhon'), ('Mark');

そして、あなたは持っているでしょう

| ID | 名前|
------------------
| LHPL001 | ジョン|
| LHPL002 | マーク|

これがSQLFiddleのデモです


トリガー内のステートメントはアトミックですか?つまり、INSERTとSETの間に別のスレッド/リクエストがtable1_seqに挿入され、LAST_INSERT_ID()がBEGIN / END内に挿入されたものではなく他のリクエストのIDを返す可能性があります。
n00b 2018年

@peterm、疑わしいトリガー部分があります。SET NEW.id = CONCAT( 'A'、LPAD(LAST_INSERT_ID()、4、 '500'));を変更しました。IDを5001から開始する必要があるため、500を設定しました。正しい出力A5001を取得しています。私の疑問は、IDが5999に達すると、次のIDはどうなるかということです。6000ですか?
user9437856 2018

データベースにレコードを追加した後、自分で試してみました。1000を超えるレコードを追加しましたが、5999以降、ID値はV1000から始まります。これは完全に間違っています。6000,6001,6002など継続する必要があります。これで何か考えはありますか?
user9437856 2018

18

通常の数値auto_incrementIDをZEROFILL使用LPADしてテーブルを作成しますが、それをで定義するか、選択時にゼロを追加するために使用します。次にCONCAT、意図した動作を取得するための値。例1:

create table so (
 id int(3) unsigned zerofill not null auto_increment primary key,
 name varchar(30) not null
);

insert into so set name = 'John';
insert into so set name = 'Mark';

select concat('LHPL', id) as id, name from so;
+---------+------+
| id      | name |
+---------+------+
| LHPL001 | John |
| LHPL002 | Mark |
+---------+------+

例2:

create table so (
 id int unsigned not null auto_increment primary key,
 name varchar(30) not null
);

insert into so set name = 'John';
insert into so set name = 'Mark';

select concat('LHPL', LPAD(id, 3, 0)) as id, name from so;
+---------+------+
| id      | name |
+---------+------+
| LHPL001 | John |
| LHPL002 | Mark |
+---------+------+

当初、LHPLを別の列として格納し、次にconcat(col1、col2)で連結することを考えていましたが、いつかLHPLを変更する必要がある場合、Sで問題が発生します。だから私はプログラムサーバー側でそれをやろうと思います。
4ままカバー

3

私はそれが遅いことを知っていますが、私がこれのためにしたことについて共有したいと思います。別のテーブルやトリガーを追加することは許可されていないため、挿入時に1つのクエリで生成する必要があります。あなたの場合、このクエリを試してみてください。

CREATE TABLE YOURTABLE(
IDNUMBER VARCHAR(7) NOT NULL PRIMARY KEY,
ENAME VARCHAR(30) not null
);

選択を実行し、この選択クエリを使用して、パラメータ@IDNUMBERに保存します

(SELECT IFNULL
     (CONCAT('LHPL',LPAD(
       (SUBSTRING_INDEX
        (MAX(`IDNUMBER`), 'LHPL',-1) + 1), 5, '0')), 'LHPL001')
    AS 'IDNUMBER' FROM YOURTABLE ORDER BY `IDNUMBER` ASC)

そして、挿入クエリは次のようになります。

INSERT INTO YOURTABLE(IDNUMBER, ENAME) VALUES 
(@IDNUMBER, 'EMPLOYEE NAME');

結果は他の回答と同じになりますが、違いは、別のテーブルやトリガーを作成する必要がないことです。私と同じ事件を抱えている人を助けられることを願っています。


0

誰かがPostgreSQLで必要な場合、トリガーなしのPostgreSQLの例を次に示します。

CREATE SEQUENCE messages_seq;

 CREATE TABLE IF NOT EXISTS messages (
    id CHAR(20) NOT NULL DEFAULT ('message_' || nextval('messages_seq')),
    name CHAR(30) NOT NULL,
);

ALTER SEQUENCE messages_seq OWNED BY messages.id;
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.