我试图将字符串标记为微型字符串"&&"和" ||"。我一直在使用strtok_r()取得了一些成功,但由于strtok_r()缺乏理解,或者对指针的误解,我似乎无法使解析器正常运行。 / p>
代码
121 char *cstr3;
122 char* sp;
123 int fc = findclosest(cstr2);
124 switch (fc){
125 case 0:
126 std::cout << "that's it"; //debug
127 cstr3 = strtok_r(cstr2, ";", &sp);
128 break;
129
130 case 1:
131 std::cout << ";"; //debug
132 cstr3 = strtok_r(cstr2, ";", &sp);
133 break;
134
135 case 2:
136 std::cout << "&&"; //debug
137 cstr3 = strtok_r(cstr2, "&", &sp);
138 break;
139
140 case 3:
141 std::cout << "||"; //debug
142 cstr3 = strtok_r(cstr2, "|", &sp);
143 break;
144
145 default:
146 break;
147 }
148
149 puts(cstr3);//debug
150 while(cstr3 != NULL)
151 {
152 char mustfail =0;
153 char mustpass =0;
154
155 int a = fcall(breakitup(cstr3));
156
157 if (a > 0){
158 delete[] cstr;
159 goto skippy;
160 }
161
162 fc = findclosest(cstr3);
163
164 switch (fc){
165 case 0:
166 cstr3 = strtok_r(NULL, ";", &sp);
167 break;
168
169 case 1:
170 std::cout << ";"; //debug
171 cstr3 = strtok_r(NULL, ";", &sp);
172 break;
173
174 case 2:
175 std::cout << "&&"; //debug
176 cstr3 = strtok_r(NULL, "&", &sp );
177 break;
178
179 case 3:
180 std::cout << "||"; //debug
181 cstr3 = strtok_r(NULL, "|", &sp);
182 break;
183
184 default:
185 break;
186 }
187 if (cstr3 != NULL){ //debug
188 puts(cstr3);
189 }
190 }
没有&#34;&amp;&#34;而不是喂食下一个命令。或者一个&#34; |&#34;,它将STARTING行与一行。因此我假设解析器只删除了第一个&#34;&amp;&#34;
为了更好的浏览,输出以下内容(行以&#39;&gt;&#39;开头表示输入)。道歉,一些调试语句仍在输出中。
> pwd; ls
;pwd //debug
/folder/folder/folder/project //function works properly
ls //debug
notrshell README.md rshell.cpp shelly supershell
> pwd && ls
&&pwd //debug
/folder/folder/folder/project
& ls //debug
由于这个额外的&#39;&amp;&#39;,下一个命令被错误地解释,是否有一种简单的方法可以跳过那个&#39;&amp;&#39;?
我尝试再次调用完全相同的命令,例如:
176 cstr3 = strtok_r(NULL, "&", &sp );
177 cstr3 = strtok_r(NULL, "&", &sp );
假设它只是删除另一个&amp;,但那并没有起作用。
我还尝试将指针移动到偷偷跳过额外的&amp ;,
176 cstr3 = 1+ strtok_r(NULL, "&", &sp );
但不仅是那种非常俗气,而且也无法发挥作用。
有什么东西我可以学习strtok_r()或指针来修复这个烂摊子吗?
答案 0 :(得分:1)
考虑这个输入:pwd && ls
。
在第一个switch
语句中,您可以:
cstr3 = strtok_r(cstr2, "&", &sp);
此次通话后,cstr3
指向字符串:"pwd "
,根据手册:
每次调用strtok()都会返回一个指向包含下一个标记的以null结尾的字符串的指针。该字符串不包括分隔字节。
sp
指向......好吧,手册并未按字面说明,但最有可能指出它在最后一次调用中完成的位置 - 在我们的例子中,就在分隔符之后,即{ {1}}。所以,它指向我们字符串的一部分:&
。这就是在第二次& ls
电话中获得标记化的内容。
我无法确定,但我想strtok_r
函数存在问题 - 你能确认它在这里返回正确的值吗?
无论如何,从手册:
delim参数指定字节集,用于分隔已解析字符串中的标记。
这意味着您应该能够使用findclosest
作为分隔符。这应该可以解决问题。
在调用系统函数时,请记住,对于某些(大多数?)shell,命令末尾的单个"&&"
会将其执行放在后台。