MySQL连表操作之⼀对多
引⼊
当我们在数据库中创建表的时候,有可能某些列中值内容量很⼤,⽽且重复。
例⼦:创建⼀个学⽣表,按学校年纪班级分,表的内容⼤致如下:
id name partment
1xxx x学校x年级x班级
2ooo x学校x年级x班级
3zzz z学校x年级x班级
4ddd y学校x年级x班级
我们看出来对应的partment对应的值很长,⽽且重复量很⼤,这样就很不合适。
因此我们考虑将复杂重复的部分单独拿出来分成2个表:
第⼀张表:
id caption
1x学校x年级x班级
2y学校x年级x班级
3z学校x年级x班级
第⼆张表在之前的基础上修改的:
id name partment
1xxx1
2ooo2
3zzz3
4ddd1
这样看起来就很简洁了,我们将很长的且重复的部分拿出来,然后规定编号,创建学⽣表的时候,学⽣对应的partment只需要取学校对应的id 便可,这样同时这2张表也就会关联起来。
说明:
1、他们的关联关系;表2中的partment和表1的id联系再⼀起。
2、表⼀的数据会对应表2中的多条数据,这就叫⼀对多。
问题:此时,两张表是相对独⽴的,都可以各⾃插⼊⾃⼰的数据,这样做很合适的,并没实质⾏的关联?
因此,必须要将其限制。表⼆的partment数据必须是表⼀中有的,不然,就不让其增加。
这⾥(partment)就引⼊了⼀个名词 —- 外键
外键
说明:
1、外键:⼀个表接收另⼀个表的主键。
2、partment外键的是表⼀中的nid。
3、当我们创建了外键,则系统变默认会为我们添加,相应的约束,如:表⼆中的partment数据必须是表⼀中nid有的;表⼆和表⼀就关联起来了。
Navicat创建外键
创建part表:
创建person表:
外键的创建注意点:
创建外键时,互相对应的表中的数据类型必须⼀样。
创建外键
1、⾸先在要创建外键的表上右键,选设计表。
2、进⼊设计表,在右边显⽰表⾏信息,然后点击外键:
3、选择字段,会出现⼀个选择框,选择你要设置外键的列;
4、选择参考表,选择要外键要关联的表,
说明:
参数数据库:因为两个数据库在同意⽬录下,所以这⾥可以不⽤选择;默认是同⽬录下的。
5、选择参考字段;选择参考表中要设为外键的列;
注意:
外键的创建,连个表中关联的列的数据类型必须⼀样,不然不能成功。
6、保存CTRL + S
使⽤外键
此时part中没有数据,如果此时你在person中添加数据,结果:
在part中添加数据,结果:
此时再在,person中添加数据:
注意:
这⾥不能输⼊part中nid没有的值,不然也会报错。
SQL命令创建外键
创建part:
part(
nid    auto_increment  ,
caption (32)
)
part
创建person:
person(
nid    auto_increment,
name (32)  ,
email (32)  ,
part_nid ,
(nid),
fk_person_part  (part_nid)  part(nid)
)
person
创建完之后外键对应代码的关系:
分析:
1、从名字可以看出代码对应的是什么位置的。
2、图中名(C对应代码中的CONSTRAINT)这⾏可以不⽤设,系统会默认帮我设置,但是最好设置,如果要删除外键的时候,就可以通过这个名字进⾏相应
的操作。
代码删除外键
alter table person1 drop foreign key fk_person1_part1;
代码增加外键
sql left join 多表连接
alter table person1 add constraint fk_p1_t1 foreign key (part1_nid) references part1(nid);
通过外键进⾏数据操作
part表:
person表:
需求:要出person表中属于x学校的⼈?
1、之前学的办法:
1、先去part中获取x对应的nid
2、然后再通过这个nid与parson中part_id对⽤的关系,查出对应的name
select name from person where part_id in (select nid form part where caption = "x")
2、链表⽅法 left join
left join
使⽤连表提供的⽅法,left join操作代码:
select * from person left join part on person.part_id = part.nid
结果显⽰:
分析:
left join:相当于将part表匹配的部分直接移动到person的列后⾯,组合起来显⽰。
既然内容都合并了,那么此时我们再加上判断,就可以将要的数据获取了。
代码1:
select * from person left join part on person.part_id = part.nid where part.caption = "x"
结果:
说明:
之前我们⽤的*获取的是全部的信息,如果我们获取指定的信息,可以将其修改。
如:只获取person的name代码:
select person.name from person left join part on person.part_id = part.nid where part.caption = "x"结果:
注意:
join连接的条件,使⽤ on 进⾏对接的,条件写在on后⾯。
A left join
B = b.xx
left join的特点:
1、以A为主
2、将A中的所有数据罗列
3、B则只显⽰与A相对应的数据
问题:验证我们说的谁在前就谁为主,谁的数据就全部显⽰,我们将person和part换个位置?
select * from part left join person on person.part_id = part.nid;
结果:
right join
在谁显⽰所有数据的上来看,他和left join刚好相反,以后⾯的表为主,显⽰其所有的数据。
inner join
会将没有建⽴关系的数据忽略掉。不管谁在前,结果都是⼀样。
select * from person inner join part on person.part_id = part.nid;
结果:
总结:
1、这⼏个join可以写多个的,意思就是⼀个表可以同时有多个外键。
2、当选择的列名,是所有表中唯⼀的话,可以不⽤写前缀的表名。如:person.part_id就可以直接写成part_id.
3、上⾯的part表,有个别名叫,字典表

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。