必须显式赋值或完全不提该列以触发DEFAULT约束;仅设DEFAULT不设NOT NULL时NULL仍合法,设NOT NULL+DEFAULT后若INSERT显式赋NULL则报错,完全不提该列才生效。
报错“Cannot insert the value NULL into column 'xxx'”时,不是数据库“不讲道理”,而是你没按规则提供值——要么显式赋值,要么让 DEFAULT 约束真正生效。
很多人写 INSERT INTO Users (id, name) VALUES (1, 'Alice'),却指望 email 列(定义为 NOT NULL DEFAULT '[email protected]')自动填上默认值。实际不会生效:只要你在列名列表里没提它,SQL Server 才会触发 DEFAULT;但如果表结构里只加了 DEFAULT 没加 NOT NULL,哪怕省略该列,插入的仍是 NULL,照样报错。
NOT NULL + DEFAULT
INSERT INTO Users (id, name, email) VALUES (1, 'Alice', NULL) —— 显式写 NULL 直接炸DEFAULT 值必须是常量或确定性表达式,比如 GETDATE() 可以,GETDATE() + 1 不行给已有数据的表加新列并设为 NOT NULL,SQL Server 会直接拒绝,除非你同步指定 DEFAULT。
ALTER TABLE Users ADD phone VARCHAR(20) NOT NULL DEFAULT 'N/A'
ALTER TABLE Users ADD phone VARCHAR(20) NOT NULL —— 错误提示明确说 “only allows columns that can contain nulls, or have a DEFAULT definition”SELECT name FROM sys.default_constraints WHERE parent_object_id = OBJECT_ID('Users'),再用 ALTER TABLE Users DROP CONSTRAINT [DF_Users_phone]
模型里写了 id = Column(INTEGER, primary_key=True, autoincrement=True),但建表 SQL 是手写的 id INTEGER NOT NULL,没用 SERIAL 或 GENERATED BY DEFAULT AS IDENTITY,就会导致 ORM 省略 id 字段插入,而数据库把它当 NULL 处理。
id SERIAL NOT NULL PRIMARY KEY(或 INTEGER GENERATED BY DEFAULT AS IDENTITY)Base.metadata.create_all() 自动建表,避免模型和 DDL 脱节[SugarColumn(IsNullable = true)],或字段是值类型(如 int)却没设默认值错误信息里已经说了,别跳过它:
Cannot insert the value NULL into column 'status', table 'Orders'; column does not allow nulls. → 直接锁定 status 列SELECT COLUMN_NAME, IS_NULLABLE, COLUMN_DEFAULT FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = 'Orders' AND COLUMN_NAME = 'status'
最容易被忽略的是:DEFAULT 约束只在“完全不提该列”时才起作用,而不是“不给值”——后者在多数方言里等价于显式 NULL。别依赖直觉,查执行计划或用 SET STATISTICS XML ON 看 SQL Server 实际生成的 INSERT 语句长什么样。