D 的个人博客

全职做开源,自由职业者

  menu

C语言中级

以下转自一位学长的C语言笔记。。。。

有些是从其他资料里找的。希望对大家有用。

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.
   
   使用场合:一个是方便人类,一个方便计算机

   另外,附一个ASCII编码表的链接http://www.blabla.cn/ref/ascii.html

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);
}

16.最后.留个题目.
   不申请新的空间,对数组进行逆序.