`

c预言的一些经典问题

    博客分类:
  • c
 
阅读更多
1.这样的初始化有什么问题?char *p = malloc(10); 编译器提示非法初始式 云云。
 
答:这个声明是静态或非局部变量吗?函数调用只能出现在自动变量(即局部非静态变量) 的初始式中。
因为静态变量的地址必须在编译的过程中就确定下来而malloc()申请的内存地址是在运行时确定的。
 
2. *p++ 自增p 还是p 所指向的变量?
 
答:后缀++ 和-- 操作符本质上比前缀一目操作的优先级高, 因此*p++ 和*(p++) 等价, 它自增p 并返回p 自增之前所指向的值。
要自增p 指向的值, 使用(*p)++, 如果副作用的顺序无关紧要也可以使用++*p。
 
3 我有一个char * 型指针正巧指向一些int 型变量, 我想跳过它们。为什么如下的代码((int *)p)++; 不行?
 
答:在C 语言中, 类型转换意味着把这些二进制位看作另一种类型, 并作相应的对待; 这是一个转换操作符, 
根据定义它只能生成一个右值(rvalue)。而右值既不能赋值, 也不能用++ 自增。(如果编译器支持这样的扩展, 
那要么是一个错误, 要么是有意作出的非标准扩展。) 要达到你的目的可以用:p = (char *)((int *)p + 1);
或者,因为p 是char * 型, 直接用p += sizeof(int);
 
4.空指针和未初始化的指针是一回事吗?
 
答:空指针在概念上不同于未初始化的指针:空指针可以确保不指向任何对象或函数;
 而未初始化指针则可能指向任何地方。
 
5.我可以用0来表示空指针吗?
 
答:根据语言定义, 在指针上下文中的常数0 会在编译时转换为空指针。也就是说, 在初始化、赋值或比较的时候, 
如果一边是指针类型的值或表达式, 编译器可以确定另一边的常数0 为空指针并生成正确的空指针值。因此下边的代码段完全合法:
char *p = 0;
if(p != 0)
 
然而, 传入函数的参数不一定被当作指针环境, 因而编译器可能不能识别未加修饰的0 表示 指针。
在函数调用的上下文中生成空指针需要明确的类型转换,强制把0 看作指针。
例如, Unix 系统调用execl 接受变长的以空指针结束的字符指针参数。它应该如下正确调用:
execl("/bin/sh", "sh", "-c", "date", (char *)0);
如果省略最后一个参数的(char *) 转换, 则编译器无从知道这是一个空指针,从而当作一个0 传入。(注意很多Unix 手册在这个例子上都弄错了。)
 
 
摘要:
==========================|=============================
|| 可以使用未加修饰的0 | 需要显示的类型转换 ||
||------------------------|---------------------------||
|| *初始化 | *函数调用, 作用域内无原型 ||
|| *赋值 | *变参函数调用中的可变参数 ||
|| *比较 | ||
|| *固定参数的函数调用 | ||
|| 且在作用域内有原型 | ||
==========================|=============================
 
有两条简单规则你必须遵循:
1) 当你在源码中需要空指针常数时, 用0 或NULL。
2) 如果在函数调用中0 或NULL 用作参数, 把它转换成被调函数需要的指针类型
 
6. 既然数组引用会蜕化为指针, 如果arr 是数组, 那么arr 和&arr 又有什么区别呢?
 
答:区别在于类型:
在标准C 中, &arr 生成一个T 型数组 的指针, 指向整个数组。
在所有的C 编译器中, 对数组的简单引用(不包括& 操作符)生成一个T 的指针类型的指针, 指向数组的第一成员。
 
7. 我如何声明一个数组指针?
 
答:通常, 你不需要。当人们随便提到数组指针的时候, 他们通常想的是指向它的第一个元素的指针。
考虑使用指向数组某个元素的指针, 而不是数组的指针。类型T 的数组蜕变成类型T 的指针, 这很方便; 
在结果的指针上使用下标或增量就可以访问数组中单独的成员。而真正的数组指针, 在使用下标或增量时, 会跳过整个数组, 
通常只在操作数组的数组时有用 如果还有一点用的话。如果你真的需要声明指向整个数组的指针, 
使用类似int (*ap)[N];这样的声明。其中N 是数组的大小。如果数组的大小未知, 原则上可以省略N, 但是这样生成的类型, 
指向大小未知的数组的指针, 毫无用处。
 
8.当我向一个接受指针的指针的函数传入二维数组的时候, 编译器报错了,这是怎么回事?
 
答:数组蜕化为指针的规则不能递归应用。数组的数组(即C 语言中的二维数组) 蜕化为数组的指针, 而不是指针的指针。
数组指针常常令人困惑, 需要小心对待; 如果你向函数传递二位数组:
int array[NROWS][NCOLUMNS];
f(array);
那么函数的声明必须匹配:
void f(int a[][NCOLUMNS])
{ ... }
或者
void f(int (*ap)[NCOLUMNS]) /* ap 是个数组指针*/
{ ... }
在第一个声明中, 编译器进行了通常的从数组的数组 到数组的指针 的隐式转换; 第二种形式中的指针定义显而易见。
因为被调函数并不为数组分配地址,所以它并不需要知道总的大小, 所以行数NROWS 可以省略。但数组的宽度依然重要,
所以列维度NCOLUMNS (对于三维或多维数组, 相关的维度) 必须保留。
如果一个函数已经定义为接受指针的指针, 那么几乎可以肯定直接向它传入二维数组毫无意义。
 
 
9. 我的strcat() 不行.我试了char *s1 = "Hello, "; char *s2 = "world!"; 
char *s3 = strcat(s1, s2); 但是我得到了奇怪的结果。
 
答:这里主要的问题是没有正确地为连接结果分配空间。C 没有提供自动管理的字符串类型。
C 编译器只为源码中明确提到的对象分配空间(对于字符串, 这包括字符数组和串常量)。
程序员必须为字符串连接这样的运行期操作的结果分配足够的空间, 
 
常可以通过声明数组或调用malloc() 完成。strcat() 不进行任何分配; 第二个串原样不动地附加在第一个之后。
因此, 一种解决办法是把第一个串声明为数组:
char s1[20] = "Hello, ";
由于strcat() 返回第一个参数的值, 本例中为s1, s3 实际上是多余的; 在strcat() 调用之后, s1 包含结果。
提问中的strcat() 调用实际上有两个问题: s1 指向的字符串常数, 除了空间不足以放入连接的字符串之外, 
甚至都不一定可写。
 
 
10. 那么返回字符串或其它集合的正确方法是什么呢?
 
答:返回指针必须是静态分配的缓冲区, 或者调用者传入的缓冲区, 
或者用malloc() 获得的内存, 但不能是局部(自动) 数组。
 
11. 我有个程序分配了大量的内存, 然后又释放了。但是从操作系统看,内存的占用率却并没有回去。
 
答:多数malloc/free 的实现并不把释放的内存返回操作系统, 而是留着供同一程序的后续malloc() 使用。
 
12. calloc() 和malloc() 有什么区别?利用calloc 的零填充功能安
全吗?free() 可以释放calloc() 分配的内存吗, 还是需要一个cfree()?
 
答:calloc(m, n) 本质上等价于:
p = malloc(m * n);
memset(p, 0, m * n);
填充的零是全零, 因此不能确保生成有用的空指针值或浮点零值free() 
 
可以安全地用来释放calloc() 分配的内存。
 
13. 我认为我的编译器有问题: 我注意到sizeof('a') 是2 而不是1 (即,不是sizeof(char))。
 
答:可能有些令人吃惊, C语言中的字符常数是int 型, 因此sizeof('a') 是sizeof(int),这是另一个与C++ 不同的地方。
 
14. 为什么声明extern int f(struct x *p); 报出了一个奇怪的警告信息结构x 在参数列表中声明?
 
答:与C 语言通常的作用范围规则大相径庭的是, 在原型中第一次声明(甚至提到)的结构不能和同一源文件中的其它结构兼容, 
它在原型的结束出就超出了作用范围。要解决这个问题, 在同一源文件的原型之前放上这样的声明:
struct x;
它在文件范围内提供了一个不完整的结构x 的声明, 这样, 后续的用到结构x的声明至少能够确定它们引用的是同一个结构x。
struct x;

static void testStruct(struct x *p);

typedef struct {
	int size;
	char *buffer;
}x;
 
 
 
15. 我不明白为什么我不能象这样在初始化和数组维度中使用常量:const int n = 5; int a[n];
 
:const 限定词真正的含义是 只读的; 用它限定的对象是运行时 (同常) 不能被赋值的对象。
因此用 const 限定的对象的值并不完全是一个真正的常量。
在这点上 C 和 C++ 不一样。如果你需要真正的运行时常量, 使用预定义宏 #define(或enum)。
注:在c++中可以使用const int n = 5; int a[n];
 
16. 我能否把 main() 定义为 void, 以避免扰人的 main无返回值警告?
 
答:不能。main() 必须声明为返回 int, 且没有参数或者接受适当类型的两个参数。
如果你调用了 exit() 但还是有警告信息, 你可能需要插入一条冗余的 return语句
(或者使用某种 未到达指令, 如果有的话)。
很多书不负责任地在例子中使用 void main(), 并宣称这样是正确的。
但他们错了。
 
 
17. #pragma 是什么, 有什么用?
 
答:#pragam 指令提供了一种单一的明确定义的 救生舱, 可以用作各种 (不可移植的) 实现相关的控制和扩展:
源码表控制、结构压缩、警告去除 (就像 lint 的老 /* NOTREACHED */注释), 等等。
 
18. #pragma once 是什么意思?我在一些头文件中看到了它。
 
答:这是某些预处理器实现的扩展用于使头文件自我识别; 它跟#ifndef技巧等价, 不过移植性差些。
  

 

分享到:
评论

相关推荐

    C语言经典问题

    《C语言经典问题》这个主题包含了C语言编程中的一些历史著名问题和数据结构的应用,旨在帮助已有一定C语言基础并且数学功底扎实的学习者提升技能。以下是对这些知识点的详细阐述: 1. **八皇后问题**:这是一个著名...

    C语言经典题目900例

    《C语言经典题目900例》是一本专注于C语言编程实践的资源集合,它包含了丰富的算法实例,旨在帮助初学者和进阶者巩固和提升C语言编程技能。在这个压缩包中,我们可以找到多个文档,可能包含了900个不同的编程题目及...

    c语言经典4本书 精通c语言必备的书籍

    通过阅读这四本书,你可以全面而深入地了解C语言,从基础知识到高级技巧,再到实践中的问题解决。它们将帮助你成为一名真正的C语言专家,掌握指针操作、内存管理、程序优化等关键技能,为你的编程生涯奠定坚实的基础...

    C语言超级经典400道题目

    根据提供的信息,我们可以总结出这份文档是关于C语言的一些经典问题及解答。下面将对给出的部分内容中的知识点进行详细解析。 ### C语言经典题目解析 #### 第1题 **题目:** C语言的基本结构有哪些? - **选项:** ...

    C语言 四大关键问题

    例如,可以挑选一些经典的算法问题或者常见的应用场景,尝试用C语言实现,并仔细分析其中涉及到的关键技术点。这种方式不仅能够帮助巩固已学知识,还能提高解决问题的能力。 #### 四、通过阅读大量程序达到熟练编写...

    C语言开发经典程序,帮助你熟练掌握C语言

    这个资源包可能包含了一些经典的算法示例,如排序算法(冒泡排序、选择排序、插入排序、快速排序等)、查找算法(线性查找、二分查找等)、字符串处理、动态规划问题等。通过实际编写和分析这些算法,可以锻炼逻辑...

    C语言常见问题,应有尽有的问题归纳。

    以下是一些关于C语言常见问题的详细解释: 1. **C语言的约定**: - **大小写敏感**:C语言是大小写敏感的,关键字、变量名、函数名的大小写必须严格区分。 - **注释**:单行注释以`//`开始,多行注释以`/*`开始并...

    C语言经典100例

    从给定文件的内容中,我们可以提取出一些与C语言编程相关的知识点。以下是针对这三道示例题目的详细知识点。 1. 绘制余弦曲线 知识点:C语言绘图、数组的使用、数学函数的使用、算法设计、屏幕输出控制。 在这个...

    C语言经典程序C语言经典程序

    "C语言经典程序"的标题和描述暗示了我们将会探讨C语言的一些核心概念、经典算法以及常见的编程实践。 首先,C语言是计算机科学教育的基础,它的语法结构清晰,适合学习底层操作系统和硬件交互。C语言经典程序往往...

    一些简单经典的C语言代码

    掌握链表的使用对于理解和解决复杂问题至关重要。 "数组"是C语言的基础,它允许我们存储同一类型的数据集合。数组可以一维、二维或多维,理解数组的索引、遍历以及动态内存分配等概念对编写高效代码非常有帮助。 ...

    100例c语言经典题目

    这些题目涵盖了C语言的基础知识,如变量、数据类型、运算符、控制结构、函数等等,同时也涵盖了一些高级的编程技巧,如数组、指针、结构体等等。 这100个C语言经典题目是一个非常有价值的学习资源,对于任何想提高...

    C语言经典编程100例

    首先,我们来看看其中的一些经典案例: 1. **约瑟夫问题**:这是一个著名的理论问题,涉及循环链表和计数器的操作。通过模拟古犹太人约瑟夫的故事,学习者可以理解和掌握如何处理循环数据结构和执行复杂逻辑。 2. ...

    C语言经典学习资料

    "C语言经典学习资料"这个压缩包包含了一些经典的C语言教材,旨在帮助初学者或有一定经验的开发者深入理解C语言的核心概念和高级特性。这些书籍被誉为"C语言三剑客",包括"The C Programming Language"、"Pointers on...

    C语言经典算法 案例

    下面我们将逐一探讨其中的一些经典算法。 1. **汉诺塔**:这是一个著名的递归问题,目标是将一堆圆盘从一根柱子移动到另一根柱子,遵循每次只能移动一个圆盘且大盘不能放在小盘之上的规则。通过汉诺塔问题,我们...

    c语言解决迷宫问题

    标题 "C语言解决迷宫问题" 指的是使用C编程语言来设计算法解决一个经典的计算机科学问题——迷宫寻路。在这个问题中,我们通常有一个二维数组或矩阵表示的迷宫,其中每个元素代表一个节点,节点可能是通路(通常是...

    c语言入门经典2008

    通过《C语言入门经典2008》,读者不仅可以学习到C语言的基本语法,还能了解到良好的编程习惯和问题解决策略。配合书中的示例代码和练习,可以有效提高编程技能。特别地,"Beginning Microsoft Visual C++ 2008.pdf" ...

Global site tag (gtag.js) - Google Analytics