字符串中数字型字符转换成实际的数值

字符串中数字型字符转换成实际的数值



                这是一篇bug笔记.... 如果你正在读C4的源码,遇到和我同样的问题,希望能互助交流~



是节选自C4的一小段代码.这段代码把p指向的字符串中的数字字符读取出来,转换为整形

比方说"0x100"会被转化为

int 类型数据 number = 256;

                

值得注意的是,这儿有行代码比较"苦涩".

ival = ival * 16 + (tk & 15) + (tk >= 'A' ? 9 : 0);

注意到,tk是个字符类型的数据,即char 长度是一个byte.


(tk & 15)得到的是第4位. tk >= 'A' ? 9:0; 这又是什么东西,为什么这么写?


这里其实巧妙的应用了字符的实际ascii码值


'0' 的ascii value : 30(注意这里都是16进制的,30是0x30,61是0x61)

'a' 和'A'的低4位,恰巧都是1, 那么如果16进制字符表示的数字有大于等于a的部分,取该字符的低4位,然后加上9,既可以得到实际的10进制数值.

eg:

          'a' == 0x61

          ('a' & 0xF) == 1;

          (('a' & 0xF) + 9) ==  10;


我自己写了个demo.这里对原作者的代码风格还是很不爽的说....不过C4的作者功力相当不简单...

/*************************************************************
 * Code writer : EOF
 * Code file   : string2num.c
 * Code date   : 2014.12.19
 * e-mail      : jasonleaster@gmail.com
 *
 * Code description:
 *   	Here is a demo for how to translate the number which
 *  is in string into integer but not character :)

 *************************************************************/
#include <stdio.h>

int main()
{
	char* p_string = "0xaa";
	char* p_copy   = p_string;
	char tk        = *p_string;

	p_string++;

	int tmp = 0;	
	
	if(tmp = tk - '0')
	{
		while(*p_string >= '0' && *p_string <= '9')
		{
			tmp = tmp*10 + *p_string - '0';
		}
	}
	else if(*p_string == 'x' || *p_string == 'X')
	{
		while((tk = *++p_string) && 
		      ( (tk >= '0' && tk <= '9') ||
			(tk >= 'a' && tk <= 'f') ||
			(tk >= 'A' && tk <= 'F') )
		     )
		{
			/*
				tmp = tmp*16 + (tk & 15);
				Aha! Don't try to do this. BUG mind.
			*/

			tmp = tmp*16 + (tk & 15) + (tk >= 'A' ? 9 : 0);
		}
	}
	
	printf("string number :%s\n result:%d '0' :%x 'a':%x 'A':%x\n",p_copy,tmp,'0','a','A');

	return 0;		
}

运行结果:







评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值