0

所以对于大学的一个实验室......我一直面临挑战,使用 fopen、fgets 等在 usr/share/dict/linux.words 文件中查找所有单词,每个元音只按顺序排列一次。

即滑稽

到目前为止,我有以下代码......但它在某处有缺陷......

int all_vowels( char *s )
{ 
    const unsigned char *p = (const unsigned char *)s;

    char *v = malloc(sizeof(char *));
    char *vowel = v;

if(*p == '\0') return -1;

while( *p != '\0' )
{    
        if( *p == 'a' || *p =='e' || *p =='i'|| *p =='o' || *p =='u' )
        {
            *v = *p;
            v++;
        }
        p++;
    }


    if ( *vowel == 'a' && (*vowel + 1) == 'e' && (*vowel + 2) == 'i' && (*vowel + 3) ==     'o' && (*vowel + 4) == 'u' ) 
    { 
        return 1; 
    }

    return -1;
}

int main (int argc, char *argv[])
{    
    FILE *file;
    char line[BUFSIZ];

    if (( file = fopen("/usr/share/dict/words", "r") ) == NULL) 
    {
        fprintf(stderr, "cannot open %s\n", "/usr/share/dict/words");
        exit(1);
    } 

    while ( !feof(file) )
    {
        fgets(line, sizeof(line), file);
        if ( all_vowels(line) == 1 )
        {
            printf("%s\n", line);
        }
    }
    fclose(file);
    return 0;

}

任何提示都会很棒!

我现在真的很迷茫……

4

6 回答 6

1

但它有缺陷的地方......

这里会不会有错误?

if ( *vowel      == 'a' &&
    (*vowel + 1) == 'e' &&
    (*vowel + 2) == 'i' &&
    (*vowel + 3) == 'o' &&
    (*vowel + 4) == 'e' )      
//                  ^^^ 'u'?

也可能有其他错误。我还没有检查你所有的代码。

于 2012-09-06T09:51:17.993 回答
1

这是一个很大的缺陷:

char *v = malloc(sizeof(char *));

这仅分配四个或八个字节(取决于您是在 32 位还是 64 位平台上)。我猜你想要的远不止这些。

PS。将来,您可能应该尝试更具体,而不是仅仅说“它有缺陷”。

于 2012-09-06T09:52:23.803 回答
1

您正在访问v,就好像它指向一个包含多个字符的位置一样,而实际上您只为一个单个保留空间char *(通常在 32 位机器上为 4 字节,在 64 位机器上为 8 字节):

char *v = malloc(sizeof(char *));

对于您要存储的内容,这可能还不够;在您的情况下,任何给定单词中的元音数量。

只要有可能,您应该避免动态分配;在您的情况下,您不需要它们,您可以声明一个固定大小的数组而不是 char*:

char v[5];

除此之外,您必须检查您是否已经阅读了 5 个元音,以免超出数组大小;如果在 5 个元音之后遇到另一个元音,则无论如何都可以停止检查;当前遇到的必须是重复元音,因此该单词不合格。

处理字符的方式也是一个问题。再次检查是做什么*的:它立即取消引用右侧的表达式。在您的情况下,它将始终取消引用 v,然后向其添加一些内容(这也是合法的,因为取消引用的结果是一个字符)。因此,如果 v 指向的第一个字符是 a,第二个字符是 e,那么*v将 yield 'a'(*v + 1)will yield 'b'(*v +2)will yield'c'等等 - 你看,结果是给定数字对字母 a 的加法;第一个字符之后的内容并不重要,因为您永远不会访问那里的值。要使用指针算术实现您想要的,您必须使用括号:*(v+1)- 即,将 1 添加到指针v,然后取消引用它。这将产生从 开始的 c 字符串中的第二个字符v,即'e'。请注意,使用如上声明的 v,您可以简单地编写v[0],v[1]v[2]来寻址每个字符。

除此之外,检查 if 条件中的最后一个比较,那里有一个“e”而不是一个“u”。

顺便说一句,作为旁注,还有一些需要考虑的事情:您的问题有一个解决方案,它根本不需要v/vowel变量......只有一个整数变量!

于 2012-09-06T09:52:52.740 回答
0

为什么要all_vowels()分配内存?而且,更有趣的是,为什么不free()呢?

我很确定all_vowels()不必分配任何内存,并且可以比您拥有的更简单。

此外,在尝试从文件中读取feof() 之前,您不能使用。删除它,只需循环直到fgets()返回NULL

我可能会编写一个辅助函数int is_vowel(char c);来使代码更清晰,然后像这样解决问题all_vowels()

vowels = "aeiou"
for each character x in the string to check:
  if is_vowel(x):
    if vowels starts with x:
      let vowels = vowels[1:]
    else
      return false
return true if vowels is empty
于 2012-09-06T09:50:49.080 回答
0

关于解析你的文件有点不确定,但我下面的函数检查一个字符是否是元音并测试下一个元音是否大于当前元音。

#include <stdio.h>

// for readability not advocating the 
// usage of #define booleans etc
#define TRUE  1
#define FALSE 0

int isVowel (char c)
{
    switch (c)
    {
        case 'a': return TRUE;
        case 'e': return TRUE;
        case 'i': return TRUE;
        case 'o': return TRUE;
        case 'u': return TRUE;
        case 'A': return TRUE;
        case 'E': return TRUE;
        case 'I': return TRUE;
        case 'O': return TRUE;
        case 'U': return TRUE;
    }

    return FALSE;
}

int hasOrderedVowels (char *str)
{    
    char c1, c2;

    c1 = *str;
    c2 = *(++str);

    // ignore words beginning in vowels other then 'a' or 'A'
    if (isVowel(c1) && !(c1 == 'a' || c1 == 'A')) return FALSE;

    do {
        // ignore case of `c1`
        if (c1 >= 'a')
            c1 -= 32;

        // ignore case of `c2`
        if (c2 >= 'a')
            c2 -= 32;

        // compare vowels and increment
        // pointers as appropriate
        if (isVowel(c1) && isVowel(c2))
        {
            // if we have found a vowel less then or equal to current
            // then they are not in order/more then one, if we have found
            // a 'U' and there are more vowels then this would be a duplicate
            if (c2 <= c1 || c1 == 'U')
                return FALSE;

            c1 = c2;
        } 
        else if (isVowel(c2))    // found first vowel so assign to c1
        {
            if (!(c1 == 'a' || c1 == 'A'))
            {
                return FALSE;
            }
            c1 = c2;
        }
        else if (!isVowel(c1))   
        {
            c1 = *(str += 2);    // skip over c2
        }
        c2 = *(++str); 
    } 
    while (c2 != '\0');

    return (c1 == 'U');
}

int main ()
{
    char *str[] = {"aeiou", "facecious", "chimpanze", "baboon"};
    int i = 0;

    for (; i<5; i++)
    {
        printf ("%s: %i\n", str[i], hasOrderedVowels(str[i]));
    }

    return 0;
}

演示

于 2012-09-06T14:57:48.140 回答
0

好的..所以我终于得到了正确的输出......任何提高效率的提示或技巧将不胜感激。

int all_vowels( char *s )
{
const unsigned char *p = (const unsigned char *)s;

char v[5];
int i = 0;
if(*p == '\0') return -1;

while( *p != '\0' )
{
    if( (*p == 'a' || *p =='e' || *p =='i'|| *p =='o' || *p =='u') && ( i < 5 )        )
    {
        v[i] = *p;
        i++;
    }
    p++;
}


if ( ( v[0] == 'a'  &&  v[1] == 'e'  &&  v[2] == 'i'  &&  v[3] == 'o'  &&  v[4] == 'u' ) && (strlen(v) == 5 )) 
{ 
    return 1; 
}

return -1;
}

int main (int argc, char *argv[])
{
FILE *file;
char line[30];

if (( file = fopen("/usr/share/dict/words", "r") ) == NULL) 
{
    fprintf(stderr, "cannot open %s\n", "/usr/share/dict/words");
    exit(1);
} 

while ( fgets(line, sizeof(line), file) )
{

    if ( all_vowels(line) == 1 )
    {
        printf("%s\n", line);
    }
}
fclose(file);
return 0;

}
于 2012-09-06T11:23:58.383 回答