acm模式stringstream
在ACM模式中,stringstream
是处理复杂输入格式的强大工具,尤其适用于字符串分割、类型转换、多行混合输入等场景。以下是其核心应用场景及示例:
一、按分隔符分割字符串
1. 按空格分割(最常见)
#include <iostream>
#include <sstream>
#include <vector>
using namespace std;int main() {string line;getline(cin, line); // 读取一整行:"1 2 3 4"stringstream ss(line);vector<int> nums;int num;while (ss >> num) { // 自动按空格分割并转换为整数nums.push_back(num);}// 输出:4(元素个数)cout << nums.size() << endl;return 0;
}
2. 按其他分隔符分割(如逗号、分号)
#include <iostream>
#include <sstream>
#include <vector>
using namespace std;int main() {string line = "a,b,c,d";stringstream ss(line);vector<string> words;string word;while (getline(ss, word, ',')) { // 指定逗号为分隔符words.push_back(word);}// 输出:a b c dfor (const string& w : words) {cout << w << " ";}return 0;
}
二、字符串与数字的互相转换
1. 字符串转整数/浮点数
#include <iostream>
#include <sstream>
using namespace std;int main() {string s = "123.45";stringstream ss(s);double num;ss >> num; // 将字符串转换为double// 输出:123.45cout << num << endl;return 0;
}
2. 数字转字符串
#include <iostream>
#include <sstream>
using namespace std;int main() {int num = 123;stringstream ss;ss << num; // 将整数写入stringstreamstring s = ss.str(); // 转换为字符串// 输出:123(字符串类型)cout << s << endl;return 0;
}
三、处理多行混合输入
示例:每行格式不同的输入
3 // 第一行:整数n
1 2 3 // 第二行:n个整数
abc // 第三行:字符串
#include <iostream>
#include <sstream>
#include <vector>
using namespace std;int main() {string line;// 1. 读取第一行:整数ngetline(cin, line);int n = stoi(line);// 2. 读取第二行:n个整数getline(cin, line);stringstream ss(line);vector<int> nums;int num;while (ss >> num) {nums.push_back(num);}// 3. 读取第三行:字符串getline(cin, line);string s = line;return 0;
}
四、处理动态长度的输入
示例:每行数字个数不确定
1 2 3
4 5
6 7 8 9
#include <iostream>
#include <sstream>
#include <vector>
using namespace std;int main() {string line;vector<vector<int>> matrix;while (getline(cin, line)) { // 逐行读取if (line.empty()) break; // 空行结束stringstream ss(line);vector<int> row;int num;while (ss >> num) {row.push_back(num);}matrix.push_back(row);}// 输出矩阵行数:3cout << matrix.size() << endl;return 0;
}
五、高级技巧:重复利用stringstream
1. 清空stringstream
stringstream ss;
ss << "123";
ss.str(""); // 清空内容
ss.clear(); // 重置状态标志
2. 高效处理大数加法
#include <iostream>
#include <sstream>
using namespace std;string add(string a, string b) {stringstream ss;int carry = 0;int i = a.size() - 1, j = b.size() - 1;while (i >= 0 || j >= 0 || carry) {int sum = carry;if (i >= 0) sum += a[i--] - '0';if (j >= 0) sum += b[j--] - '0';carry = sum / 10;ss << (sum % 10); // 按逆序写入}string result = ss.str();reverse(result.begin(), result.end()); // 反转字符串return result;
}int main() {string a = "123", b = "456";cout << add(a, b) << endl; // 输出:579return 0;
}
六、注意事项
-
性能考虑
stringstream
比直接处理字符数组慢,对于大数据量输入(如1e6级别),建议使用scanf
或手动解析。 -
状态管理
- 每次使用前需用
ss.str("")
和ss.clear()
重置。 - 读取失败时(如字符串转整数格式错误),需用
ss.clear()
恢复状态。
- 每次使用前需用
-
与
getline
配合
若先使用cin
读取整数,再用getline
读取字符串,需用cin.ignore()
清除换行符:int n; cin >> n; cin.ignore(); // 清除换行符 string line; getline(cin, line);
七、常见应用场景
场景 | 实现方式 |
---|---|
读取一行整数列表 | stringstream ss(line); while (ss >> num) nums.push_back(num); |
解析CSV格式 | getline(ss, word, ','); |
字符串转数字(含错误处理) | if (stringstream(s) >> num) { ... } else { ... } |
大数运算(字符串拼接) | stringstream ss; ss << digit; result = ss.str(); |
掌握stringstream
的灵活应用,能大幅提升处理复杂输入格式的效率,是ACM竞赛和在线评测中的必备技能。