찾다

 >  Q&A  >  본문

innodb - 如何实现mysql线程安全的nextval()方法

尝试通过建立序列表与函数实现,可以正常返回下条序列,但通过jmeter测试并发100个线程,发现返回的结果中存在重复的记录。

下面是链接中的实现方法

-- 表
CREATE TABLE `sequence_data` (
    `sequence_name` varchar(100) NOT NULL,
    `sequence_increment` int(11) unsigned NOT NULL DEFAULT 1,
    `sequence_min_value` int(11) unsigned NOT NULL DEFAULT 1,
    `sequence_max_value` bigint(20) unsigned NOT NULL DEFAULT 18446744073709551615,
    `sequence_cur_value` bigint(20) unsigned DEFAULT 1,
    `sequence_cycle` boolean NOT NULL DEFAULT FALSE,
    PRIMARY KEY (`sequence_name`)
);
-- 函数
delimiter //
CREATE FUNCTION `nextval` (`seq_name` varchar(100))
RETURNS bigint(20) NOT DETERMINISTIC
BEGIN
    DECLARE cur_val bigint(20);
 
    SELECT
        sequence_cur_value INTO cur_val
    FROM
        sequence_data
    WHERE
        sequence_name = seq_name
    ;
 
    IF cur_val IS NOT NULL THEN
        UPDATE
            sequence_data
        SET
            sequence_cur_value = IF (
                (sequence_cur_value + sequence_increment) > sequence_max_value,
                IF (
                    sequence_cycle = TRUE,
                    sequence_min_value,
                    NULL
                ),
                sequence_cur_value + sequence_increment
            )
        WHERE
            sequence_name = seq_name
        ;
    END IF;
 
    RETURN cur_val;
END

//
delimiter ;

-- insert new sequence
-- INSERT INTO sequence_data (sequence_name) VALUE ('sq_my_sequence');

-- 调用
-- SELECT nextval('sq_my_sequence') as nextval
PHP中文网PHP中文网2822일 전1185

모든 응답(1)나는 대답할 것이다

  • 黄舟

    黄舟2017-04-17 15:37:29

    mysql의 기본 격리 수준은 반복 가능_읽기입니다. 트랜잭션이 커밋되기 전에는 다른 스레드가 수정 사항을 읽을 수 없습니다. 마지막에 커밋을 추가하여 업데이트 결과를 제출할 수 있습니다. 그래도 작동하지 않으면 행 수준 잠금을 사용합니다(업데이트를 위해 WHERE 시퀀스 이름 = seq_name에서 * 시퀀스_데이터 선택). 한 스레드가 제출되지 않으면 다른 스레드는 읽을 수 없으므로 동시성 안전성이 보장됩니다.

    회신하다
    0
  • 취소회신하다