[英]Split values in string separated by both comma and a letter
I am currently working with Arduino where I have a string, that for example looks like this:我目前正在使用 Arduino,在那里我有一个字符串,例如看起来像这样:
"30,30,20,10,50,50M20"
I say for example, because the values can always change.我说例如,因为价值观总是会改变。 The letter however will always be a 1 word letter.但是,该字母将始终是 1 个单词的字母。 There will always be 7 numbers and 1 letter.总会有 7 个数字和 1 个字母。 The letter will always be in between value 6 and 8.该字母将始终介于值 6 和 8 之间。
In that string there are different values and they are separated by both commas and also a letter which makes the solution it a bit more tricky.在那个字符串中有不同的值,它们由逗号和一个字母分隔,这使得解决方案有点棘手。
If I break the string above I need to store it like this:如果我打破了上面的字符串,我需要像这样存储它:
A = 30 A = 30
B = 30乙 = 30
C = 20 C = 20
D = 10 D = 10
E = 50 E = 50
F = 50 F = 50
G = M G = M
H = 20 H = 20
These values regularly change (and the length of each value) so the code has to be setup in a way so it separates the values by the ','
and also the letter, in this case 'M'
.这些值会定期更改(以及每个值的长度),因此必须以某种方式设置代码,以便通过','
和字母(在本例中为'M'
分隔值。
This is what I tried to do myself:这是我自己尝试做的:
char c = Serial.read();
FindA = c.indexOf(',');
A = c.substring(0, ind1);
FindB = c.indexOf(',', ind1+1 );
B = c.substring(ind1+1, ind2+1);
And then somehow implement so it takes out the value before and after the letter (that also needs to be stored).然后以某种方式实现,以便取出字母前后的值(也需要存储)。
I have this outcommented too String recievedMessage += (char)Serial.read();
我也有这个outcommented String recievedMessage += (char)Serial.read();
recievedMessage String recievedMessage += (char)Serial.read();
. .
I am developing on an Arduino.我正在 Arduino 上开发。
You can write a small parser for your string, like:您可以为您的字符串编写一个小型解析器,例如:
#include<cstdlib>
#include<cctype>
#include<iterator>
#include<vector>
std::vector<int> values(1);
char separator = ',';
// read every char from the stream and store it in i
for (int i = Serial.read(); i >= 0; i = Serial.read())
{
// check if i is a digit (and if so, append it to the previous digit)
if (std::isdigit(i))
{
values.back() = values.back() * 10 + std::atoi(i);
}
// check if i is a letter [A-Z][a-z]
else if (std::isalpha(i))
{
values.back() = i;
i = separator;
}
// check if i is the separator
if (i == separator)
{
values.push_back(0);
}
}
A = values[0];
B = values[1]
// ...
Assuming the strings are sent as lines (with a new line character \\n
at the end).假设字符串作为行发送(末尾带有换行符\\n
)。
typedef struct {
int A = 0;
int B = 0;
int C = 0;
int D = 0;
int E = 0;
int F = 0;
char G = 0;
int H = 0;
} Line;
typedef void (*LineHandler)(Line data);
bool line_parser(char c, LineHandler line_handler) {
static Line data;
static int value = 0;
static byte state = 0;
bool complete = false;
// ignore the carriage return character
if (c == '\r') return complete;
if (state < 8) {
if (isdigit(c)) {
value = value*10 + c-'0'; // add the digit
}
else if (c == ',' || c == '\n') {
// if a comma or end of line is found, save the value
switch (state) {
case 0: data.A = value; break;
case 1: data.B = value; break;
case 2: data.C = value; break;
case 3: data.D = value; break;
case 4: data.E = value; break;
// the F and G are set when the letter is found
case 7: data.H = value; break;
}
// advance the parsing state and reset the value
state++;
value = 0;
}
else if (state == 5 && isalpha(c)) { // if parsing the 6th number and a letter is found
data.F = value; // save the 6th number
data.G = c; // save the letter
state += 2; // advance to parsing the 8th value
value = 0; // reset the value
}
else
state = 10; // unexpected character; stop the parser
}
if (c == '\n') {
if (state == 8) {
//got complete line
line_handler(data);
complete = true;
}
else {
// parsing failed
}
// reset the parser
value = 0;
state = 0;
}
return complete;
}
void handle_line(Line data) {
// just print out the data
Serial.println("Line:");
Serial.print("A = ");
Serial.println(data.A);
Serial.print("B = ");
Serial.println(data.B);
Serial.print("C = ");
Serial.println(data.C);
Serial.print("D = ");
Serial.println(data.D);
Serial.print("E = ");
Serial.println(data.E);
Serial.print("F = ");
Serial.println(data.F);
Serial.print("G = ");
Serial.println(data.G);
Serial.print("H = ");
Serial.println(data.H);
Serial.println();
}
void setup() {
Serial.begin(115200);
}
void loop() {
if (Serial.available()) {
line_parser(Serial.read(), handle_line);
}
}
This parser is non-blocking and is feed character by character and after it receives the whole line, it calls a function that should handle the parsed data.这个解析器是非阻塞的,逐个字符地输入,在它接收到整行之后,它调用一个应该处理解析数据的函数。
Sending it:发送:
54,125,11045,11,78,4H45
invalid line
11,22,33,44,55,66K88
30,30,20,10,50,50M20
should send back:应该发回:
Line:
A = 54
B = 125
C = 11045
D = 11
E = 78
F = 4
G = H
H = 45
Line:
A = 11
B = 22
C = 33
D = 44
E = 55
F = 66
G = K
H = 88
Line:
A = 30
B = 30
C = 20
D = 10
E = 50
F = 50
G = M
H = 20
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.