回答:
最善の方法は、アルゴリズムremove_if
とisspace を使用することです。
remove_if(str.begin(), str.end(), isspace);
これで、アルゴリズム自体はコンテナーを変更できなくなり(値を変更するだけ)、そのため、実際には値をシャッフルし、終了が必要な場所へのポインターを返します。そのため、コンテナーの長さを実際に変更するには、string :: eraseを呼び出す必要があります。
str.erase(remove_if(str.begin(), str.end(), isspace), str.end());
また、remove_ifは最大で1つのデータのコピーを作成することにも注意してください。次に実装例を示します。
template<typename T, typename P>
T remove_if(T beg, T end, P pred)
{
T dest = beg;
for (T itr = beg;itr != end; ++itr)
if (!pred(*itr))
*(dest++) = *itr;
return dest;
}
erase
後で呼び出す必要があることを明示的に述べています。それは正しい結果を返します。
isspace
元の7ビットASCIIを除くすべての文字セットのUBです。C99§7.4/ 1。非常に悪いアドバイスであるにもかかわらず、今までに71票の投票に賛成されていることは驚くことではありません。
isspace
、すべての非ASCII文字について、負の値(EOFとは異なる)にを渡しますchar
。したがって、未定義の動作があります。私はそれを繰り返しています。私はその事実を騒音で溺れさせようとする意図的な試みを疑っているからです。
std::string::iterator end_pos = std::remove(str.begin(), str.end(), ' ');
str.erase(end_pos, str.end());
<algorithm>
これを機能させるには、を含める必要があります。
gamedevから
string.erase(std::remove_if(string.begin(), string.end(), std::isspace), string.end());
::isspace
はUB であることに注意してください。
Boost String Algoを使用できますか?http://www.boost.org/doc/libs/1_35_0/doc/html/string_algo/usage.html#id1290573
erase_all(str, " ");
remove_if(str.begin(), str.end(), isspace);
マット価格が述べたものより遅いです。理由はわかりません。実際、STLの代替案があるすべてのboostのものは、対応するgccのもの(私がテストしたものすべて)よりも低速です。それらのいくつかは非常に遅いです!(unordered_map挿入で最大5回)多分それは、共有環境のCPUキャッシュまたはそのようなものが原因である可能性があります。
トリミングには、ブースト文字列アルゴリズムを使用します。
#include <boost/algorithm/string.hpp>
using namespace std;
using namespace boost;
// ...
string str1(" hello world! ");
trim(str1); // str1 == "hello world!"
このソリューションを使用して、charを削除できます。
#include <algorithm>
#include <string>
using namespace std;
str.erase(remove(str.begin(), str.end(), char_to_remove), str.end());
こんにちは、あなたはそのようなことをすることができます。この関数は、すべてのスペースを削除します。
string delSpaces(string &str)
{
str.erase(std::remove(str.begin(), str.end(), ' '), str.end());
return str;
}
不要なスペースをすべて削除する別の関数を作成しました。
string delUnnecessary(string &str)
{
int size = str.length();
for(int j = 0; j<=size; j++)
{
for(int i = 0; i <=j; i++)
{
if(str[i] == ' ' && str[i+1] == ' ')
{
str.erase(str.begin() + i);
}
else if(str[0]== ' ')
{
str.erase(str.begin());
}
else if(str[i] == '\0' && str[i-1]== ' ')
{
str.erase(str.end() - 1);
}
}
}
return str;
}
string replaceinString(std::string str, std::string tofind, std::string toreplace)
{
size_t position = 0;
for ( position = str.find(tofind); position != std::string::npos; position = str.find(tofind,position) )
{
str.replace(position ,1, toreplace);
}
return(str);
}
これを使って:
string replace = replaceinString(thisstring, " ", "%20");
string replace2 = replaceinString(thisstring, " ", "-");
string replace3 = replaceinString(thisstring, " ", "+");
これを簡単なマクロで実行したい場合は、次のようになります。
#define REMOVE_SPACES(x) x.erase(std::remove(x.begin(), x.end(), ' '), x.end())
#include <string>
もちろんこれはあなたがやったことを前提としています。
次のように呼び出します。
std::string sName = " Example Name ";
REMOVE_SPACES(sName);
printf("%s",sName.c_str()); // requires #include <stdio.h>
私は以下の回避策を長い間使用しました-その複雑さについてはわかりません。
s.erase(std::unique(s.begin(),s.end(),[](char s,char f){return (f==' '||s==' ');}),s.end());
キャラクター' '
や一部を削除する-
場合など
s.erase(std::unique(s.begin(),s.end(),[](char s,char f){return ((f==' '||s==' ')||(f=='-'||s=='-'));}),s.end());
同様に、||
削除したい文字数が1でない場合は、単に増やします
しかし、他の人が述べたように、消去と削除のイディオムも問題ないようです。
#include <algorithm> using namespace std; int main() { . . s.erase( remove( s.begin(), s.end(), ' ' ), s.end() ); . . }
このフォーラムから引用された参照。
C ++ 20では、無料の関数std :: eraseを使用できます
std::string str = " Hello World !";
std::erase(str, ' ');
完全な例:
#include<string>
#include<iostream>
int main() {
std::string str = " Hello World !";
std::erase(str, ' ');
std::cout << "|" << str <<"|";
}
印刷する| 最初のスペースも削除されていることは明らかです。
注:これにより、スペースのみが削除され、空白と見なされる可能性のある他のすべての文字は削除されません。https: //en.cppreference.com/w/cpp/string/byte/isspaceを参照してください。
タブや改行などのすべての空白文字を削除します(C ++ 11):
string str = " \n AB cd \t efg\v\n";
str = regex_replace(str,regex("\\s"),"");
string removespace(string str)
{
int m = str.length();
int i=0;
while(i<m)
{
while(str[i] == 32)
str.erase(i,1);
i++;
}
}
length()
返します。ではなく、を取ります。インデックスは常にインクリメントされるため、2つの連続したスペースが検出されると、関数はおそらく失敗します。1つのスペースが削除されると、ループは文字列の境界を超えて読み取ります。多くの助けが必要なので、おそらくこの答えを削除する必要があります。size_t
int
erase()
size_type
int
それは私が考えることができる最良の解決策だと思います。ただし、reserve()を使用して、必要最小限のメモリを事前に割り当てておき、少し速度を上げることができます。おそらく短くなるが、同じ量のメモリを消費する新しい文字列が作成されますが、再割り当ては回避されます。
編集:あなたの状況によっては、これは周りの文字を混乱させるよりもオーバーヘッドが少ないかもしれません。
さまざまなアプローチを試して、何が最適かを確認する必要があります。パフォーマンスの問題はまったく発生しない可能性があります。