面试题:从一个字符串中修剪多个连续的空格

Posted

技术标签:

【中文标题】面试题:从一个字符串中修剪多个连续的空格【英文标题】:Interview Question : Trim multiple consecutive spaces from a string 【发布时间】:2011-04-06 03:30:23 【问题描述】:

这是一道面试题 寻找从字符串中修剪多个空格的最佳解决方案。这个操作应该是就地操作。

input  = "I    Like    *** a      lot"
output = "I Like *** a lot"

不允许使用字符串函数,因为这是一道面试题。寻找问题的算法解决方案。

【问题讨论】:

你的意思是“in place”而不是“in memory”? @MAK:是的,感谢您指出。 【参考方案1】:

使用<algorithm> 是否符合“算法解决方案”的条件?

#include <iostream>
#include <string>
#include <algorithm>
#include <iterator>
struct BothAre

    char c;
    BothAre(char r) : c(r) 
    bool operator()(char l, char r) const
    
            return r == c && l == c;
    
;
int main()

    std::string str = "I    Like    *** a      lot";
    std::string::iterator i = unique(str.begin(), str.end(), BothAre(' '));
    std::copy(str.begin(), i, std::ostream_iterator<char>(std::cout, ""));
    std::cout << '\n';

试运行:https://ideone.com/ITqxB

【讨论】:

@Matthieu:我不会称之为滥用 - unique stl 算法正是为这种情况设计的。诚然,要求函子类很尴尬,但这是 C++0X 之前 stl 算法的一个相当普遍的限制。 @Eclipse:我称其为滥用,因为它对于无函子的版本有点牵强,它仅使用==。在这里,仿函数改变了语义,因为比较相等的两个项目不一定合并......虽然我对这个技巧感到非常惊讶,特别是因为仿函数上的一个简单的“语义”改变帮助劫持了算法:) @Matthieu M:这与 std::inner_product 在使用创造性选择的函子(正如 APL 程序员几十年前探索过的那样)调用时所能做的相比无济于事 这是一个C++标准库算法的绝妙使用!【参考方案2】:

C++0x - 使用 lambda 而不是常规函数对象的解决方案。与 Cubbi 的解决方案进行比较。

#include <string>
#include <algorithm>

int main()

    std::string str = "I    Like    *** a      lot";

    str.erase(std::unique(str.begin(), str.end(),
      [](char a, char b)  return a == ' ' && b == ' ';  ), str.end() );  

【讨论】:

而不是a == ' ' 可以使用isspace(a) 更一致。【参考方案3】:

保留两个索引:下一个可以放入字母的位置(例如,i)和您正在检查的当前索引(例如,j)。

只需使用j 循环遍历所有字符,每当您看到一个字母时,将其复制到索引i,然后递增i。如果您看到前面没有空格的空格,请同时复制该空格。

认为这将在原地工作......

【讨论】:

确保“前面没有空格”检查不只是检查s[j-1] == ' ',因为数组正在被修改。维护一个标记来跟踪您看到的前一个字符是否是空格会更安全。 @Kannan:是的,我确实考虑过写这个,但我让读者自己想办法。 ;) @Kannan:在这种情况下实际上不需要这样做。只需检查s[j-1] == ' ' 就足够了。【参考方案4】:

我会选择这个:

int main(int argc, char* argv[])

    char *f, *b, arr[] = "  This        is    a test.                ";
    f = b = arr;

    if (f) do
    
        while(*f == ' ' && *(f+1) == ' ') f++;
     while (*b++ = *f++);

    printf("%s", arr);

    return 0;

【讨论】:

@VJo:想解释一下评论吗?该算法(虽然可能效率低下,因为它复制数组的所有有效元素,无论它们是否必须移动 - 即即使它们留在同一个地方 - 似乎是正确的。 @David 是的,它可以正常工作,除了空字符串的情况。你能猜出 arr 包含空字符串时的输出是什么吗? @VJo,它打印一个空字符串。它还会打印什么? @edA 啊,对。如果第一次检查失败,则根本不会执行第二次检查。很抱歉造成混乱【参考方案5】:

我建议使用一个小型状态机(只是一个简单的 switch 语句)。因为如果面试官和我一样,他们会要求你做的第一个改进是完全修剪任何前导或尾随空格,以便:

"    leading and    trailing    "

转化为:

"leading and trailing"

代替:

" leading and trailing "

这是对状态机设计的一个非常简单的修改,对我来说,通过“直接”编码循环来理解一般的状态机逻辑似乎更容易,即使它需要多几行代码代码而不是直接循环。

如果您认为对直接循环的修改不会太糟糕(可以合理地争论),那么我(作为面试官)会提出我也希望数字中的前导零是修剪。

另一方面,许多面试官实际上可能不喜欢状态机解决方案,因为它是“非最佳”的。我想这取决于您要优化的内容。

【讨论】:

如果您同时实施这两个建议,这将有助于人们与此建议相关联,从而使影响变得切实;-)。 非常好的增强。公平地说,它实际上是被问到的:)。如果您也可以为此提供一些算法,那就太好了...【参考方案6】:

这里只使用 stdio:

#include <stdio.h>

int main(void)
    char str[] = "I    Like    *** a      lot";
    int i, j = 0, lastSpace = 0;
    for(i = 0;str[i]; i++)
        if(!lastSpace || str[i] != ' ')
            str[j] = str[i];
            j++;
        
        lastSpace = (str[i] == ' ');
    
    str[j] = 0;
    puts(str);
    return 0;

【讨论】:

【参考方案7】:

修剪多个空格也意味着一个空格应该总是跟一个非空格字符。

int pack = 0;
char str[] = "I    Like    *** a      lot";
for (int iter = 1; iter < strlen(str); iter++)

    if (str[pack] == ' ' && str[iter] == ' ')
        continue;
    str[++pack] = str[iter];

str[++pack] = NULL;

【讨论】:

【参考方案8】:
int j = 0;
int k=0;
char str[] = "I    Like    *** a      lot"; 
int length = strlen(str);
char str2[38];
for (int i = 0; i < length; i++) 
 
    if (str[i] == ' ' && str[i+1] == ' ') 
     continue; 
    str2[j] = str[i];
    j++;
 

str2[j] =NULL;

cout<<str2;

【讨论】:

您是否使用以 ' ' 结尾的字符串进行测试?当 i 指向最后一个字符时,str[i+1] 应该会失败 问题要求提供就地版本,这将复制到单独的缓冲区 (str2)。此外,str2 是在堆栈上静态分配的,并且对于某些输入会溢出——这不是一个好主意。而且你已经声明了一个你不使用的变量“k”。【参考方案9】:
void trimspaces(char * str)
    int i = 0;
    while(str[i]!='\0')
        if(str[i]==' ')
            for(int j = i + 1; j<strlen(str);j++)
                if(str[j]!=' ')
                    memmove(str + i + 1, str + j, strlen(str)-j+1);
                    break;
                
            
        
        i++;
    

【讨论】:

【参考方案10】:

Haskell 中的函数变体:

import Data.List (intercalate)

trimSpaces :: String -> String
trimSpaces =  intercalate " " . words

接下来的算法:

    将字符串分解为单词列表,由空格分隔 连接列表,在列表中的每个元素之间插入一个空格

【讨论】:

【参考方案11】:

这是删除多余空格的非常简单的实现。

#include <iostream>
std::string trimExtraWhiteSpaces(std::string &str);
int main()
    std::string str = "  Apple    is a     fruit  and I like     it   .  ";
    str = trimExtraWhiteSpaces(str);
    std::cout<<str;

std::string trimExtraWhiteSpaces(std::string &str)
    std::string s;
    bool first = true;
    bool space = false;
    std::string::iterator iter;
    for(iter = str.begin(); iter != str.end(); ++iter)
        if(*iter == ' ')
            if(first == false)
                space = true;
            
        else
            if(*iter != ',' && *iter != '.')
                if(space)
                    s.push_back(' ');
                
            
            s.push_back(*iter);
            space = false;
            first = false;
        
    
    return s;

【讨论】:

以上是关于面试题:从一个字符串中修剪多个连续的空格的主要内容,如果未能解决你的问题,请参考以下文章

剑指Offer面试题67. 把字符串转换成整数

剑指Offer面试题67. 把字符串转换成整数

可以将 Jackson 配置为从所有字符串属性中修剪前导/尾随空格吗?

[LeetCode]面试题67. 把字符串转换成整数(字符串)

[LeetCode]面试题67. 把字符串转换成整数(字符串)

从 NSString 的末尾修剪空格