malloc/free 的使用要点
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> 函数malloc的原型如下:
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> void * malloc(size_t size);
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> 用malloc申请一块长度为length的整数类型的内存,程序如下:
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> int *p = (int *) malloc(sizeof(int) * length);
class="MsoNormal" style="margin: 0cm 0cm 0pt; text-indent: 21pt;">我们应当把注意力集中在两个要素上:“类型转换”和“sizeof”。
class="MsoNormal" style="margin: 0cm 0cm 0pt 21pt; text-indent: -21pt; mso-list: l0 level1 lfo1; tab-stops: list 21.0pt;">u malloc返回值的类型是void *,所以在调用malloc时要显式地进行类型转换,将void * 转换成所需要的指针类型。
class="MsoNormal" style="margin: 0cm 0cm 0pt 21pt; text-indent: -21pt; mso-list: l0 level1 lfo1; tab-stops: list 21.0pt;">u malloc函数本身并不识别要申请的内存是什么类型,它只关心内存的总字节数。我们通常记不住int, class="tags" href="/tags/FLOAT.html" title=float>float等数据类型的变量的确切字节数。例如int变量在16位系统下是2个字节,在32位下是4个字节;而class="tags" href="/tags/FLOAT.html" title=float>float变量在16位系统下是4个字节,在32位下也是4个字节。最好用以下程序作一次测试:
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(char) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(int) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(unsigned int) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(long) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(unsigned long) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(class="tags" href="/tags/FLOAT.html" title=float>float) << endl;
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">cout << sizeof(double) << endl;
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> cout << sizeof(void *) << endl;
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> 在malloc的“()”中使用sizeof运算符是良好的风格,但要当心有时我们会昏了头,写出 p = malloc(sizeof(p))这样的程序来。
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
class="MsoNormal" style="margin: 0cm 0cm 0pt 21pt; text-indent: -21pt; mso-list: l1 level1 lfo2; tab-stops: list 21.0pt;">u 函数free的原型如下:
class="MsoNormal" style="margin: 0cm 0cm 0pt 20.75pt; text-indent: 21.25pt;">void free( void * memblock );
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> 为什么free函数不象malloc函数那样复杂呢?这是因为指针p的类型以及它所指的内存的容量事先都是知道的,语句free(p)能正确地释放内存。如果p是NULL指针,那么free对p无论操作多少次都不会出问题。如果p不是NULL指针,那么free对p连续操作两次就会导致程序运行错误。
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
class="MsoNormal" style="margin: 0cm 0cm 0pt;">
new/class="tags" href="/tags/DELETE.html" title=delete>delete 的使用要点
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> 运算符new使用起来要比函数malloc简单得多,例如:
class="MsoNormal" style="margin: 0cm 0cm 0pt 40.95pt; mso-para-margin-left: 3.9gd;">int *p1 = (int *)malloc(sizeof(int) * length);
class="MsoNormal" style="margin: 0cm 0cm 0pt 40.95pt; mso-para-margin-left: 3.9gd;">int *p2 = new int[length];
class="MsoNormal" style="margin: 0cm 0cm 0pt; text-indent: 21pt;">这是因为new内置了sizeof、类型转换和类型安全检查功能。对于非内部数据类型的对象而言,new在创建动态对象的同时完成了初始class="tags" href="/tags/HuaGong.html" title=化工>化工作。如果对象有多个构造函数,那么new的语句也可以有多种形式。例如
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">class Obj
class="MsoDate" style="margin: 0cm 0cm 0pt 21.25pt;">{
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">public :
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> Obj(void); // 无参数的构造函数
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> Obj(int x); // 带一个参数的构造函数
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">…
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">}
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">void Test(void)
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">{
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> Obj *a = new Obj;
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> Obj *b = new Obj(1); // 初值为1
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> …
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> class="tags" href="/tags/DELETE.html" title=delete>delete a;
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;"> class="tags" href="/tags/DELETE.html" title=delete>delete b;
class="MsoNormal" style="margin: 0cm 0cm 0pt 21.25pt;">}
class="MsoNormal" style="margin: 0cm 0cm 0pt;">如果用new创建对象数组,那么只能使用对象的无参数构造函数。例如
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> Obj *objects = new Obj[100]; // 创建100个动态对象
class="MsoNormal" style="margin: 0cm 0cm 0pt;">不能写成
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> Obj *objects = new Obj[100](1);// 创建100个动态对象的同时赋初值1
class="MsoNormal" style="margin: 0cm 0cm 0pt;">在用class="tags" href="/tags/DELETE.html" title=delete>delete释放对象数组时,留意不要丢了符号‘[]’。例如
class="MsoNormal" style="margin: 0cm 0cm 0pt;"> class="tags" href="/tags/DELETE.html" title=delete>delete []objects; // 正确的用法
class="MsoNormal" style="margin: 0cm 0cm 0pt; text-indent: 21.25pt;">class="tags" href="/tags/DELETE.html" title=delete>delete objects; // 错误的用法
class="MsoNormal" style="margin: 0cm 0cm 0pt;">后者相当于class="tags" href="/tags/DELETE.html" title=delete>delete objects[0],漏掉了另外99个对象。