2408. 设计 SQL 🔒
题目描述
给定两个字符串数组 names 和 columns,大小都为 n。其中 names[i] 是第 i 个表的名称,columns[i] 是第 i 个表的列数。
您需要实现一个支持以下 操作 的类:
- 在特定的表中 插入 一行。插入的每一行都有一个 id。id 是使用自动递增方法分配的,其中第一个插入行的 id 为 1,同一个表中的后续其他行的 id 为上一个插入行的 id (即使它已被删除) 加 1。
 - 从指定表中 删除 一行。注意,删除一行 不会 影响下一个插入行的 id。
 - 从任何表中 查询 一个特定的单元格并返回其值。
 - 从任何表以 csv 格式 导出 所有行。
 
实现 SQL 类:
SQL(String[] names, int[] columns)- 创建 
n个表。 
- 创建 
 bool ins(String name, String[] row)- 将 
row插入表name中并返回true。 - 如果 
row.length不 匹配列的预期数量,或者name不是 一个合法的表,不进行任何插入并返回false。 
- 将 
 void rmv(String name, int rowId, int columnId)- 从表 
name中移除行rowId。 - 如果 
name不是 一个合法的表或者没有 id 为rowId的行,不进行删除。 
- 从表 
 String sel(String name, int rowId, int columnId)- 返回表 
name中位于特定的rowId和columnId的单元格的值。 - 如果 name 不是 一个合法的表,或者单元格 
(rowId, columnId)不合法,返回"<null>"。 
- 返回表 
 String[] exp(String name)- 返回表 
name中出现的行。 - 如果 
name不是 一个合法的表,返回一个空数组。每一行以字符串表示,每个单元格的值(包括 行的 id)以","分隔。 
- 返回表 
 
示例 1:
输入:
["SQL","ins","sel","ins","exp","rmv","sel","exp"] [[["one","two","three"],[2,3,1]],["two",["first","second","third"]],["two",1,3],["two",["fourth","fifth","sixth"]],["two"],["two",1],["two",2,2],["two"]]
输出:
[null,true,"third",true,["1,first,second,third","2,fourth,fifth,sixth"],null,"fifth",["2,fourth,fifth,sixth"]]
解释:
// 创建 3 张表。
SQL sql = new SQL(["one", "two", "three"], [2, 3, 1]);
// 将 id 为 1 的行添加到表 "two"。返回 True。
sql.ins("two", ["first", "second", "third"]);
// 从表 "two" 中 id 为 1 的行 
// 其中第 3 列返回值 "third"。
sql.sel("two", 1, 3);
// 将另外一个 id 为 2 的行添加到表 "two"。返回 True。
sql.ins("two", ["fourth", "fifth", "sixth"]);
// 导出表 "two" 的行。
// 目前表中有两行 id 为 1 和 2 。
sql.exp("two");
// 删除表 "two" 当中的第一行。注意第二行的 id
// 依然为 2。
sql.rmv("two", 1);
// 从表 "two" 中 id 为 2 的行
// 其中第 2 列返回值 "fifth"。
sql.sel("two", 2, 2);
// 导出表 "two" 的行。
// 目前表中有一行 id 为 2。
sql.exp("two");
示例 2:
输入:
["SQL","ins","sel","ins","exp","rmv","sel","exp"] [[["one","two","three"],[2,3,1]],["two",["first","second","third"]],["two",1,3],["two",["fourth","fifth","sixth"]],["two"],["two",1],["two",2,2],["two"]]
输出:
[null,true,"third",true,["1,first,second,third","2,fourth,fifth,sixth"],null,"fifth",["2,fourth,fifth,sixth"]]
解释:
// 创建 3 张表
SQL sQL = new SQL(["one", "two", "three"], [2, 3, 1]); 
// 将 id 为 1 的行添加到表 "two"。返回 True。
sQL.ins("two", ["first", "second", "third"]); 
// 从表 "two" 中 id 为 1 的行
// 其中第 3 列返回值 "third"。
sQL.sel("two", 1, 3); 
// 删除表 "two" 的第一行。
sQL.rmv("two", 1); 
// 返回 "<null>" 因为 id 为 1 的单元格
// 已经从表 "two" 中删除。
sQL.sel("two", 1, 2); 
// 返回 False 因为列的数量不正确。
sQL.ins("two", ["fourth", "fifth"]); 
// 将 id 为 2 的行添加到表 "two"。返回 True。
sQL.ins("two", ["fourth", "fifth", "sixth"]); 
提示:
n == names.length == columns.length1 <= n <= 1041 <= names[i].length, row[i].length, name.length <= 10names[i],row[i],name由小写英文字母组成。1 <= columns[i] <= 101 <= row.length <= 10- 所有的 
names[i]都是 不同 的。 - 最多调用 
ins和rmv2000次。 - 最多调用 
sel104次。 - 最多调用 
exp500次。 
进阶:如果表因多次删除而变得稀疏,您会选择哪种方法?为什么?考虑对内存使用和性能的影响。
解法
方法一:哈希表
创建哈希表 tables 用于存储表名和表数据行的映射。直接模拟题目中的操作即可。
每个操作的时间复杂度均为 \(O(1)\),空间复杂度 \(O(n)\)。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19  |  | 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26  |  | 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25  |  | 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27  |  |