2

我想用单个空格替换字符串中的多个空格,但是我的以下代码不起作用。逻辑错误是什么?

#include<stdio.h>
#include<string.h>
main()
{
char input[100];
int i,j,n,z=0;
scanf("%d",&n);
z=n;
for(i=0;i<n;i++)
scanf("%c",&input[i]);
for(i=0;i<n;i++)
{
    if(input[i]==' ' && (input[i+1]==' ' || input[i-1]==' '))
    {
        --z;
        for(j=i;j<n;j++)
        input[j]=input[j+1];
    }
}
for(i=0;i<z;i++)
    printf("%c",input[i]);
printf("\n");
}
4

8 回答 8

7

我会做这样的事情:

void replace_multi_space_with_single_space(char *str)
{
    char *dest = str;  /* Destination to copy to */

    /* While we're not at the end of the string, loop... */
    while (*str != '\0')
    {
        /* Loop while the current character is a space, AND the next
         * character is a space
         */
        while (*str == ' ' && *(str + 1) == ' ')
            str++;  /* Just skip to next character */

       /* Copy from the "source" string to the "destination" string,
        * while advancing to the next character in both
        */
       *dest++ = *str++;
    }

    /* Make sure the string is properly terminated */    
    *dest = '\0';
}

当然,上述函数需要您正确终止字符串,而您目前没有这样做。

上面的函数所做的基本上是将字符串复制到自身之上。例外是当有一个空格时,多个空格被简单地丢弃。

由于该函数修改了源字符串,因此它不能用于字符串文字。

于 2013-05-28T11:07:43.803 回答
2

这给你带来了一些问题:它 在输入长度后scanf读取你给出的。因此,您将错过循环退出后的最后一个字符。已经给出的答案已经足够好了。但是,如果您想遵循自己的逻辑,请尝试以下操作:\nnfor

void main()
{
    char input[100];
    int i = 0,j,n = 0;
    while ((input[n] = getchar()) != '\n') {
        n++;
    }
    input[n] = '\0';
    while (i < n)
    {
        if(input[i]==' ' && (input[i+1]==' ' || input[i-1]==' '))
        {
            for(j=i;j<n;j++)
            input[j]=input[j+1];
            n--;
        }
        else
        {
            i++;
        }
    }
    printf("%s\n",input);
    printf("\n");
}
于 2013-05-28T11:33:14.790 回答
1

if(input[i]==' ' && (input[i+1]==' ' || input[i-1]==' '))

case " 1 3" : 当 i == 0 accses input[i-1] Out-of-Bounds

scanf("%d",&n);

保持换行符,(input[0] <-- '\n')

修复到

scanf("%d%*c",&n);

#include <stdio.h>

char* uniq_spc(char* str){
    char *from, *to;
    int spc=0;
    to=from=str;
    while(1){
        if(spc && *from == ' ' && to[-1] == ' ')
            ++from;
        else {
            spc = (*from==' ')? 1 : 0;
            *to++ = *from++;
            if(!to[-1])break;
        }
    }
    return str;
}

int main(){
    char input[]= "  abc   de  f  ";

    printf("\"%s\"\n", uniq_spc(input));//output:" abc de f "
    return 0;
}
于 2013-05-28T11:01:33.000 回答
1

为什么让它比它需要的更复杂?您可以使用strtok检查单个空格并忽略它们。然后你可以使用strcat将字符串连接成一个完整的句子,然后你就完成了。

我是这样做的:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>

int main(void) {
    char *pch;
    char sentence[1000];
    char without[1000];

    printf("Sentence: ");
    fgets(sentence,1000, stdin);
    strtok(sentence, "\n"); // remove any newlines
    pch = strtok(sentence, " ");
    while(pch != NULL) {
      strcat(without, pch);
      strcat(without, " \0");
      pch = strtok(NULL, " ");
    }
    without[strlen(without)-1] = '\0'; // remove extra whitespace at the end
    printf("|%s|\n",without);
    return 0;
}
于 2015-10-26T22:15:31.023 回答
1
#include <stdio.h>
#include <stdlib.h>
void remove_blanks(char* s);

int main()
{
    char const s[] = {'1',' ',' ','2',' ',' ','3'};
    remove_blanks(s);
    printf("%s",s);
    return 0;
}

void remove_blanks(char* s){
    int i=0, delta=0, cnt=0;

    for (i=0;s[i];++i){
        if (s[i]==' ') cnt++;
        if (cnt>1){
            delta+=1;
            cnt=0;
        }
        s[i-delta]=s[i];
        if(delta>0) s[i]='\0';

    }
}
于 2017-03-06T00:22:19.877 回答
1

你不能尝试这个简单的代码:

#include <stdio.h>

#define IN 1
#define OUT 0

int main() {
  int c, spaces, state;

  spaces = 0;
  state = OUT;  
  while ((c = getchar()) != EOF) {

      if ( c == ' ') {
           ++spaces;
           state = OUT;
       }
      else if (state == OUT) {
          state = IN;
          spaces = 0;
      }
      if (c == ' ' && spaces > 1 && state == OUT)
          c = 0;            
      putchar(c);
  }
  return 0;
}
于 2017-10-04T14:44:55.020 回答
0

您必须修复以下 for 循环。你的 for 循环的限制应该是z而不是n

for(j=i;j<n;j++)
input[j]=input[j+1];

经过

for(j=i;j<z;j++)
input[j]=input[j+1];

顺便说一句:你的第scanf()一个字符(读取字符)是换行符(\n)。这个换行符来自scanf()小数的第一个(%d

于 2013-05-28T11:23:13.700 回答
0
#include<stdio.h>
#include<string.h>
int main(void)
    {
        char input[1000];
        int i=0;
        gets(input); 
        for(i=0;input[i]!='\0';i++)
        {
            if(input[i]!=' ' || input[i+1]!=' ')
                printf("%c",input[i]);
        }
        return 0;
    }
于 2016-04-06T11:09:52.230 回答