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.length
1 <= n <= 104
1 <= names[i].length, row[i].length, name.length <= 10
names[i]
,row[i]
,name
由小写英文字母组成。1 <= columns[i] <= 10
1 <= row.length <= 10
- 所有的
names[i]
都是 不同 的。 - 最多调用
ins
和rmv
2000
次。 - 最多调用
sel
104
次。 - 最多调用
exp
500
次。
进阶:如果表因多次删除而变得稀疏,您会选择哪种方法?为什么?考虑对内存使用和性能的影响。
解法
方法一:哈希表
创建哈希表 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 |
|