답변:
가장 좋은 방법은 알고리즘 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는 최대 하나의 데이터 사본을 작성합니다. 다음은 샘플 구현입니다.
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 투표의 조정 upvoted 됐어요 저를.
isspace
모든 비 ASCII 문자에 대해 음의 값 (EOF와 다른)을로 전달합니다 ( 실제로 기본 부호 있음 선택) char
. 따라서 정의되지 않은 동작이 있습니다. 나는 그 사실을 소음 속에서 익사시키려는 의도적 인 시도를 의심하기 때문에 그것을 반복하고 있습니다.
에서 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);
Matt Price가 언급 한 것보다 느립니다 . 이유를 모르겠습니다. 실제로 STL 대안이있는 모든 부스트 항목은 해당 gcc보다 느립니다 (내가 테스트 한 모든 것). 그들 중 일부는 엄청나게 느립니다! (정렬되지 않은 맵 삽입에서 최대 5 번) 공유 환경의 CPU 캐시 또는 이와 유사한 것 때문일 수 있습니다.
트리밍의 경우 부스트 문자열 알고리즘을 사용하십시오 .
#include <boost/algorithm/string.hpp>
using namespace std;
using namespace boost;
// ...
string str1(" hello world! ");
trim(str1); // str1 == "hello world!"
이 솔루션을 사용하여 문자를 제거 할 수 있습니다.
#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이 아닌 경우
그러나 다른 사람들이 언급했듯이 지우기 제거 관용구도 좋습니다.
string removeSpaces(string word) {
string newWord;
for (int i = 0; i < word.length(); i++) {
if (word[i] != ' ') {
newWord += word[i];
}
}
return newWord;
}
이 코드는 기본적으로 문자열을 가져와 그 안에있는 모든 문자를 반복합니다. 그런 다음 해당 문자열이 공백인지 확인합니다. 그렇지 않으면 문자가 새 문자열에 추가됩니다.
#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()
a size_t
가 아닌 a를 반환 합니다 int
. erase()
소요 size_type
아닌 int
. 인덱스가 항상 증가하기 때문에 두 개의 연속 공백이 있으면 함수가 실패 할 수 있습니다. 하나의 공백이 제거되면 루프는 문자열의 경계를 넘어 읽습니다. 이 답변은 많은 도움이 필요하므로 삭제해야합니다.
나는 그것이 내가 생각할 수있는 최고의 솔루션이라는 것을 두려워합니다. 그러나 reserve ()를 사용하여 필요한 최소 메모리를 미리 할당하여 속도를 높일 수 있습니다. 아마도 더 짧아 지지만 같은 양의 메모리를 차지하는 새로운 문자열이 생길 수 있지만 재 할당은 피할 수 있습니다.
편집 : 상황에 따라 주변의 캐릭터를 방해하는 것보다 오버 헤드가 덜 발생할 수 있습니다.
다른 접근 방식을 시도하고 자신에게 가장 적합한 것을 확인해야합니다. 성능 문제가 전혀 없을 수 있습니다.