本贴是495个问题的一部分
第1章 声明和初始化
基本类型
1.1 我该如何决定使用哪种整数类型?
1.2 为什么不精确定义标准类型的大小?
1.3 因为C语言没有精确定义类型的大小,所以我一般都用typedef定义int16和int32。然后根据实际的机器环境把它们定义为int、short、long等类型。这样看来,所有的问题都解决了,是吗?
1.4 新的64位机上的64位类型是什么样的?
指针声明
1.5 这样的声明有什么问题?char*p1,p2;我在使用p2的时候报错了。
1.6 我想声明一个指针,并为它分配一些空间,但却不行。这样的代码有什么问题?char*p;*p=malloc(10);
声明风格
1.7 怎样声明和定义全局变量和函数最好?
1.8 如何在C中实现不透明(抽象)数据类型?
1.9 如何生成“半全局变量”,就是那种只能被部分源文件中的部分函数访问的变量?
存储类型
1.10 同一个静态(static)函数或变量的所有声明都必需包含static存储类型吗?
1.11 extern在函数声明中是什么意思?
1.12 关键字auto到底有什么用途?
类型定义(typedef)
1.13 对于用户定义类型,typedef和#define有什么区别?
1.14 我似乎不能成功定义一个链表。我试过typedefstruct{char*item;NODEPTRnext;}*NODEPTR;但是编译器报了错误信息。难道在C语言中结构不能包含指向自己的指针吗?
1.15 如何定义一对相互引用的结构?
1.16 Struct{ }x1;和typedefstruct{ }x2;这两个声明有什么区别?
1.17 “typedefint(*funcptr)();”是什么意思?
const限定词
1.18 我有这样一组声明:typedefchar*charp;constcharpp;为什么是p而不是它指向的字符为const?
1.19 为什么不能像下面这样在初始式和数组维度值中使用const值?constintn=5;inta[n];
1.20 constchar*p、charconst*p和char*constp有什么区别?
复杂的声明
1.21 怎样建立和理解非常复杂的声明?例如定义一个包含N个指向返回指向字符的指针的函数的指针的数组?
1.22 如何声明返回指向同类型函数的指针的函数?我在设计一个状态机,用函数表示每种状态,每个函数都会返回一个指向下一个状态的函数的指针。可我找不到任何方法来声明这样的函数——感觉我需要一个返回指针的函数,返回的指针指向的又是返回指针的函数……,如此往复,以至无穷。
数组大小
1.23 能否声明和传入数组大小一致的局部数组,或者由其他参数指定大小的参数数组?
1.24 我在一个文件中定义了一个extern数组,然后在另一个文件中使用,为什么sizeof取不到数组的大小?
声明问题
1.25 函数只定义了一次,调用了一次,但编译器提示非法重声明了。
*1.26 main的正确定义是什么?voidmain正确吗?
1.27 我的编译器总在报函数原型不匹配的错误,可我觉得没什么问题。这是为什么?
1.28 文件中的第一个声明就报出奇怪的语法错误,可我看没什么问题。这是为什么?
1.29 为什么我的编译器不允许我定义大数组,如doublearray[256][256]?
命名空间
1.30如何判断哪些标识符可以使用,哪些被保留了?
初始化
1.31 对于没有显式初始化的变量的初始值可以作怎样的假定?如果一个全局变量初始值为“零”,它可否作为空指针或浮点零?
1.32 下面的代码为什么不能编译?intf(){chara[]="Hello,world!";}
*1.33 下面的初始化有什么问题?编译器提示“invalidinitializers”或其他信息。char*p=malloc(10);
1.34 chara[]="stringliteral";和char*p="stringliteral";初始化有什么区别?当我向p赋值的时候,我的程序崩溃了。
1.35 chara{[3]}="abc";是否合法?
1.36 我总算弄清楚函数指针的声明方法了,但怎样才能初始化呢?
1.37 能够初始化联合吗?
第2章 结构、联合和枚举
结构声明
2.1 structx1{ };和typedefstruct{ }x2;有什么不同?
2.2 这样的代码为什么不对?structx{ };xthestruct;
2.3 结构可以包含指向自己的指针吗?
2.4 在C语言中用什么方法实现抽象数据类型最好?
*2.5 在C语言中是否有模拟继承等面向对象程序设计特性的好方法?
2.6 为什么声明externf(structx*p);给我报了一个晦涩难懂的警告信息?
2.7 我遇到这样声明结构的代码:structname{intnamelen;charnamestr[1];};然后又使用一些内存分配技巧使namestr数组用起来好像有多个元素,namelen记录了元素个数。它是怎样工作的?这样是合法的和可移植的吗?
2.8 我听说结构可以赋给变量也可以对函数传入和传出。为什么K&R1却明确说明不能这样做?
2.9 为什么不能用内建的==和!=操作符比较结构?
2.10结构传递和返回是如何实现的?
2.11 如何向接受结构参数的函数传入常量值?怎样创建无名的中间的常量结构值?
2.12 怎样从/向数据文件读/写结构?
结构填充
2.13 为什么我的编译器在结构中留下了空洞?这导致空间浪费而且无法与外部数据文件进行“二进制”读写。能否关掉填充,或者控制结构域的对齐方式?
2.14 为什么sizeof返回的值大于结构大小的期望值,是不是尾部有填充?
2.15 如何确定域在结构中的字节偏移量?
2.16 怎样在运行时用名字访问结构中的域?
2.17 C语言中有和Pascal的with等价的语句吗?
2.18 既然数组名可以用作数组的基地址,为什么对结构不能这样?
2.19 程序运行正确,但退出时却“coredump”(核心转储)了,怎么回事?
联合
2.20 结构和联合有什么区别?
2.21 有办法初始化联合吗?
2.22 有没有一种自动方法来跟踪联合的哪个域在使用?
枚举
2.23 枚举和一组预处理的#define有什么不同?
2.24 枚举可移植吗?
2.25 有什么显示枚举值符号的容易方法吗?
位域
2.26 一些结构声明中的这些冒号和数字是什么意思?
2.27 为什么人们那么喜欢用显式的掩码和位操作而不直接声明位域?
第3章 表达式
求值顺序
3.1 为什么这样的代码不行?a=i++;
3.2 使用我的编译器,下面的代码inti=7;printf("%d\n",i++*i++);打印出49。不管按什么顺序计算,难道不该是56吗?
3.3 对于代码inti=3;i=i++;不同编译器给出不同的i值,有的为3,有的为4,哪个是正确的?
*3.4 有这样一个巧妙的表达式:a^=b^=a^=b;它不需要临时变量就可以交换a和b的值。
3.5 可否用显式括号来强制执行我所需要的计算顺序并控制相关的副作用?就算括号不行,操作符优先级是否能够控制计算顺序呢?
3.6 可是&&和||操作符呢?我看到过类似while((c=getchar())!=EOF&&c!='\n')的代码……
3.7 是否可以安全地认为,一旦&&和||左边的表达式已经决定了整个表达式的结果,则右边的表达式不会被求值?
3.8 为什么表达式printf("%d%d",f1(),f2());先调用了f2?我觉得逗号表达式应该确保从左到右的求值顺序。
3.9 怎样才能理解复杂表达式并避免写出未定义的表达式?“序列点”是什么?
3.10在a=i++;中,如果不关心a[]的哪一个分量会被写入,这段代码就没有问题,i也的确会增加1,对吗?
3.11 人们总是说i=i++的行为是未定义的。可我刚刚在一个ANSI编译器上尝试过,其结果正如我所期望的。
3.12 我不想学习那些复杂的规则,怎样才能避免这些未定义的求值顺序问题呢?
其他的表达式问题
*3.13 ++i和i++有什么区别?
3.14 如果我不使用表达式的值,那我应该用i++还是++i来做自增呢?
3.15 我要检查一个数是不是在另外两个数之间,为什么if(abc)不行?
3.16 为什么如下的代码不对?inta=1000,b=1000;longintc=a*b;
3.17 为什么下面的代码总是给出0?doubledegC,degF;degC=5.0/9*(degF-32);
3.18 需要根据条件把一个复杂的表达式赋给两个变量中的一个。可以用下面这样的代码吗?((condition)?a:b)=complicated_expression;
3.19 我有些代码包含这样的表达式。a?b=c:d有些编译器可以接受,有些却不能。为什么?
保护规则
3.20 “semanticsof‘’changeinANSIC”的警告是什么意思?
3.21 “无符号保护”和“值保护”规则的区别在哪里?
第4章 指针
基本的指针应用
4.1 指针到底有什么好处?
4.2 我想声明一个指针并为它分配一些空间,但却不行。这些代码有什么问题呢?char*p;*p=malloc(10);
4.3 *p++自增p还是p所指向的变量?
指针操作
4.4 我用指针操作int数组的时候遇到了麻烦。
4.5 我有一个char*型指针碰巧指向一些int型变量,我想跳过它们。为什么((int*)p)++;这样的代码不行?
4.6 为什么不能对void*指针进行算术操作?
4.7 我有些解析外部结构的代码,但是它却崩溃了,显示出了“unalignedaccess”(未对齐的访问)的信息。这是什么意思?
作为函数参数的指针
4.8 我有个函数,它应该接受并初始化一个指针:voidf(int*ip){staticintdummy=5;ip=&dummy;}但是当我如下调用时:int*ip;f(ip);调用者的指针没有任何变化。
4.9 能否用void**通用指针作为参数,使函数模拟按引用传递参数?
4.10 我有一个函数externintf(int*);,它接受指向int型的指针。我怎样用引用方式传入一个常数?调用f(&5);似乎不行。
4.11 C语言可以“按引用传参”吗?
其他指针问题
4.12 我看到了用指针调用函数的不同语法形式。到底怎么回事?
4.13 通用指针类型是什么?当我把函数指针赋向void*类型的时候,编译通不过。
4.14 怎样在整型和指针之间进行转换?能否暂时把整数放入指针变量中,或者相反?
*4.15 我怎样把一个int变量转换为char*型?我试了类型转换,但是不行。
第5章 空指针
空指针和空指针常量
5.1 臭名昭著的空指针到底是什么?
5.2 怎样在程序里获得一个空指针?
5.3 用缩写的指针比较“if(p)”检查空指针是否有效?如果空指针的内部表达不是0会怎样?
NULL宏
5.4 NULL是什么,它是怎么定义的?
5.5 在使用非零位模式作为空指针的内部表示的机器上,NULL是如何定义的?
5.6 如果NULL定义成#defineNULL((char*)0),不就可以向函数传入不加转换的NULL了吗?
5.7 我的编译器提供的头文件中定义的NULL为0L。为什么?
5.8 NULL可以合法地用作函数指针吗?
5.9 如果NULL和0作为空指针常量是等价的,那我到底该用哪一个呢?
5.10但是如果NULL的值改变了,比如在使用非零内部空指针的机器上,用NULL(而不是0)
不是更好吗?
5.11 我曾经使用过一个编译器,不使用NULL就不能编译。
5.12 我用预处理宏#defineNullptr(type)(type*)0帮助创建正确类型的空指针。
回顾 59
5.13 这有点奇怪:NULL可以确保是0,但空(null)指针却不一定?
5.14 为什么有那么多关于空指针的疑惑?为什么这些问题如此频繁地出现?
5.15 有没有什么简单点儿的办法理解所有这些与空指针有关的东西呢?
5.16 考虑到有关空指针的所有这些困惑,要求它们的内部表示都必须为0不是更简单吗?
5.17 说真的,真有机器用非零空指针吗,或者不同类型用不同的表示?
地址0上到底有什么?
5.18 运行时的整数值0转换为指针以后一定是空指针吗?
5.19 如何访问位于机器地址0处的中断向量?如果我将指针值设为0,编译器可能会自动将它转换为非零的空指针内部表示。
5.20运行时的“nullpointerassignment”错误是什么意思?应该怎样捕捉它?
第6章 数组和指针
数组和指针的基本关系
6.1 我在一个源文件中定义了chara[6],在另一个源文件中声明了externchar*a。为什么不行?
6.2 可是我听说chara[]和char*a是等价的。是这样的吗?
6.3 那么,在C语言中“指针和数组等价”到底是什么意思?
6.4 既然它们这么不同,那为什么作为函数形参的数组和指针声明可以互换呢?
数组不能被赋值
6.5 为什么不能这样向数组赋值?externchar*getpass();charstr[10];str=getpass("Enterpassword:");
6.6 既然不能向数组赋值,那这段代码为什么可以呢?intf(charstr[]){if(str[0]=='\0')str="none";…}
6.7 如果你不能给它赋值,那么数组如何能成为左值呢?
回顾
6.8 现实地讲,数组和指针的区别是什么?
6.9 有人跟我讲,数组不过是常指针。这样讲准确吗?
6.10 我还是很困惑。到底指针是一种数组,还是数组是一种指针?
6.11 我看到一些“搞笑”的代码,包含5["abcdef"]这样的“表达式”。这为什么是合法的C语言表达式呢?
数组的指针
6.12 既然数组引用会退化为指针,如果array是数组,那么array和&array又有什么区别呢?
6.13 如何声明一个数组的指针?
动态数组分配
6.14 如何在运行时设定数组的大小?怎样才能避免固定大小的数组?
6.15 我如何声明大小和传入的数组一样的局部数组?
6.16 如何动态分配多维数组?
6.17 有个很好的窍门,如果我这样写:intrealarray[10];int*array=&realarray[-1];我就可以把“array”当作下标从1 开始的数组。
函数和多维数组
6.18 当我向一个接受指针的指针的函数传入二维数组的时候,编译器报错了。
6.19 我怎样编写接受编译时宽度未知的二维数组的函数?
6.20 我怎样在函数参数传递时混用静态和动态多维数组?
数组的大小
6.21 当数组是函数的参数时,为什么sizeof不能正确报告数组的大小?
6.22 如何在一个文件中判断声明为extern的数组的大小(例如,数组定义和大小在另一个文件中)?sizeof操作符似乎不行。
6.23 sizeof返回的大小是以字节计算的,怎样才能判断数组中有多少个元素呢?
第7章 内存分配
基本的内存分配问题
7.1 为什么这段代码不行?char*answer;printf("Typesomething:\n");gets(answer);printf("Youtyped\"%s\"\n",answer);
7.2 我的strcat()不行。我试了下面的代码:char*s1="Hello,";char*s2="world!";char*s3=strcat(s1,s2);但是我得到了奇怪的结果。
7.3 但是strcat的文档说它接受两个char*型参数。我怎么知道(空间)分配的事情呢?
*7.4 我刚才试了这样的代码:char*p;strcpy(p,"abc");它运行正常。怎么回事?为什么它没有出错?
*7.5 一个指针变量分配多少内存?
7.6 我使用fgets将文件的所有行读入一个数组,为什么读入的每一行都是最后一行的内容呢?
7.7 我有个函数,本该返回一个字符串,但当它返回调用者的时候,返回的字符串却是垃圾信息。
为什么?
*7.8 那么返回字符串或其他聚集的正确方法是什么呢?
调用malloc
7.9 为什么在调用malloc()时报出了“waring:assignmentofpointerfromintegerlacksacast”?
7.10为什么有些代码小心翼翼地把malloc返回的值转换为分配的指针类型?
*7.11 在调用malloc()的时候,错误“不能把void*转换为int*”是什么意思?
7.12 我看到下面这样的代码:char*p=malloc(strlen(s)+1);strcpy(p,s);难道不应该是malloc((strlen(s)+1)*sizeof(char))吗?
7.13 我为malloc写了一个小小的封装函数。它为什么不行?
7.14 我想声明一个指针并向它分配一些内存,但是不行。这样的代码有什么问题?char*p;*p=malloc(10);
7.15 我如何动态分配数组?
7.16 怎样判断还有多少内存?
7.17 malloc(0)是返回空指针还是指向0个字节的指针?
7.18 我听说有的操作系统在程序使用的时候才真正分配malloc申请的内存。这合法吗?
有关malloc的问题
7.19 为什么malloc返回了离谱的指针值?我的确读过问题7.9,而且也在调用之前包含了externvoid*malloc();声明。
7.20 我用一行这样的代码分配一个巨大的数组,用于数值运算:double*array=malloc(256 *256 *sizeof(double));malloc()并没有返回空指针,但是程序运行得有些奇怪,好像改写了某些内存,或者malloc()并没有分配我申请的那么多内存。为什么?
7.21 我的PC机有8兆内存。为什么我只能分配640K左右的内存?
7.22 我的应用程序非常依赖数据结构的节点的动态分配,而malloc/free的代价成了瓶颈。我该怎么做?
7.23 我的程序总是崩溃,显然发生在malloc内部的某个地方。但是我看不出哪里有问题。是malloc有bug吗?
释放内存
7.24 动态分配的内存一旦释放之后就不能再使用,是吧?
7.25 为什么在调用free()之后指针没有变空?使用(赋值、比较)释放之后的指针有多么不安全?
7.26 当我调用malloc()为一个函数的局部指针分配内存时,我还需要用free()显式地释放吗?
7.27 我在分配一些结构,它们包含指向其他动态分配的对象的指针。我在释放结构的时候,还需要释放每一个下级指针吗?
7.28 我必须在程序退出之前释放分配的所有内存吗?
7.29 我有个程序分配了大量的内存,然后又释放了。但是从操作系统看,内存的占用率却并没有变回去。
分配内存块的大小
7.30 free()怎么知道有多少字节需要释放?
7.31 那么我能否查询malloc包,以查明可分配的最大块是多大?
7.32 为什么sizeof不能告诉我它所指的内存块的大小?
其他分配函数
7.33 (像问题6.14中那样)动态分配数组之后,还能改变它的大小吗?
7.34 向realloc()的第一个参数传入空指针合法吗?你为什么要这样做?
7.35 calloc()和malloc()有什么区别?应该用哪一个?利用calloc的零填充功能安全吗?free()可以释放calloc()分配的内存吗,还是需要一个cfree()?
7.36 alloca是什么?为什么不提倡使用它? |