【C语言-13】C之大小写转换

库函数toupper和tolower也有与随机数类似的历史。它们起初被实现为宏:

#define toupper(c) ((c)+'A'-'a')
#define tolower(c) ((c)+'a'-'A')

当定义一个小写字母作为输出,toupper将返回对应的大写字母。而tolower的作用刚好相反。这两个宏都依赖于特定实现字符集的性质,即需要所有的大写字母与相对应的小写字母之间的差值是一个常量。这个假定对ASCII字符集来说是成立的。而且,因为这些宏不能被移植,且这些定义都被封装在一个文件中,所以这个假定也不那么危险。

然而,这个宏确实有一些不足之处:如果输入的字母大小写不对,那么返回的值就是一些无用的垃圾信息。考虑下面的程序,其作用是将文件中所有的大写字母全部转换成小写字母,看似没有问题,实际上程序不能工作。 

int c;
while ((c = getchar()) != EOF )
            putchar (tolower (c));

其实我们应该写成这样才对:

int c;
while ((c = getchar()) != EOF )
            putchar (isupper (c) ? tolower (c): c);

大多数toupper和tolower的使用都需要首先进行检查以保证参数是合适的。重写如下:

#define toupper(c) ((c) >='a' && (c) <= 'z'? (c) + 'A' - 'a':(c))
#define tolower(c) ((c) >='A' && (c) <= 'Z'? (c) + 'a' - 'A':(c))

重写toupper函数看上去大致像这样: 

int 
toupper (int c){
    if(c >= 'a' && c <='z')
            return c + 'A' -'a';
        return c;
}

 重写后的tolower函数与此类似。

这样改动之后程序的健壮性无疑得到了增强,而代价是每次使用这些函数时却有引入了函数调用的开销。他意识到某个人也许不愿意付出效率方面损失的代价,因此他又重新引入了这些宏,不过使用了新的宏名:

#define _toupper(c) ((c) + 'A' -'a')
#define _tolower(c) ((c) + 'a' -'A')

这样,宏的使用者就可以在速度与方便之间自由选择。

 

 

 

你可能感兴趣的:(【C语言-13】C之大小写转换)