我有一个字符串,我想从中删除所有标点符号。我怎么做?我做了一些研究,发现人们使用 ispunct() 函数(我试过了),但我似乎无法让它在我的代码中工作。有人有什么想法吗?
#include <string>
int main() {
string text = "this. is my string. it's here."
if (ispunct(text))
text.erase();
return 0;
}
我有一个字符串,我想从中删除所有标点符号。我怎么做?我做了一些研究,发现人们使用 ispunct() 函数(我试过了),但我似乎无法让它在我的代码中工作。有人有什么想法吗?
#include <string>
int main() {
string text = "this. is my string. it's here."
if (ispunct(text))
text.erase();
return 0;
}
使用算法remove_copy_if
:-
string text,result;
std::remove_copy_if(text.begin(), text.end(),
std::back_inserter(result), //Store output
std::ptr_fun<int, int>(&std::ispunct)
);
如果您需要将结果作为新字符串,POW 已经有一个很好的答案。如果您想要就地更新,这个答案是如何处理它。
配方的第一部分是std::remove_if
,它可以有效地删除标点符号,将所有非标点符号打包。
std::remove_if (text.begin (), text.end (), ispunct)
不幸的是,std::remove_if
不会将字符串缩小到新的大小。它不能,因为它无法访问容器本身。因此,打包结果后的字符串中会留下垃圾字符。
要处理此问题,std::remove_if
请返回一个迭代器,该迭代器指示仍需要的字符串部分。这可以与字符串erase
方法一起使用,导致以下成语...
text.erase (std::remove_if (text.begin (), text.end (), ispunct), text.end ());
我将其称为习语,因为它是一种适用于许多情况的常用技术。string
除了提供合适erase
的方法之外的其他类型,并且std::remove
(可能还有一些我暂时忘记的其他算法库函数)采用这种方法来关闭它们删除的项目的间隙,但将容器调整大小留给调用者。
#include <string>
#include <iostream>
#include <cctype>
int main() {
std::string text = "this. is my string. it's here.";
for (int i = 0, len = text.size(); i < len; i++)
{
if (ispunct(text[i]))
{
text.erase(i--, 1);
len = text.size();
}
}
std::cout << text;
return 0;
}
输出
this is my string its here
当您删除一个字符时,字符串的大小会发生变化。每当发生删除时,都必须更新它。而且,您删除了当前字符,因此下一个字符成为当前字符。如果您不减少循环计数器,则不会检查标点字符旁边的字符。
ispunct
接受一个char
值而不是一个字符串。
你可以这样做
for (auto c : string)
if (ispunct(c)) text.erase(text.find_first_of(c));
这将起作用,但它是一个缓慢的算法。
Steve314 的回答非常好。我想添加一个小改动:
text.erase (std::remove_if (text.begin (), text.end (), ::ispunct), text.end ());
在函数 ispunct 之前添加 :: 负责重载。
这里的问题是 ispunct() 将一个参数作为一个字符,而您正在尝试发送一个字符串。您应该遍历字符串的元素并删除每个字符,如果它是像这里这样的标点符号:
for(size_t i = 0; i<text.length(); ++i)
if(ispunct(text[i]))
text.erase(i--, 1);
#include <iostream>
#include <string>
#include <algorithm>
using namespace std;
int main() {
string str = "this. is my string. it's here.";
transform(str.begin(), str.end(), str.begin(), [](char ch)
{
if( ispunct(ch) )
return '\0';
return ch;
});
}
#include <iostream>
#include <string>
using namespace std;
int main()
{
string s;//string is defined here.
cout << "Please enter a string with punctuation's: " << endl;//Asking for users input
getline(cin, s);//reads in a single string one line at a time
/* ERROR Check: The loop didn't run at first because a semi-colon was placed at the end
of the statement. Remember not to add it for loops. */
for(auto &c : s) //loop checks every character
{
if (ispunct(c)) //to see if its a punctuation
{
c=' '; //if so it replaces it with a blank space.(delete)
}
}
cout << s << endl;
system("pause");
return 0;
}
您可以执行此操作的另一种方法如下:
#include <ctype.h> //needed for ispunct()
string onlyLetters(string str){
string retStr = "";
for(int i = 0; i < str.length(); i++){
if(!ispunct(str[i])){
retStr += str[i];
}
}
return retStr;
这最终会创建一个新字符串,而不是实际从旧字符串中删除字符,但与使用一些更复杂的内置函数相比,它更容易绕开你的脑袋。
尝试使用这个,它将删除文本文件中字符串上的所有标点符号。
str.erase(remove_if(str.begin(), str.end(), ::ispunct), str.end());
如果有帮助请回复
我尝试应用@Steve314 的答案,但直到我在 cppreference.com 上看到此注释后才使其工作:
笔记
与 中的所有其他函数一样, 如果参数的值既不能表示为 也不等于 EOF
<cctype>
,则 的行为是未定义的。要在普通s(或s)中安全地使用这些函数,首先应将参数转换为.std::ispunct
unsigned char
char
signed char
unsigned char
通过研究它提供的示例,我可以使它像这样工作:
#include <string>
#include <iostream>
#include <cctype>
#include <algorithm>
int main()
{
std::string text = "this. is my string. it's here.";
std::string result;
text.erase(std::remove_if(text.begin(),
text.end(),
[](unsigned char c) { return std::ispunct(c); }),
text.end());
std::cout << text << std::endl;
}
我知道了。
size_t found = text.find('.');
text.erase(found, 1);