Функция разделения строки по разделителям c учетом обрамления

 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
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#include <string>
#include <vector>
using namespace std;
/**
* Функция разделения строки по разделителям c учетом обрамления.
* Возвращает массив строк.
*/
int explodeCmd(const string& str, vector<string>& mas, const char& delimiter = ' ', const char& enclosed_by = '"')
{
string::size_type st_pos = 0;
string::size_type end_pos = 0;
while(end_pos != str.length())
{
// Поиск первого не разделителя
st_pos = str.find_first_not_of(delimiter, end_pos);
if (st_pos == string::npos)
break;
// Если обрамитель
if (str.at(st_pos) == enclosed_by) {
end_pos = str.find_first_of(enclosed_by, st_pos + 1);
// Если нашелся еще один обрамитель до конца строки
if (end_pos != string::npos) {
// проверка на корректность (после обрамителя должен
// быть разделитель или конец строки)
if (str.at(end_pos + 1) != delimiter && end_pos + 1 != str.length())
return -1;
mas.push_back(str.substr(st_pos + 1, end_pos - st_pos - 1));
end_pos ++;
// Если не нашлось ни одного обрамителя до конца строки
} else {
// то ищем первый разделитель
end_pos = str.find_first_of(delimiter, st_pos);
// Если не нашлось ни обрамителя, ни разделителя
if (end_pos == string::npos) {
end_pos = str.length();
// возвращаем все до конца строки и завершаем работу
mas.push_back(str.substr(st_pos, end_pos - st_pos));
break;
} else {
// возвращаем все до разделителя
mas.push_back(str.substr(st_pos, end_pos - st_pos));
}
}
// Если не обрамитель и не разделитель (значащий символ)
} else {
end_pos = str.find_first_of(delimiter, st_pos);
if (end_pos == string::npos)
end_pos = str.length();
// возвращаем все до следующего разделителя или до конца строки
mas.push_back(str.substr(st_pos, end_pos - st_pos));
}
}
return 0;
}