INSERT INTO 语句的基本写法
MySQL 插入单条数据最常用的是
INSERT INTO语句,核心在于字段名和值的严格对应。不指定字段名时,必须按表定义顺序提供所有非空(且无默认值)字段的值,极易出错。 推荐始终显式列出字段名:避免因表结构变更导致插入失败或字段错位 字符串值必须用单引号
' '包裹,数字不用引号,
NULL直接写(不加引号) 如果某字段有默认值或允许
NULL,可跳过该字段(前提是显式写了字段列表)
示例:
INSERT INTO users (name, age, email) VALUES ('张三', 28, 'zhang@example.com');
插入时处理主键冲突:ON DUPLICATE KEY UPDATE
当表有唯一索引或主键,且想“存在则更新、不存在则插入”时,
INSERT ... ON DUPLICATE KEY UPDATE比先查再判再插更安全高效。 只对触发唯一约束(
PRIMARY KEY或
UNIQUE索引)的列生效
UPDATE后面只能写字段赋值,不能用子查询(MySQL 8.0.19+ 支持部分限制下的子查询,但慎用) 注意:
LAST_INSERT_ID()在冲突时返回原记录 ID,不是新生成的 ID
示例:
INSERT INTO configs (key_name, value) VALUES ('theme', 'dark') ON DUPLICATE KEY UPDATE value = VALUES(value);
插入 NULL、当前时间、函数值的注意事项
插入特殊值时容易忽略隐式类型转换或默认行为,导致数据不符合预期。
NULL表示空值,不是字符串
'NULL';若字段定义为
NOT NULL且无默认值,插入
NULL会报错
ERROR 1048 (23000): Column 'xxx' cannot be null插入当前时间优先用
NOW()或
CURRENT_TIMESTAMP,不要拼字符串(如
'2024-05-20 10:30:00'),否则时区/格式易出问题 自增主键字段通常不写,让 MySQL 自动赋值;若强制指定,需确保不违反唯一性和递增逻辑
示例:
INSERT INTO logs (message, created_at) VALUES ('system start', NOW());
批量插入一条?别被名字骗了
INSERT INTO ... VALUES (...), (...)看似是“批量”,但哪怕只写一组括号,语法上仍是标准单行插入。真正影响性能和事务行为的是是否合并多行 —— 单条语句插入多行比多条
INSERT语句快得多,也更节省网络往返。 即使只插一条,也建议保持和批量一致的括号格式,方便后续扩展 注意:每组
VALUES中的字段数、类型必须与前面字段列表完全一致 大数量插入时,建议分批(如 1000 行/批),避免单语句过长触发
max_allowed_packet限制
示例(单条但带括号):
INSERT INTO tags (name, category) VALUES ('mysql', 'database');
字段名拼错、引号漏写、值类型和列定义不匹配——这些错误在执行瞬间就暴露,但真正难排查的是隐式截断(比如
VARCHAR(10)插入 15 个字符)、时区导致的时间偏移、以及
ON DUPLICATE KEY UPDATE中误用了
VALUES(col)却没意识到它取的是本次 INSERT 的值而非原值。
