BEGIN
引入或更新版本:v1.2.371
开始一个新的事务。BEGIN 和 COMMIT/ROLLBACK 必须一起使用,以开始事务并随后保存或撤销事务。
-
Databend 不 支持嵌套事务,因此不匹配的事务语句将被忽略。
示例:BEGIN; -- 开始一个事务
MERGE INTO ... -- 该语句属于事务
BEGIN; -- 在事务中执行 BEGIN 会被忽略,不会开始新事务,也不会报错
INSERT INTO ... -- 该语句也属于事务
COMMIT; -- 结束事务
INSERT INTO ... -- 该语句属于单语句事务
COMMIT; -- 在多语句事务外执行 COMMIT 会被忽略,不会执行提交操作,也不会报错
BEGIN; -- 开始另一个事务
... -
当在多语句事务中执行 DDL 语句时,它将提交当前的多语句事务,并将后续语句作为单语句事务执行,直到再次发出 BEGIN。
示例:BEGIN; -- 开始一个多语句事务
-- 这里的 DML 语句属于当前事务
INSERT INTO table_name (column1, column2) VALUES (value1, value2);
-- 在事务中执行 DDL 语句
CREATE TABLE new_table (column1 data_type, column2 data_type);
-- 这将提交当前事务
-- 后续语句将作为单语句事务执行
UPDATE table_name SET column1 = value WHERE condition;
BEGIN; -- 开始一个新的多语句事务
-- 这里的 DML 语句属于新事务
DELETE FROM table_name WHERE condition;
COMMIT; -- 结束新事务
语法
BEGIN [ TRANSACTION ]
事务 ID 和状态
Databend 会自动为每个事务生成一个事务 ID。该 ID 允许用户识别哪些语句属于同一事务,便于问题排查。
如果您在 Databend Cloud 上,可以在 Monitor > SQL History 中找到事务 ID:
在 Transaction 列中,您还可以看到 SQL 语句在执行期间的事务状态:
事务状态 | 描述 |
---|---|
AutoCommit | 该语句不属于多语句事务。 |
Active | 该语句属于多语句事务,并且事务中它之前的所有语句都成功执行。 |
Fail | 该语句属于多语句事务,并且事务中它之前的至少一个语句执行失败。 |
示例
在此示例中,所有三个语句(INSERT、UPDATE、DELETE)都属于同一个多语句事务。它们作为一个单元执行,并在发出 COMMIT 时一起提交更改。
-- 首先创建一个表
CREATE TABLE employees (
id INT,
name VARCHAR(50),
department VARCHAR(50)
);
-- 开始一个多语句事务
BEGIN;
-- 事务中的第一个语句:插入一个新员工
INSERT INTO employees (id, name, department) VALUES (1, 'Alice', 'HR');
-- 事务中的第二个语句:插入另一个新员工
INSERT INTO employees (id, name, department) VALUES (2, 'Bob', 'Engineering');
-- 事务中的第三个语句:更新第一个员工的部门
UPDATE employees SET department = 'Finance' WHERE id = 1;
-- 提交所有更改
COMMIT;
-- 验证表中的数据
SELECT * FROM employees;
┌───────────────────────────────────────────────────────┐
│ id │ name │ department │
├─────────────────┼──────────────────┼──────────────────┤
│ 1 │ Alice │ Finance │
│ 2 │ Bob │ Engineering │
└───────────────────────────────────────────────────────┘
在此示例中,ROLLBACK 语句撤销了事务期间所做的所有更改。因此,最后的 SELECT 查询应显示一个空的 employees 表,确认没有更改被提交。
-- 首先创建一个表
CREATE TABLE employees (
id INT,
name VARCHAR(50),
department VARCHAR(50)
);
-- 开始一个多语句事务
BEGIN;
-- 事务中的第一个语句:插入一个新员工
INSERT INTO employees (id, name, department) VALUES (1, 'Alice', 'HR');
-- 事务中的第二个语句:插入另一个新员工
INSERT INTO employees (id, name, department) VALUES (2, 'Bob', 'Engineering');
-- 事务中的第三个语句:更新第一个员工的部门
UPDATE employees SET department = 'Finance' WHERE id = 1;
-- 回滚事务
ROLLBACK;
-- 验证表是否为空
SELECT * FROM employees;
此示例设置了一个流和一个任务来消费该流,并使用事务块(BEGIN; COMMIT)将数据插入到两个目标表中。
CREATE DATABASE my_db;
USE my_db;
CREATE TABLE source_table (
id INT,
source_flag VARCHAR(50),value VARCHAR(50)
);
CREATE TABLE target_table_1 (
id INT,value VARCHAR(50)
);
CREATE TABLE target_table_2 (
id INT,value VARCHAR(50)
);
CREATE STREAM source_stream ON TABLE source_table;
INSERT INTO source_table VALUES
(1, 'source1', 'value1'),
(2, 'source2', 'value2'),
(3, 'source3', 'value3'),
(4, 'source4', 'value4');
CREATE TASK insert_task
WAREHOUSE = 'system'
SCHEDULE = 1 SECOND AS
BEGIN
BEGIN;
INSERT INTO my_db.target_table_1
SELECT id, value
FROM my_db.source_stream;
INSERT INTO my_db.target_table_2
SELECT id, value
FROM my_db.source_stream;
COMMIT;
END;
EXECUTE TASK insert_task;