mysql插入数据应显式指定字段名,字符串用单引号,null不加引号,可跳过有默认值或允许null的字段;主键冲突用on duplicate key update;插入时间用now();批量插入建议分批以避免max_allowed_packet限制。

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 的值而非原值。










