四,primary key

关系型数据库中的一条记录中有若干个属性,若其中某一个属性组(注意是组)能唯一标识一条记录,该属性组就可以成为一个主键。

例如

学生表(学号,姓名,性别,班级)    其中每个学生的学号是唯一的,学号就是一个主键,这叫单列主键    

课程表(课程编号,课程名,学分)    其中课程编号是唯一的,课程编号就是一个主键,这叫单列主键成绩    

表(学号,课程号,    成绩)成绩表中唯一一个属性无法唯一标识一条记录,学号和课程号的组合才可以唯一标识一条记录,所以学号和课程号的属性组是一个主键,这叫多列主键

成绩表中的学号不是成绩表的主键,但它和学生表中的学号相对应,并且学生表
中的学号是学生表的主键,则称成绩表中的学号是学生表的外键。

同理成绩表中的课程号是课程表的外键 primary key关键字用于定义列为主键。

单列主键

在某一个字段后用primary key

 create table t6(
     id int primary key,
     name varchar(20),
     number char(10));

 desc t6;

执行结果为:

mysql>  create table t6(     id int primary key,     name varchar(20),     number char(10));
Query OK, 0 rows affected (0.53 sec)

mysql> desc t6;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| id     | int(11)     | NO   | PRI | NULL    |       |
| name   | varchar(20) | YES  |     | NULL    |       |
| number | char(10)    | YES  |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
3 rows in set (0.00 sec)

在所有前端后单独定义主键

create table t7 (
	id int,
	name varchar20),
	number char10),
	constraint pk_id primary key(id)); #创建主键并为此命名

desc t7; 

执行结果为:

mysql> create table t7(
    -> id int,
    -> name varchar(20),
    -> number varchar(20),
    -> constraint pk_id primary key(id));
Query OK, 0 rows affected (0.68 sec)

mysql> desc t7;
+--------+-------------+------+-----+---------+-------+
| Field  | Type        | Null | Key | Default | Extra |
+--------+-------------+------+-----+---------+-------+
| id     | int(11)     | NO   | PRI | NULL    |       |
| name   | varchar(20) | YES  |     | NULL    |       |
| number | varchar(20) | YES  |     | NULL    |       |
+--------+-------------+------+-----+---------+-------+
3 rows in set (0.00 sec)

多列主键

mysql> create table t8(
     -> student_number char(10),
     -> class_number char(10),
     -> achievement int not null default 0,
     -> primary key(student_number, class_number));
 Query OK, 0 rows affected (0.73 sec)

 mysql> desc t8;
 +----------------+----------+------+-----+---------+-------+
 | Field          | Type     | Null | Key | Default | Extra |
 +----------------+----------+------+-----+---------+-------+
 | student_number | char(10) | NO   | PRI | NULL    |       |
 | class_number   | char(10) | NO   | PRI | NULL    |       |
 | achievement    | int(11)  | NO   |     | 0       |       |
 +----------------+----------+------+-----+---------+-------+
 3 rows in set (0.00 sec)

 mysql> insert into t8 values
     -> ("18023300", "186000", 100),
     -> ("19205200", "192000", 90);
 Query OK, 2 rows affected (0.18 sec)
 Records: 2  Duplicates: 0  Warnings: 0

 mysql> insert into t8 values("18023300", "186000", 90);
 ERROR 1062 (23000): Duplicate entry '18023300-186000' for key 'PRIMARY'

 mysql> select * from t8;
 +----------------+--------------+-------------+
 | student_number | class_number | achievement |
 +----------------+--------------+-------------+
 | 18023300       | 186000       |         100 |
 | 19205200       | 192000       |          90 |
 +----------------+--------------+-------------+
 2 rows in set (0.00 sec)

auto_increment

我们每次往表中插入数据的时候都得记住id,这样非常考验脑力,有的时候我们可能记得不是很清楚,那就容易报错,使用auto_increment约束字段会自动增长,被约束的字段必须同时被key约束。

 create table t9(
     id int primary key auto_increment,
     name varchar(20),
     sex enum("male", "female"));

 desc t9;

 insert into t9(name) values("Alex"),("Coco");
 select * from t9;

 insert into t9 values(5, "BeiBei", "female");
 select * from t9;

 delete from t9;
 select * from t9;

 insert into t9(name, sex) values("Alex", "male"),("Coco", "female");
 select * from t9;

 truncate t9;
 insert into t9(name, sex) values("Alex", "male"),("Coco", "female");
 select * from t9;

执行结果为:

mysql> create table t9(
    -> id int primary key auto_increment,    # 定义自增长字段
    -> name varchar(20),
    -> sex enum("male", "female"));
Query OK, 0 rows affected (1.60 sec)

mysql> desc t9;
+-------+-----------------------+------+-----+---------+----------------+
| Field | Type                  | Null | Key | Default | Extra          |
+-------+-----------------------+------+-----+---------+----------------+
| id    | int(11)               | NO   | PRI | NULL    | auto_increment |
| name  | varchar(20)           | YES  |     | NULL    |                |
| sex   | enum('male','female') | YES  |     | NULL    |                |
+-------+-----------------------+------+-----+---------+----------------+
3 rows in set (0.05 sec)

mysql> insert into t9(name) values("Alex"),("Coco");
Query OK, 2 rows affected (0.31 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> select * from t9;    # 不指定id,则自动增长
+----+------+------+
| id | name | sex  |
+----+------+------+
|  1 | Alex | NULL |
|  2 | Coco | NULL |
+----+------+------+
2 rows in set (0.00 sec)

mysql> insert into t9 values(5, "BeiBei", "female");    # 也可指指定id
Query OK, 1 row affected (0.23 sec)

mysql> select * from t9;
+----+--------+--------+
| id | name   | sex    |
+----+--------+--------+
|  1 | Alex   | NULL   |
|  2 | Coco   | NULL   |
|  5 | BeiBei | female |
+----+--------+--------+
3 rows in set (0.00 sec)

mysql>  delete from t9;
Query OK, 3 rows affected (0.52 sec)

mysql> select * from t9;
Empty set (0.00 sec)

mysql> insert into t9(name, sex) values("Alex", "male"),("Coco", "female");
Query OK, 2 rows affected (0.18 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> select * from t9;    # 对于自增的字段,在用delete删除后,再插入值,该字段仍按照删除前的位置继续增长
+----+------+--------+
| id | name | sex    |
+----+------+--------+
|  6 | Alex | male   |
|  7 | Coco | female |
+----+------+--------+
2 rows in set (0.00 sec)

mysql> truncate t9;        # 应该用truncate清空表,比起delete一条一条地删除记录,truncate是直接清空表,在删除大表时用它
Query OK, 0 rows affected (0.87 sec)

mysql> insert into t9(name, sex) values("Alex", "male"),("Coco", "female");
Query OK, 2 rows affected (0.09 sec)
Records: 2  Duplicates: 0  Warnings: 0

mysql> select * from t9;
+----+------+--------+
| id | name | sex    |
+----+------+--------+
|  1 | Alex | male   |
|  2 | Coco | female |
+----+------+--------+
2 rows in set (0.00 sec)

创建完表后可以修改自增细分的初始值

create table t10(
	id int primary key auto_increment,
	name varchar(20),
	sex enum('male','female') default 'male');

alter table t10 auto_increment = 5;		# 修改步长

show create table t10;

insert into t10(name) values("Alex");

select * from t10;

show create table t10;

执行结果为:

mysql> create table t10(
    -> id int primary key auto_increment,
    -> name varchar(20),
    -> sex enum('male','female') default 'male');
Query OK, 0 rows affected (0.72 sec)

mysql> alter table t10 auto_increment = 5;
Query OK, 0 rows affected (0.33 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> show create table t10;
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table                                                                                                                                                                                                             |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| t10   | CREATE TABLE `t10` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(20) DEFAULT NULL,
  `sex` enum('male','female') DEFAULT 'male',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8 |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

mysql> insert into t10(name) values("Alex");
Query OK, 1 row affected (0.24 sec)

mysql> select * from t10;
+----+------+------+
| id | name | sex  |
+----+------+------+
|  5 | Alex | male |
+----+------+------+
1 row in set (0.00 sec)

mysql> show create table t10;
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table | Create Table                                                                                                                                                                                                             |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| t10   | CREATE TABLE `t10` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(20) DEFAULT NULL,
  `sex` enum('male','female') DEFAULT 'male',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8 |
+-------+--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

查看MySQL的步长:

show session variables like "auto_inc%";

执行结果为:

mysql> show session variables like "auto_inc%";
+--------------------------+-------+
| Variable_name            | Value |
+--------------------------+-------+
| auto_increment_increment | 1     |	# auto_increment_increment 表示步长
| auto_increment_offset    | 1     |	# auto_increment_offset    表示其实偏移量
+--------------------------+-------+
2 rows in set, 1 warning (0.15 sec)

可以通过下列命令设置步长:

set session auth_increment_increment=2 #修改会话级别的步长

set global auth_increment_increment=2 #修改全局级别的步长(所有会话都生效)

设置全局的起始偏移量和步长需要重新启动MySQL。

如果auto_increment_offset的值大于auto_increment_increment的值,则auto_increment_offset的值会被忽略。