您的位置:首页 > 编程语言 > C语言/C++

从uboot中学习C语言基本功(uboot/lib_generic/vsprintf.c/simple_strtoul)

2012-11-29 10:03 387 查看
// 此函数有以下几点值得注意:

// 1、第一个参数中的const。一般在函数的形参中,如果我们只是希望调用者使用该参数,而不会去改变该

// 参数内容(一般是指针指向的内容),则可以声明为const。

// 2、第二个参数。C语言中函数只能有一个返回值,但是有时候我们希望从函数中得到不止一个返回内容怎

// 么办呢?只能通过参数了。一般的参数是用来向函数输入信息的,但是指针型参数可以间接用来从函数

// 输出信息。使用时用户只需传入一个相应的指针,函数中会把需要输出的信息地址传给这个指针,这样

// 在函数调用完成后,用户即可到endp指针处去取函数传出的值了。这就是用参数实现返回值的方式。但是

// 要注意,在函数中千万不可把局部变量的地址传给输出型指针,因为局部变量存在栈里,函数调用结束后

// 即释放了,传出的指针指向的内容是栈上已经被释放的部分,因此是无效的。(PS:高级语言中

// 譬如C#有ref,out关键字,以明确指明该引用为输出型)

// 3、第三个参数base。要注意这个base的机制,本函数中是按照这样的理念来设计的。即如果str中有0x

// 开头且接下来是数字则忽略用户调用时输入的base,强制为16进制。如果str不是上面情况然后才看用户

// 输入的base。用户输入非0则使用用户指定的base,若用户输入0则自动判断是8进制还是10进制。

// 需要强调的是:使用习惯决定理念,理念决定代码逻辑。

// 4、注意 if(endp)这里,在输出型参数使用中,函数内首先判断endp是否为NULL,并以此来决定是否输出

// 这个机制来自于一个理念:这个输出参数有可能是用户关注的,也有可能是用户不在意的。这样处理可以

// 给用户自由,即用户如果在意就传一个有效指针过来接收;如果不在意调用时直接给个NULL就行。提供

// 服务但不强迫,这样很好。在OS的API中很多时候都有类似的技巧,请注意体会。

unsigned long simple_strtoul(const char *cp,char **endp,unsigned int base)

{

unsigned long result = 0,value;

// 对于base,首先如果*cp以0x开头且下来是数字,则一定为16进制。

// 然后如果用户指定了一个非0的进制,则遵循用户输入的进制。如果用户输入了0进制,则根据是否0开头来

// 确定是8进制还是10进制

if (*cp == '0') {

cp++;

if ((*cp == 'x') && isxdigit(cp[1])) {

base = 16; // 如果cp是0x开头的,下面一个又是数字,那么base一定为16.此时即使

cp++; // 用户输入了一个base,也忽略这个base而强制其为16

}

if (!base) {

base = 8; // 如果用户输入base为0,cp是0开头,接下来不是x,

} // 那么强制为8进制

}

if (!base) {

base = 10; // 如果用户输入base为0,cp是非0开头,则使用10进制。

}

// while循环里value<base是精髓所在。这个解析停止的条件不能是null,而应该是str中第一个不是

// 数字的字母。这个字母的范围取决于base,譬如如果是16进制那么f也算是数字。而如果是8进制那么

// 9都不算是数字了。因此解析结束的范围只能和base比较来限定。

while (isxdigit(*cp) && (value = isdigit(*cp) ? *cp-'0' : (islower(*cp)

? toupper(*cp) : *cp)-'A'+10) < base) {

result = result*base + value;

cp++;

}

if (endp) // 此处if判断的作用是:用户在使用这个函数时,如果不关心endp则可以直接使用

*endp = (char *)cp; // null,而不必担心程序运行会出错。这种处理技巧很实用,用户可以

return result; // 自行决定是否使用这个函数提供的参数式返回值。

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: