对于以下问题,请提出更好的解决方案(就时间复杂度而言)。我已经在最后解释了我的方法。
有一个文件,其记录具有以下格式:- RecordType; Symbol; price; id; parentId
示例文件如下-
RecordType;Symbol;price;id;parentId
- A;BANK_X;20;2345;0
- A;BANK_Y;30;2346;0
- A;BANK_Z;40;2347;0
- M;BANK_X;50;2348;2345
- M;BANK_Y;10;2349;2346
- A;BANK_X;20;2350;0
- A;BANK_E;40;2351;0
- M;BANK_X;45;2352;2345
- M;BANK_X;20;2353;2350
这样的文件包含数百万条记录。目标是编写一个有效的C ++程序,将文件拆分为多个文件,以便每个较小的文件包含Y个记录,其中Y是作为输入提供的整数。
要记住的要点:
例如,如果将示例文件拆分为至少包含两行的文件,则以下记录应位于一个文件中:
- A;BANK_X;20;2345;0
- M;BANK_X;50;2348;2345
- M;BANK_X;45;2352;2345
我解决问题的方法:
使用的数据结构:
算法:
如果我考虑声明中提到的示例文件,则在应用我的算法后,数据结构将具有以下值:-
Queue< int, std::vector < int> >: [ {2345, <2348, 2352>}, {2346, <2349>}, {2347, <empty>}, {2350, <2353>}, {2351, <empty>}]
Unordered_map 1 < int, std::string >: [{2345, "A;BANK_X;20;2345;0"}, {2346, "A;BANK_Y;30;2346;0"}, {2347, "A;BANK_Z;40;2347;0"}, {2350, "A;BANK_X;20;2350;0"}, {2351, "A;BANK_E;40;2351;0"}]
Unordered_map 2 < int, std::string >: [{2348, "M;BANK_X;50;2348;2345"}, {2349, "M;BANK_Y;10;2349;2346"}, {2352, "M;BANK_X;45;2352;2345"}, {2353, "M;BANK_X;20;2353;2350"}]
答案 0 :(得分:1)
以下是您问题的陈述:
“这样的文件包含数百万条记录。”
“每个记录都有唯一的ID(即记录中倒数第二个字段)”
..断言我建议您使用SQL数据库。这样,您可以将所有内容保存在单个文件中,以便于访问。您将来可以有效地select, insert, update, delete
,而不会失去第一天获得的灵活性。
SQLite确实是一种轻量级的选择。
答案 1 :(得分:0)
您可以使用矢量和地图进行此操作。声明一个带有整数的vector [SIZE_OF_SYMBLE] .map符号。然后,每次获得一个条目时,首先从map中获取该符号的映射的int值,然后将条目推入该矢量。
struct record{string recordType;char symbol;double price;int id;};
map<char,int> symbmol_to_int;
vector<record> piles[SIZE_OF_SYMBOL];
答案 2 :(得分:0)
更新:
我想出了一个更好的解决方案。由于ID似乎是按排序顺序排列的,因此您可以在每一行的处理之后立即写入文件。对于每个子记录,只需将其写入父文件所在的文件即可。只需记住您将父记录写入哪个文件即可。
unordered_map<int, int> id_to_file_id;
实际上,您不需要将整个字符串存储在地图中,只需要存储它在哪一行。这样可以节省一半的空间。
并使用这样的数据结构:
unordered_map<int, int> id_to_line;
map<int, vector<int>> groups; // map<parent_id, vector<child_id>>