0

这是我的代码。我想从用户那里获得 5 个字符串,并在用户埋葬它时读取每个字符串。但我收到segmentation fault(core dumped)消息。

#include <string.h>
#include <malloc.h>
#include <espeak/speak_lib.h>

int test()
{

espeak_POSITION_TYPE position_type;
espeak_AUDIO_OUTPUT output;
char *path=NULL;
int Buflength = 500, Options=0;
void* user_data;
t_espeak_callback *SynthCallback;
espeak_PARAMETER Parm;



char Voice[] = {"English"};

int i=0;
char text[1000];
unsigned int Size,position=0, end_position=0, flags=espeakCHARS_AUTO, *unique_identifier;


    output = AUDIO_OUTPUT_PLAYBACK;

    espeak_Initialize(output, Buflength, path, Options ); 
    espeak_SetVoiceByName(Voice);
    const char *langNativeString = "en_US";
    espeak_VOICE voice={0};

        voice.languages = langNativeString;
        voice.name = "US";
        voice.variant = 2;
        voice.gender = 1;
       Size = strlen(text)+1;    


for (i=0; i<5; i++)
{

scanf("%s ", &text);

printf("%s", text);

    espeak_Synth( text, Size, position, position_type, end_position, flags,
    unique_identifier, user_data );
    espeak_Synchronize( );
fflush(stdout);

}

return 0;
}






int main(int argc, char* argv[] ) 
{
    test();

    return 0;
}

我尝试了一些修改,但没有一个起作用。我希望程序像这样工作:

用户输入:你好

espeak 说:嗨

用户输入:一

espeak 说:一个

(用于 5 个输入)

但是当我尝试插入超过 4 个字符作为输入时,它会segmentation fault出错!

4

1 回答 1

1

两个主要问题是:

  1. strlen在未初始化的字符数组上使用;
  2. unique_identifier参数espeak_Synth必须是NULL或指向一个无符号整数(参见源代码),而现在它是一个指向随机存储器的无符号指针。

移动strlen之后scanf,使用NULL代替,unique_identifier你的代码会突然工作(有点)。

但是还有许多其他问题:无用的变量、未初始化的变量、没有输入清理等等。IMO 更好的方法是丢弃该test函数并正确地从头开始重写它。

附录

这就是我重写上面代码的方式。它仍然不是最理想的(没有输入清理,没有错误检查),但 IMO 它干净。

#include <stdio.h>
#include <string.h>
#include <espeak/speak_lib.h>

static void say(const char *text)
{
    static int initialized = 0;
    if (! initialized) {
        espeak_Initialize(AUDIO_OUTPUT_PLAYBACK, 0, NULL, 0);
        espeak_SetVoiceByName("en");
        initialized = 1;
    }
    espeak_Synth(text, strlen(text)+1,
                 0, POS_CHARACTER, 0,
                 espeakCHARS_UTF8, NULL, NULL);
    espeak_Synchronize();
}

int main()
{
    char text[1000];
    int i;

    for (i = 0; i < 5; ++i) {
        scanf("%s", text);
        say(text);
    }

    return 0;
}
于 2017-12-13T11:27:03.163 回答