MySQL表完整性約束

=======MySQL表完整性約束======

目錄:
    1、介紹
    2、not nulldefault
    3、unique
    4、primary key
    5、auto_increment
    6、foreign key


1、介紹
    約束條件與數據類型的寬度同樣,都是可選參數
    做用:用於保證數據的完整性和一致性

    主要分爲:
        PRIMARY KEY (PK) 標識該字段爲表的主鍵,能夠惟一的標識記錄
        FOREIGN KEY (FK) 標識該字段爲該表的外鍵
        NOT NULL  標識該字段不能爲空
        UNIQUE KEY (UK)  標識該字段是惟一值
        AUTO_INCREMENT 標識該字段的值自動增加(整數類型,並且爲主鍵)
        DEFAULT  爲該字段設置默認值

        UNSIGNED  無符號
        ZEROFILL  使用0填充


    說明:
        1. 是否容許爲空,默認NULL,可設置NOT NULL,字段不容許爲空,必須賦值
        2. 字段是否有默認值,缺省的默認值是NULL,若是插入記錄時不給字段賦值,此字段使用默認值
        sex enum('male','female') not null default 'male'
        age int unsigned NOT NULL default 20 必須爲正值(無符號) 不容許爲空 默認是20
        3. 是不是key
        主鍵 primary key
        外鍵 foreign key
        索引 (index,unique...)

2、not null與default

        是否爲空,null表示空,非字符串
        not null  -不可空
        null      -可空

        默認值,建立列時能夠指定默認值,當插入數據時若是未主動設置,則自動添加默認值
        create table tb1(
            nid int not null defalut 2,
            num int not null
        )

3、unique
    ============設置惟一約束 UNIQUE===============
        方法一:
        create table department1(
        id int,
        name varchar(20) unique,
        comment varchar(100)
        );


        方法二:
        create table department2(
            id int,
            name varchar(20),
            comment varchar(100),
            constraint uk_name unique(name)  #爲惟一索引發名字
        );

        注:惟一聯合
            create table service(
                id int primary key auto_increment,
                name varchar(20),
                host varchar(15) not null,
                port int not null,
                unique(host,port) #聯合惟一
            );


4、primary
    從約束角度看primary key字段的值不爲空且惟一,那咱們直接使用not null+unique不就能夠了嗎,要它幹什麼?
    主鍵primary key是innodb存儲引擎組織數據的依據,innodb稱之爲索引組織表,一張表中必須有且只有一個主鍵。

    一個表中能夠:
        單列作主鍵
        多列作主鍵(複合主鍵)

        - 單列作主鍵
            ============單列作主鍵===============
            #方法一:not null+unique
            create table department1(
            id int not null unique, #主鍵
            name varchar(20) not null unique,
            comment varchar(100)
            );

            mysql> desc department1;
            +---------+--------------+------+-----+---------+-------+
            | Field   | Type         | Null | Key | Default | Extra |
            +---------+--------------+------+-----+---------+-------+
            | id      | int(11)      | NO   | PRI | NULL    |       |
            | name    | varchar(20)  | NO   | UNI | NULL    |       |
            | comment | varchar(100) | YES  |     | NULL    |       |
            +---------+--------------+------+-----+---------+-------+
            rows in set (0.01 sec)

            #方法二:在某一個字段後用primary key
            create table department2(
            id int primary key, #主鍵
            name varchar(20),
            comment varchar(100)
            );

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

            #方法三:在全部字段後單獨定義primary key
            create table department3(
            id int,
            name varchar(20),
            comment varchar(100),
            constraint pk_name primary key(id); #建立主鍵併爲其命名pk_name

            mysql> desc department3;
            +---------+--------------+------+-----+---------+-------+
            | Field   | Type         | Null | Key | Default | Extra |
            +---------+--------------+------+-----+---------+-------+
            | id      | int(11)      | NO   | PRI | NULL    |       |
            | name    | varchar(20)  | YES  |     | NULL    |       |
            | comment | varchar(100) | YES  |     | NULL    |       |
            +---------+--------------+------+-----+---------+-------+
            rows in set (0.01 sec)

        - 多列作主鍵
            ==================多列作主鍵================
            create table service(
            ip varchar(15),
            port char(5),
            service_name varchar(10) not null,
            primary key(ip,port)
            );


            mysql> desc service;
            +--------------+-------------+------+-----+---------+-------+
            | Field        | Type        | Null | Key | Default | Extra |
            +--------------+-------------+------+-----+---------+-------+
            | ip           | varchar(15) | NO   | PRI | NULL    |       |
            | port         | char(5)     | NO   | PRI | NULL    |       |
            | service_name | varchar(10) | NO   |     | NULL    |       |
            +--------------+-------------+------+-----+---------+-------+
            3 rows in set (0.00 sec)

            mysql> insert into service values
                -> ('172.16.45.10','3306','mysqld'),
                -> ('172.16.45.11','3306','mariadb')
                -> ;
            Query OK, 2 rows affected (0.00 sec)
            Records: 2  Duplicates: 0  Warnings: 0

            mysql> insert into service values ('172.16.45.10','3306','nginx');
            ERROR 1062 (23000): Duplicate entry '172.16.45.10-3306' for key 'PRIMARY'


5、auto_increment
    - 約束字段爲自增加,被約束的字段必須同時被key約束
        - 不指定id,則自動增加
        - 也能夠指定id
        - 對於自增的字段,在用delete刪除後,再插入值,該字段仍按照刪除前的位置繼續增加
        - 應該用truncate清空表,比起delete一條一條地刪除記錄,truncate是直接清空表,在刪除大表時用它


    - 步長:auto_increment_increment,起始偏移量:auto_increment_offset
        #在建立完表後,修改自增字段的起始值
        mysql> create table student(
            -> id int primary key auto_increment,
            -> name varchar(20),
            -> sex enum('male','female') default 'male'
            -> );

        mysql> alter table student auto_increment=3;

        mysql> show create table student;
        .......
        ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8

        mysql> insert into student(name) values('jake');
        Query OK, 1 row affected (0.01 sec)

        mysql> select * from student;
        +----+------+------+
        | id | name | sex  |
        +----+------+------+
        |  3 | jake | male |
        +----+------+------+
        row in set (0.00 sec)

        mysql> show create table student;
        .......
        ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8


        #也能夠建立表時指定auto_increment的初始值,注意初始值的設置爲表選項,應該放到括號外
        create table student(
        id int primary key auto_increment,
        name varchar(20),
        sex enum('male','female') default 'male'
        )auto_increment=3;




        #設置步長
        sqlserver:自增步長
            基於表級別
            create table t1(
                id int。。。
            )engine=innodb,auto_increment=2 步長=2 default charset=utf8

        mysql自增的步長:
            show session variables like 'auto_inc%';
            
            #基於會話級別
            set session auth_increment_increment=2 #修改會話級別的步長

            #基於全局級別的
            set global auth_increment_increment=2 #修改全局級別的步長(全部會話都生效)


        #!!!注意了注意了注意了!!!
        If the value of auto_increment_offset is greater than that of auto_increment_increment, the value of auto_increment_offset is ignored. 
        翻譯:若是auto_increment_offset的值大於auto_increment_increment的值,則auto_increment_offset的值會被忽略 ,這至關於第一步步子就邁大了,扯着了蛋
        好比:設置auto_increment_offset=3,auto_increment_increment=2




        mysql> set global auto_increment_increment=5;
        Query OK, 0 rows affected (0.00 sec)

        mysql> set global auto_increment_offset=3;
        Query OK, 0 rows affected (0.00 sec)

        mysql> show variables like 'auto_incre%'; #須要退出從新登陸
        +--------------------------+-------+
        | Variable_name            | Value |
        +--------------------------+-------+
        | auto_increment_increment | 1     |
        | auto_increment_offset    | 1     |
        +--------------------------+-------+



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

        mysql> insert into student(name) values('jake1'),('jake2'),('jake3');
        mysql> select * from student;
        +----+-------+------+
        | id | name  | sex  |
        +----+-------+------+
        |  3 | jake1 | male |
        |  8 | jake2 | male |
        | 13 | jake3 | male |
        +----+-------+------+

6、foreign key
    - 快速理解foreign key
        員工信息表有三個字段:工號  姓名  部門
        公司有3個部門,可是有1個億的員工,那意味着部門這個字段須要重複存儲,部門名字越長,越浪費

        解決方法:
            咱們徹底能夠定義一個部門表
            而後讓員工信息表關聯該表,如何關聯,即foreign key

        #表類型必須是innodb存儲引擎,且被關聯的字段,即references指定的另一個表的字段,必須保證惟一
        create table department(
        id int primary key,
        name varchar(20) not null
        )engine=innodb;

        #dpt_id外鍵,關聯父表(department主鍵id),同步更新,同步刪除
        create table employee(
        id int primary key,
        name varchar(20) not null,
        dpt_id int,
        constraint fk_name foreign key(dpt_id)
        references department(id)
        on delete cascade
        on update cascade 
        )engine=innodb;

        #先往父表department中插入記錄
        #再往子表employee中插入記錄
        #刪父表department,子表employee中對應的記錄跟着刪
        #更新父表department,子表employee中對應的記錄跟着改
    - 如何找出兩張表中的關係
        分析步驟:
        #1、先站在左表的角度去找
        是否左表的多條記錄能夠對應右表的一條記錄,若是是,則證實左表的一個字段foreign key 右表一個字段(一般是id)

        #2、再站在右表的角度去找
        是否右表的多條記錄能夠對應左表的一條記錄,若是是,則證實右表的一個字段foreign key 左表一個字段(一般是id)

        #3、總結:
        #多對一:
        若是隻有步驟1成立,則是左表多對一右表
        若是隻有步驟2成立,則是右表多對一左表

        #多對多
        若是步驟1和2同時成立,則證實這兩張表時一個雙向的多對一,即多對多,須要定義一個這兩張表的關係表來專門存放兩者的關係

        #一對一:
        若是1和2都不成立,而是左表的一條記錄惟一對應右表的一條記錄,反之亦然。這種狀況很簡單,就是在左表foreign key右表的基礎上,將左表的外鍵字段設置成unique便可

    - 創建表之間的關係
        #一對多或稱爲多對一
        三張表:出版社,做者信息,書

        一對多(或多對一):一個出版社能夠出版多本書

        關聯方式:foreign key

            =====================多對一=====================
            create table press(
            id int primary key auto_increment,
            name varchar(20)
            );

            create table book(
            id int primary key auto_increment,
            name varchar(20),
            press_id int not null,
            foreign key(press_id) references press(id)
            on delete cascade
            on update cascade
            );



        #多對多
        三張表:出版社,做者信息,書

        多對多:一個做者能夠寫多本書,一本書也能夠有多個做者,雙向的一對多,即多對多
          
        關聯方式:foreign key+一張新的表

            =====================多對多=====================
            create table author(
            id int primary key auto_increment,
            name varchar(20)
            );


            #這張表就存放做者表與書表的關係,即查詢兩者的關係查這表就能夠了
            create table author2book(
            id int not null unique auto_increment,
            author_id int not null,
            book_id int not null,
            constraint fk_author foreign key(author_id) references author(id)
            on delete cascade
            on update cascade,
            constraint fk_book foreign key(book_id) references book(id)
            on delete cascade
            on update cascade,
            primary key(author_id,book_id)
            );



        #一對一
        兩張表:學生表和客戶表

        一對一:一個學生是一個客戶,一個客戶有可能變成一個學校,即一對一的關係

        關聯方式:foreign key+unique

            =====================一對一=====================
            #必定是student來foreign key表customer,這樣就保證了:
            #1 學生必定是一個客戶,
            #2 客戶不必定是學生,但有可能成爲一個學生


            create table customer(
            id int primary key auto_increment,
            name varchar(20) not null,
            qq varchar(10) not null,
            phone char(16) not null
            );


            create table student(
            id int primary key auto_increment,
            class_name varchar(20) not null,
            customer_id int unique, #該字段必定要是惟一的
            foreign key(customer_id) references customer(id) #外鍵的字段必定要保證unique
            on delete cascade
            on update cascade
            );
相關文章
相關標籤/搜索