magic number(转自www.baidu.com)

1.计算机中的含义magic number 在计算机中可译为"幻数",或者"魔数",或者直接称其为magic number.中国台湾的叫法是魔术数字.
  它可以用来标记文件或者协议的格式.很多文件都有magic标志来表明该文件的格式.
  例如win32可执行程序的开头标记一般为[MZ],这是一种源于DOS时代的格式.
  rar压缩档文件的开头有[Rar]三个字符,zip文档开头为[PK] ("pack")
  可以用记事本(或者Ultra Edit)直接打开以上文件类型的文件,查看开头的magic.

a.应用

  一般而言,硬盘数据恢复软件(如 EasyRecovery),就是靠分析磁盘上的原始数据,然后根据文件magic来试图匹配文件格式,从而尝试识别出磁盘中那些已经从文件系统登记表中删除的文件(真实的文件内容可能没有被覆盖).但是这种方法不是100%精确,因为磁盘中数据的随机性也很大,很多没有意义的字符串,可能被误认为是有效的File Magic,从而造成回复出无效/错误的文件.
  (p.s.你可以在EasyRecovery中自己定义文件的Magic然后让它帮你回复,不过常用的文件格式它都有记录)
  通常在应用开发中,文件读写也可能使用文件的magic.例如读取文件时,用它来判断文件的格式是否匹配.如果不匹配则报告错误不处理文件,或者尝试读取文件的magic标记来识别.
  常见的例子.例如,把一个bmp图像文件的扩展名改名为png,可能有些图像浏览/编辑软件提示:类型错误,加载失败,但是有的软件却可以识别并读出,并提示格式跟扩展名不匹配.

2.程序开发中的含义

在源代码编写中,用这么一种情况:编码者在写源代码的时候,使用了一个数字,比如0x2123,0.021f等,他当时是明白这个数字的意思的,但是别的程序员看他的代码,可能很难理解,甚至,过了一段时间,代码的作者自己再看代码的时候也忘记了这个数字代表的含义.于是感叹,
  -虽然不知道这个数字是干什么用的,究竟代表什么,但是编译后的程序可以正常运行,真是"魔术般的数字"啊
  magic number 即源于此.
  magic number的这个含义跟上一个不同,这个通常含贬义.因为在编程中使用magic number是不好的习惯,开发中应当尽量避免.
  magic number 的两大弊端:
  1)代码可读性差,例如
  
float time = 1.0f; //这个是时间 
float speed = time * 2.13f
  如果没有说明,很难猜到那个2.13f的含义,假如它代表加速度,那么修改如下:
  
#define ACCELERATION (2.13f) 
float speed = time * ACCELERATION
  这样对于代码阅读者来说更好理解.
  2)修改不方便,例如
  
setfontcolor(string,0xFFFFFFFF); //假设此函数设置一个字符串的颜色 
.... 
setbkcolor(widget,0xFFFFFFFF); //假设此函数设置控件背景色 
...
  暂且不说0xFFFFFFFF代表的含义,如果程序中很多地方使用了统一的一个常量,如果要修改值的时候很麻烦,也容易出错.可能有遗漏等等诸多问题.
  同样可以改为如下
  
static const in WHITE = 0xFFFFFFFF; 
setfontcolor(string,WHITE ); 
... 
setbkcolor(widget,WHITE ); 
...
  这样程序代码不仅便于阅读,而且要替换他的值,只需要替换一次就好了.
  解决魔术数字的方法主要是将这些数字定义为常量,或者枚举类型,或者使用编译器的宏定义(如C/C++的#define)
  magic number在程序开发中还有一个用途(这个时候它是中性词),就是作为调试符号,便于观察和调试程序中出现的错误.
  举一个常见例子,windows下的程序员在调试程序时候,如果报错,可能对如下数字(地址)比较熟悉:0xcdcd ,0xcccc等.
  0xcdcd 是微软的C++ Debug 运行库 为没有初始化的堆内存所做的标记,例如malloc分配出的内存,其内容可能全部都是0xcdcd.由于0xcdcd的编码,解释为中文的话为"屯",所以windows下的程序员,甚至是windows用户应该对"屯屯屯屯屯屯"这样的字符串并不陌生.(可以百度一下 ,关键字 :字符串 屯)
  0xcccc同样是微软的运行库为未初始化的栈空间所做的调试标记.
  类似的还有 0xFDFDFD, 0xFEEEFEEE,0xDEADDEAD,0xABABABAB
  这些都是微软用到的magic nubmer,在win32下程序调试的可以参考,但是不能在程序开发的代码中使用,原因很简单,这个跟平台,运行库和编译模式有很大关系,只是为了调试所设置的标记,仅此而已.
  其它平台也有很多magic number 例如著名的0xDEADBEEF (dead beef)
  如果自己需要编写内存管理模块,使用自己的magic number 也可以很方便的做为调试所用.




posted @ 2011-10-13 22:38  hellomsg  阅读(645)  评论(0编辑  收藏  举报