CHECK约束是SQL中用于限制列中数据值的完整性约束。它允许在插入或更新数据时验证数据的有效性,确保只有满足指定条件的数据才能进入数据库。
-- 列级约束
CREATE TABLE Employees (
EmployeeID INT PRIMARY KEY,
EmployeeName VARCHAR(100),
Age INT CHECK (Age >= 18 AND Age <= 65),
Salary DECIMAL(10,2) CHECK (Salary > 0),
Email VARCHAR(100) CHECK (Email LIKE '%@%.%')
);
-- 表级约束(多列检查)
CREATE TABLE Orders (
OrderID INT PRIMARY KEY,
OrderDate DATE,
ShipDate DATE,
TotalAmount DECIMAL(10,2),
CHECK (ShipDate >= OrderDate),
CHECK (TotalAmount >= 0)
);
-- 添加单个约束
ALTER TABLE Employees
ADD CONSTRAINT CHK_Age_Range
CHECK (Age >= 18 AND Age <= 65);
-- 添加多列约束
ALTER TABLE Orders
ADD CONSTRAINT CHK_Date_Logic
CHECK (ShipDate >= OrderDate AND CancelDate >= OrderDate);
ALTER TABLE Employees
DROP CONSTRAINT CHK_Age_Range;
-- 年龄限制
CHECK (Age BETWEEN 0 AND 120)
-- 数值范围
CHECK (Quantity >= 0 AND Quantity <= 1000)
-- 日期范围
CHECK (OrderDate >= '2020-01-01' AND OrderDate <= GETDATE())
-- 邮箱格式
CHECK (Email LIKE '%_@__%.__%')
-- 电话号码格式(示例)
CHECK (PhoneNumber LIKE '+[1-9]%' OR PhoneNumber LIKE '[0-9]%')
-- 邮政编码
CHECK (PostalCode LIKE '[0-9][0-9][0-9][0-9][0-9]')
-- 折扣与价格关系
CHECK (DiscountAmount <= OriginalPrice)
-- 开始与结束时间
CHECK (EndDate > StartDate)
-- 库存逻辑
CHECK (QuantityInStock >= ReorderLevel)
-- 状态值检查
CHECK (Status IN ('Active', 'Inactive', 'Pending', 'Cancelled'))
-- 性别检查
CHECK (Gender IN ('M', 'F', 'Other'))
-- 优先级检查
CHECK (Priority IN ('High', 'Medium', 'Low'))
-- 根据会员等级设置不同的折扣上限
CHECK (
(MembershipLevel = 'Basic' AND Discount <= 0.10) OR
(MembershipLevel = 'Premium' AND Discount <= 0.20) OR
(MembershipLevel = 'VIP' AND Discount <= 0.30)
)
-- 工作时间验证
CHECK (
(EXTRACT(HOUR FROM ShiftStart) >= 8 AND
EXTRACT(HOUR FROM ShiftEnd) <= 20) OR
ShiftType = 'Night'
)
-- MySQL示例:使用函数验证
CHECK (LENGTH(Password) >= 8 AND
Password REGEXP '[A-Z]' AND
Password REGEXP '[a-z]' AND
Password REGEXP '[0-9]')
-- SQL Server示例:日期计算
CHECK (DATEDIFF(year, BirthDate, GETDATE()) >= 18)
-- PostgreSQL示例:正则表达式
CHECK (Email ~ '^[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Za-z]{2,}$')
-- 使用有意义的约束名
ALTER TABLE Products
ADD CONSTRAINT CHK_Products_Price_Positive
CHECK (Price > 0);
-- 检查约束状态
SELECT * FROM INFORMATION_SCHEMA.CHECK_CONSTRAINTS;
-- 临时禁用约束(某些数据库支持)
ALTER TABLE Employees NOCHECK CONSTRAINT CHK_Age_Range;
-- 重新启用
ALTER TABLE Employees CHECK CONSTRAINT CHK_Age_Range;
| 特性 | CHECK约束 | 触发器 |
|---|---|---|
| 执行时机 | 语句执行前 | 语句执行后 |
| 性能 | 通常更快 | 相对较慢 |
| 复杂度 | 简单条件 | 复杂逻辑 |
| 错误处理 | 立即回滚 | 可灵活处理 |
| 多表验证 | 不支持 | 支持 |
CREATE TABLE Products (
ProductID INT PRIMARY KEY,
ProductName VARCHAR(255) NOT NULL,
Price DECIMAL(10,2) CHECK (Price > 0),
StockQuantity INT CHECK (StockQuantity >= 0),
DiscountPercent DECIMAL(5,2) CHECK (DiscountPercent BETWEEN 0 AND 100),
Category VARCHAR(50) CHECK (Category IN ('Electronics', 'Clothing', 'Books', 'Home')),
CONSTRAINT CHK_Price_Discount CHECK (Price * (1 - DiscountPercent/100) > CostPrice)
);
CREATE TABLE Orders (
OrderID INT PRIMARY KEY,
OrderDate DATE DEFAULT CURRENT_DATE,
ShipDate DATE,
TotalAmount DECIMAL(10,2) CHECK (TotalAmount >= 0),
Status VARCHAR(20) CHECK (Status IN ('Pending', 'Processing', 'Shipped', 'Delivered', 'Cancelled')),
CHECK (ShipDate >= OrderDate),
CHECK (Status != 'Shipped' OR ShipDate IS NOT NULL)
);
CHECK约束的核心价值:
数据质量:确保存储在数据库中的数据符合业务规则 简化应用:减少应用程序中的验证代码 性能优化:数据库级验证效率更高 一致性:所有数据操作都遵循相同规则使用建议:
CHECK约束是SQL数据库设计中确保数据完整性的重要工具,合理使用可以大幅提升数据质量并简化应用程序开发。