开发者社区> 问答> 正文

如何在一个表中存储多个选项?

image.png

展开
收起
因为相信,所以看见。 2020-05-08 15:11:07 430 0
1 条回答
写回答
取消 提交回答
  • 阿里,我所有的向往

    请仔细阅读“ 数据规范化”,“ 常规索引”概念和“ 外键”约束,以确保数据干净,引用完整性。这会带你去。

    在纸上,将数据存储在数组中似乎很自然,但是对于db引擎,性能几乎没有索引的使用。此外,您将在第2天发现获取和维护数据是一场噩梦。

    以下内容将使您在开始修补时有个良好的开端。也加入。

    create table student ( studentId int auto_increment primary key, fullName varchar(100) not null -- etc );

    create table dept ( deptId int auto_increment primary key, deptName varchar(100) not null -- Economics -- etc );

    create table course ( courseId int auto_increment primary key, deptId int not null, courseName varchar(100) not null, -- etc CONSTRAINT fk_crs_dept FOREIGN KEY (deptId) REFERENCES dept(deptId) );

    create table SCJunction ( -- Student/Course Junction table (a.k.a Student is taking the course) -- also holds the attendance and grade id int auto_increment primary key, studentId int not null, courseId int not null, term int not null, -- term (I am using 100 in below examples for this term) attendance int not null, -- whatever you want, 100=always there, 0=he must have been partying, grade int not null, -- just an idea
    -- See (Note Composite Index) at bottom concerning next two lines. unique key(studentId,courseId,term), -- no duplicates allowed for the combo (note student can re-take it next term) key (courseId,studentId), CONSTRAINT fk_sc_student FOREIGN KEY (studentId) REFERENCES student(studentId), CONSTRAINT fk_sc_courses FOREIGN KEY (courseId) REFERENCES course(courseId) ); 创建测试数据 insert student(fullName) values ('Henry Carthage'),('Kim Billings'),('Shy Guy'); -- id's 1,2,3 insert student(fullName) values ('Shy Guy');

    insert dept(deptName) values ('History'),('Math'),('English'); -- id's 1,2,3

    insert course(deptId,courseName) values (1,'Early Roman Empire'),(1,'Italian Nation States'); -- id's 1 and 2 (History dept) insert course(deptId,courseName) values (2,'Calculus 1'),(2,'Linear Algebra A'); -- id's 3 and 4 (Math dept) insert course(deptId,courseName) values (3,'World of Chaucer'); -- id 5 (English dept)

    -- show why FK constraints are important based on data at the moment insert course(deptId,courseName) values (66,'Fly Fishing 101'); -- will generate error 1452. That dept 66 does not exist -- That error is a good error to have. Better than faulty data

    -- Have Kim (studentId=2) enrolled in a few courses insert SCJunction(studentId,courseId,term,attendance,grade) values (2,1,100,-1,-1); -- Early Roman Empire, term 100 (made up), unknown attendance/grade insert SCJunction(studentId,courseId,term,attendance,grade) values (2,4,100,-1,-1); -- Linear Algebra A insert SCJunction(studentId,courseId,term,attendance,grade) values (2,5,100,-1,-1); -- World of Chaucer

    -- Have Shy Guy (studentId=3) enrolled in one course only. He is shy insert SCJunction(studentId,courseId,term,attendance,grade) values (3,5,100,-1,-1); -- Early Roman Empire, term 100 (made up), unknow attendance/grade -- note if you run that line again, the Error 1062 Duplicate entry happens. Can't take same course more than once per term 一些简单的问题。

    什么部门在什么课程? 全部显示,使用表别名(缩写)来减少键入,提高可读性(有时)

    select c.courseId,c.courseName,d.deptId,d.deptName from course c join dept d on c.deptId=d.deptId order by d.deptName,c.courseName -- note the order +----------+-----------------------+--------+----------+ | courseId | courseName | deptId | deptName | +----------+-----------------------+--------+----------+ | 5 | World of Chaucer | 3 | English | | 1 | Early Roman Empire | 1 | History | | 2 | Italian Nation States | 1 | History | | 3 | Calculus 1 | 2 | Math | | 4 | Linear Algebra A | 2 | Math | +----------+-----------------------+--------+----------+ 这学期谁在上乔科世界课程? (知道courseId = 5)

    以下内容受益于我们在SCJunction中的综合指数之一。组合是一个以上列的索引。

    select s.StudentId,s.FullName from SCJunction j join student s on j.studentId=s.studentId where j.courseId=5 and j.term=100 +-----------+--------------+ | StudentId | FullName | +-----------+--------------+ | 2 | Kim Billings | | 3 | Shy Guy | +-----------+--------------+ 金比林斯(King Billings)入学这个词是什么? select s.StudentId,s.FullName,c.courseId,c.courseName from SCJunction j join student s on j.studentId=s.studentId join course c on j.courseId=c.courseId where s.studentId=2 and j.term=100 order by c.courseId DESC -- descending, just for the fun of it +-----------+--------------+----------+--------------------+ | StudentId | FullName | courseId | courseName | +-----------+--------------+----------+--------------------+ | 2 | Kim Billings | 5 | World of Chaucer | | 2 | Kim Billings | 4 | Linear Algebra A | | 2 | Kim Billings | 1 | Early Roman Empire | +-----------+--------------+----------+--------------------+ 金不堪重负,所以放弃数学课

    delete from SCJunction where studentId=2 and courseId=4 and term=100 在select语句上方运行该语句,以显示Kim正在采取的措施:

    +-----------+--------------+----------+--------------------+ | StudentId | FullName | courseId | courseName | +-----------+--------------+----------+--------------------+ | 2 | Kim Billings | 5 | World of Chaucer | | 2 | Kim Billings | 1 | Early Roman Empire | +-----------+--------------+----------+--------------------+ 嗯,这个词更容易理解。爸爸虽然不会高兴。

    注意诸如SCJunction.term之类的东西。关于这一点,可以写很多东西,我现在大部分时间会跳过它,只是说它也应该在FK中。您可能希望您的术语看起来更像是SPRING2015,而不是int。

    就ID而言。这就是我要做的方式。这是个人喜好。这将需要知道id#,然后查找它们。其他人可能会选择一个具有HIST101之类的courseId而不是17的那些。它们具有更高的可读性(但索引的速度(几乎没有))。因此,对您来说最好的方法是。

    票据综合指数 复合索引(INDEX表示KEY,反之亦然)是一种组合多列以进行快速数据检索的索引。翻转SCJunction表中两个组合的顺序,以便db引擎根据要追随数据的查询范围,可以根据要追寻的最左边的列选择用于快速检索的索引。

    至于唯一密钥#1,它旁边的注释表明不重复(意味着垃圾数据),这是不言自明的。例如,该表中不能存在两次学生1课程1术语1。

    要理解的一个关键概念是left-most索引中列名的顺序概念。

    对于studentId 仅进行的查询,则使用studentId列在第一(left-most)处的键。在courseId 仅执行的查询中,则courseId使用最左边的键。在studentId和courseId之后的查询中,数据库引擎可以决定要使用哪个组合键。

    当我说“去”时,我的意思是“ on clause或” where clause条件。

    如果不让这两个复合键(其中的第1列和第2列翻转),那么在查询中未对要查询的列建立left-most索引的情况下,您将不会受益于键的使用,并且将需要很慢的表扫描来返回数据。

    因此,这两个索引结合了以下两个概念

    基于最左边或最左边(studentId和courseId列)的快速数据检索 根据studentId,courseId和term值在该表中强制非重复数据

    2020-05-08 15:56:24
    赞同 展开评论 打赏
问答分类:
问答地址:
问答排行榜
最热
最新

相关电子书

更多
RowKey与索引设计:技巧与案例分析 立即下载
低代码开发师(初级)实战教程 立即下载
阿里巴巴DevOps 最佳实践手册 立即下载