- UID
 - 66114
 
 注册时间2010-4-1
阅读权限30
最后登录1970-1-1
龙战于野 
  
 
 
 
TA的每日心情  | 慵懒 2019-3-12 17:25 | 
|---|
 
  签到天数: 3 天 [LV.2]偶尔看看I  
 | 
 
1.表达式的值:表达式有表达式的值,它是无名的,短暂的。 
2.case 常量(不可以是运算) 
3.数据区(栈,静态区,堆)和代码区 
4.sizeof与数组(下边的说明不考虑机器或者平台等因素). 
一. 
  int i[10]; 
  int *p 
  p=i; 
  上边sizeof(i)为40.但是sizeof(p)是4.原因:数组名不是指针. 
 
二. 
  数组的大小: 
  sizeof(i)/sizeof(i[0]); 
  前者为40,后者为4,于是就是10个. 
三. 
  如果把数组名作为参数传给函数,那么函数的形参一定是指针了.于是在函数内就无法算出来数组的大小了.因为sizeof(这个参数)是4.所以我觉得很多函数需要传size进去.而不是函数内计算大小. 
5.const 
 
const int a; 
int const a; 
const int *a; 
int * const a; 
int const * a const; 
前两个的作用是一样:a是一个常整型数。第三个意味着a是一个指向常整型数的指针(也就是,整型数是不可修改的,但指针可以)。第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是可以修改的,但指针是不可修改的)。最后一个意味着a是一个指向常整型数的常指针(也就是说,指针和所指向的整型数都是不可修改的)。 
 
6.typedef的陷阱: 
 
    typedef  char * pstr; 
    /*若我们这样使用:*/ 
    int mystrcmp(const pstr, const pstr);  
    /*我们希望表达的是:*/ 
    int mystrcmp( const char*, const char* ); 
    /*(即 2个指向常量的char指针)*/ 
    /*可它被解释为: 
    int mystrcmp( char* const, char* const )  ; 
    /*(2个指向char的常指针)*/ 
 
分析: 
const修饰的是pstr,而pstr被定义为char *,而不是char。 
 
 
应修改为:  
        typedef const char* cpstr; 
        int myctrcmp( cpstr, cpstr); 
 
 
 
 
 
7.复杂指针解析 
  比如int (*(*func)[5][6])[7][8]; 
  func是指向三维数组的指针,这类数组的大元素是具有5X6个int元素的二维数组,而指向三维数组的指针又是另一个三维指针数组的元素。 
 
8.为什么会有匿名结构体呢? 
  为了不想让其他人再声明该类型的变量. 
9.struct的大小 
  struct{ 
    char ch; 
    int i; 
    float f; 
    char ch2; 
  }a; 
  struct{ 
    char ch; 
    char ch2; 
    float f; 
    int i; 
  }b; 
 
   sizeof(a)==16 
   sizeof(b)==12 
   为什么成员一样,但是顺序不同,占的空间不一样呢? 
   a的存储:ch,_,_,_(补了三个字节),i,float f,ch2,_,_,_ 
   b的存储:ch,ch2,_,_f,i 
   得出结论,用成员中占最大字节的类型为单位,然后填充变量.要是填得下就填,填不下就新找个该单位填.如b的存储ch,ch2,放到四字节的内存,还空两个位置. 
   再一个结论:声明struct 的时候要注意成员的先后顺序. 
10.malloc和calloc 
   一.malloc(大小),calloc(个数,每个的大小)用哪个更好? 
      用calloc好. 
      原因:因为大小虽然可以个数*每个的大小得出来.但是要使这个值很大很大怎么办?所以分开写(calloc)比计算这个大小(malloc)要好.这个原因是个人认为.但是应该用calloc的结论非个人认为. 
   二.malloc(0) 
      虽然申请0大小的内存,但是结果是malloc返回的结果并不等于NULL,就是说它是有返回内存的.有多大?忘了... 
 
11.FILE指针 
一. 
    当使用文件指针的时候,指针指的是一个结构体,该结构体除了包含各种各样的和该文件相关的信息,还指着一个buffer,而该buffer和文件又有一个有时读入有时写入的流. 
    就是说,fopen,不单单返回了一个文件指针,而是构建了整个读取文件的环境. 
二.文件的文本方式与二进制方式. 
   假设123,文本方式用每个字的ASCII值表示'49','50','51' 
   二进制方式则一个字节就可以表示了1111011. 
 
   使用场合:一个是方便人类,一个方便计算机 
 
12.预处理 
一. 
不要忽视宏定义中的空格: 
#define f  (x) (x) +1 //实际上是把f 定义成了(x) (x)+1  
 
对于不带参的宏,若宏值多于一项,一定要使用括号 
#define MAX (M+N) 
要给每个参数加上括号,否则可能影响计算的优先级 
#define abs(x)  (x>=0)?x:-x 
z =abs(a+b); /*相当于 z= (a+b>=0)?a+b:-a+b 
//修改后: 
#define abs(x) ((x)>=0?(x): -(x)) 
尽量用typedef而不是宏定义去定义类型。 
 
二.  
用typedef和宏都可以定义新类型.那么哪个更好. 
答:typedef更好. 
原因:int* a,b;   a和b的类型,a是指针,b是整形.同样的,用宏定义出来的新类型,使用的时候也会出现这样不知道是新类型指针还是新类型的情况. 
 
三. 
由于头文件包含可以嵌套,那么C文件就有可能多次包含同一个头文件,就可能出现重复定义的问题 
通过条件编译开关来避免重复包含 
例如 
#ifndef  __headerfileXXX__ 
    #define  __headerfileXXX__ 
   … 
    文件内容 
   … 
#endif 
四. 
#include “xxxx.h” 
到本级目录去找和到系统默认目录去找。 
#include <xxxx.h> 
到系统默认目录。 
 
13.常用库函数(总结略,举assert为例) 
诊断函数(assert.h) 
测试一个条件test ,当结果为假时使程序终止,为真时没反应 。 
 
不要将assert函数用于程序的执行逻辑中,它仅用于调试。 
即:程序的 if( 条件 ) 语句不可用 assert( 条件 ) 来替代。 
 
如果在程序开始加入代码: 
#define  NDEBUG         
则该程序中的assert函数全部失效,不必逐一删除。(有待验证.因为我试验了怎么不管用) 
 
14.关键字static的作用是什么? 
在C语言中,关键字static有三个明显的作用: 
  在函数体,一个被声明为静态的变量在这一函数被调用过程中维持其值不变。 
  在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问。它是一个本地的全局变量。 
  在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那就是,这个函数被限制在声明它的模块的本地范围内使用。 
 
15.竟然允许这样使用指针: 
#include "stdio.h" 
void main() 
{ 
        int a = 1234; 
                     // 0x0012ff7c是&a的值。 
        int *p =( (int*)0x0012ff7c ); 
        printf("&a = %x\n",&a); 
        printf("a = %d\n*p = %d\n",a,*p); 
} |   
 
 
 
 |