istringstream 将字符串放回输入并再次读取
istringstream put string back on input and read again
所以我从文件中读出行,然后通过字符串流读出这些行。
我发现问题是由于行的格式很少将 2 个单独的部分写在一起并作为一个字符串一起读取。我试图通过将错误的读取值放回流中并再次读取来解决这种情况,但看起来 istringstream 并不关心我将字符放回原处。他们根本不会再读出来。
这里问题分解了。 S1 是一个很好的字符串。 S2 在评论中解决了错误阅读的问题:
简而言之。是否可以将一个字符串放回istringstream 并在下一个操作中读取它??
#include <sstream>
#include <string>
#include <vector>
int main()
{
std::string device_id; //126, I_VS_MainVoltageAvailabl
std::string ea_type; //E
std::string address; //0.1
std::string data_type; //BOOL
std::vector<std::string> comment; //VS - Steuerspannung vorhanden / Main voltage available"
std::string s1 = "126,I_Btn_function_stop E 1.2 BOOL Taster Stopp Funktion / Button Stop Function";
std::string s2 = "126,I_VS_MainVoltageAvailablE 0.1 BOOL VS - Steuerspannung vorhanden / Main voltage available";
std::istringstream ist{ s2 };
ist >> device_id; // Read 126, I_VS_MainVoltageAvailablE the E should be read in ea_type
ist >> ea_type; // 0.1
//my idea
if (!ea_type.empty() && isdigit(static_cast<unsigned char>(ea_type[0]))) { //first is a digit so already next was read
for (const auto& x : ea_type) //Try to put 0.1 in the stream
ist.putback(x);
ea_type = device_id[device_id.size() - 1]; // = "E"
device_id.pop_back(); // = "126, I_VS_MainVoltageAvailabl"
}
ist >> address; // Expected "0.1" instead "BOOL" why 0.1 was putback on the stream???
ist >> data_type;
for (std::string in; ist >> in;)
comment.push_back(in);
}
像往常一样,人们忽略了 return 代码。 putback
有一个 return 代码是有原因的,当它为 false 时,表示 putback
失败。
特别是,std::istringstream
是 input 字符串流,因此是 input-only 流。因此,你不能在上面使用 putback
,它总是会失败。
但是,您可以改用 std::stringstream
,putback
将按您希望的方式运行。
我认为你的逻辑有问题。您实际拥有的是固定字段格式,不适合 istream 提取运算符。
您最好读取整行输入,然后按列偏移量提取 "fields"。
或者,一次读取一个字节,附加到要提取的字符串变量,直到读取足够的字节来填充它。也就是说,将 29 个字节读入 device_id,然后将多少(1?8?)字节读入 ea_type,等等
不过,我想质疑你的评论。 istream 字符串提取器 operator>>(std::istream&, std::string&)
将从输入流中提取一个 space 定界 标记。换句话说,您的第一次提取完成了 "126,"
。所以剩下的逻辑是完全错误的。
所以我从文件中读出行,然后通过字符串流读出这些行。
我发现问题是由于行的格式很少将 2 个单独的部分写在一起并作为一个字符串一起读取。我试图通过将错误的读取值放回流中并再次读取来解决这种情况,但看起来 istringstream 并不关心我将字符放回原处。他们根本不会再读出来。
这里问题分解了。 S1 是一个很好的字符串。 S2 在评论中解决了错误阅读的问题:
简而言之。是否可以将一个字符串放回istringstream 并在下一个操作中读取它??
#include <sstream>
#include <string>
#include <vector>
int main()
{
std::string device_id; //126, I_VS_MainVoltageAvailabl
std::string ea_type; //E
std::string address; //0.1
std::string data_type; //BOOL
std::vector<std::string> comment; //VS - Steuerspannung vorhanden / Main voltage available"
std::string s1 = "126,I_Btn_function_stop E 1.2 BOOL Taster Stopp Funktion / Button Stop Function";
std::string s2 = "126,I_VS_MainVoltageAvailablE 0.1 BOOL VS - Steuerspannung vorhanden / Main voltage available";
std::istringstream ist{ s2 };
ist >> device_id; // Read 126, I_VS_MainVoltageAvailablE the E should be read in ea_type
ist >> ea_type; // 0.1
//my idea
if (!ea_type.empty() && isdigit(static_cast<unsigned char>(ea_type[0]))) { //first is a digit so already next was read
for (const auto& x : ea_type) //Try to put 0.1 in the stream
ist.putback(x);
ea_type = device_id[device_id.size() - 1]; // = "E"
device_id.pop_back(); // = "126, I_VS_MainVoltageAvailabl"
}
ist >> address; // Expected "0.1" instead "BOOL" why 0.1 was putback on the stream???
ist >> data_type;
for (std::string in; ist >> in;)
comment.push_back(in);
}
像往常一样,人们忽略了 return 代码。 putback
有一个 return 代码是有原因的,当它为 false 时,表示 putback
失败。
特别是,std::istringstream
是 input 字符串流,因此是 input-only 流。因此,你不能在上面使用 putback
,它总是会失败。
但是,您可以改用 std::stringstream
,putback
将按您希望的方式运行。
我认为你的逻辑有问题。您实际拥有的是固定字段格式,不适合 istream 提取运算符。
您最好读取整行输入,然后按列偏移量提取 "fields"。
或者,一次读取一个字节,附加到要提取的字符串变量,直到读取足够的字节来填充它。也就是说,将 29 个字节读入 device_id,然后将多少(1?8?)字节读入 ea_type,等等
不过,我想质疑你的评论。 istream 字符串提取器 operator>>(std::istream&, std::string&)
将从输入流中提取一个 space 定界 标记。换句话说,您的第一次提取完成了 "126,"
。所以剩下的逻辑是完全错误的。