前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C语言书籍——A/陷阱之处

C语言书籍——A/陷阱之处

作者头像
用户11015888
发布2024-03-11 20:06:23
800
发布2024-03-11 20:06:23
举报
文章被收录于专栏:csdncsdn

词法陷阱

一、= 不同于 ==

在 if 判断时容易出错。

=:赋值运算,a=3;表示的是将3赋值给a变量。 ==:比较运算,a==3;表示判断a是否等于3,若等于则返回1,否则返回0。

二、& 和 | 不同于 && 和 ||

这是两个 逻辑操作符与位操作符的区别。

详见《逻辑操作符》《位操作符》

三、词法分析中的“ 贪心法 ”

代码语言:javascript
复制
a---b;等于a-- -b;  //先a-b,再a--
不等于a- --b;  //--b先做自减运算,再a-b

四、整型常量

如果一个整型常量的第一个数字是0,则该常量会被当做是八进制数,因此10和010代表是分别是十进制的10和十进制的8。

五、字符与字符串

单引号引起来的字符代表的是该字符的ASCII码值;

双引号引起来的字符串代表的是一个指向无名数组的起始字符的指针,该数组被双引号之间的字符以及一个额外’\0’(字符串标志)初始化。

在双引号引起来的字符串中,注释符号/*属于字符串的一部分;在注释中出现的双引号“”又属于注释的一部分。

因此,语句:printf("The world") 和以下语句是等价的:

代码语言:javascript
复制
char str[]= {'T','h','e',' ','w', 'o','r','l','d','\n'};
printf(str);

语法陷阱

一、理解函数声明

指针例题

二、运算符的优先级问题

详见 优先级

三、注意作为语句结束标志的分号

代码语言:javascript
复制
//代码1
if(x[i] > b);
	b = x[i];
//代码2
if(x[i] > b){}
	b = x[i];

//代码3
if(x[i] > b)
	b = x[i];

一个分号就代表一个语句的结束。代码1与代码2是等价,if和赋值语句是两个独立的语句;而第三句中赋值语句在if中。

四、swith语句

语句详解

五、函数调用

C语言要求:在函数调用时即使函数不带参数也应该包括参数列表。因此,如果f是一个函数,

代码语言:javascript
复制
f();

是一个函数调用语句。

代码语言:javascript
复制
f;

却是一个什么也不做的语句。更精确地说,这个语句计算函数f的地址,却并不调用该函数。

六、“悬挂”else引发的问题

代码语言:javascript
复制
if (x == 0)
		if (y == 0) error();
	else {
			Z = X + Y;
			f(&z);
		 }

两种解读:

解读一:

代码语言:javascript
复制
if (x == 0) 
{
	if (y == 0) error();
	else 
	{
		Z = X + Y;
		f(&z);
	}
}

解读二:

代码语言:javascript
复制
if (x == 0)
{
	if (y == 0) error();
}
	else
	{
		Z = X + Y;
		f(&z);
	}

语义陷阱

一、指针与数组

深入理解指针系列文章

二、非数组的指针

三、作为参数的数组声明

四、避免“ 举隅法 ”

常见错误解释:避免以整体代表部分,或者以部分代表整体。

常见错误:混淆指针与指针所指向的数据。

代码语言:javascript
复制
char *p,*q;
p ="xyz";

上面的赋值语句使得p的值就是字符串"xyz",然而实际情况并不是这样,实际上,p的值是一个指向由'x'、'y'、’z"和\0 4个字符组成的数组的起始元素的指针。

因此,如果我们执行下面的语句:q=p;

p和q现在是两个指向内存中同一地址的指针,但这个赋值语句并没有同时复制内存中的字符。

复制指针并不同时复制指针所指向的数据,因此,当我们执行完下面的语句之后:q[1]='y';q所指向的内存现在存储的是字符串'xyz’。因为p和q所指向的是同一块内存,所以p指向的内存中存储的当然也是字符串’xyz'。

五、空指针并非空字符串

在C语言中将一个整数转换为一个指针,最后得到的结果都取决于具体的C编译器实现。这个特殊情况就是常数0,编译器保证由0转换而来的指针不等于任何有效的指针。

出于代码文档化的考虑,常数0这个值经常用一个符号来代替:#define NULL 0

当然无论是直接用常数0,还是用符号ULL,效果都是相同的。

需要记住的重要一点是,当常数0被转换为指针使用时,这个指针绝对不能被解除引用(dereference)。换句话说,当我们将0赋值给一个指针变量时,绝对不能企图使用该指针所指向的内存中存储的内容。

合法格式:

代码语言:javascript
复制
if(p == (char *) 0)

非法格式:

代码语言:javascript
复制
if(strcmp(p,(char*)0) == 0)

原因在于库函数 strcmp 的实现中会包括查看它的指针参数所指向内存中的内容的操作。

六、边界计算与不对称计算

七、求职顺序

八、逻辑运算符&&、| 和 !

九、整数溢出

C语言中存在两类整数算术运算,有符号运算与无符号运算。

1、两个无符号算术运算中,没有所谓的“溢出”一说:所有的无符号运算都是以2的n次方为模,这里n是结果中的位数。

2、一个操作数是有符号整数,另一个是无符号整数,那么有符号整数会被转换为无符号整数,“溢出”也不可能发生。

3、当两个操作数都是有符号整数时,“溢出”就有可能发生,而且“溢出”的结果是术定义的。当一个运算的结果发生“溢出”时,作出任何假设都是不安全的。

正确的方式是将a和b都强制转换为无符号整数:

代码语言:javascript
复制
if ((unsigned)a + (unsigned)b > INT_MAX)
		complain();

此处的 INT_MAX是一个已定义常量,代表可能的最大整数值。ANSIC标准在<limits.h>中定义了INTMAX:如果是在其他C语言实现上,读者也许需要自己重新定义。

代码语言:javascript
复制
//不需要用到无符号算术运算的另一种可行方法是:
if(a >INT_MAX - b)
    complain();

十、为函数main提供返回值

函数 main 与其他任何函数一样,如果并未显式声明返回类型,那么函数返回类型就默认为是整型。但是这个程序中并没有给出任何返回值。

通常说来,这不会造成什么危害。一个返回值为整型的函数如果返回失败,实际上是隐含地返回了某个“垃圾”整数。只要该数值不被用到,就无关紧要。

严格说来,我们前面的最简单的C程序应该像下面这样编写代码:

代码语言:javascript
复制
int main()
{
	//语句

	return 0;
}
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2024-03-08,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 词法陷阱
    • 一、= 不同于 ==
      • 二、& 和 | 不同于 && 和 ||
        • 三、词法分析中的“ 贪心法 ”
          • 四、整型常量
            • 五、字符与字符串
            • 语法陷阱
              • 一、理解函数声明
                • 二、运算符的优先级问题
                  • 三、注意作为语句结束标志的分号
                    • 四、swith语句
                      • 五、函数调用
                        • 六、“悬挂”else引发的问题
                        • 语义陷阱
                          • 一、指针与数组
                            • 二、非数组的指针
                              • 三、作为参数的数组声明
                                • 四、避免“ 举隅法 ”
                                  • 五、空指针并非空字符串
                                    • 六、边界计算与不对称计算
                                      • 七、求职顺序
                                        • 八、逻辑运算符&&、| 和 !
                                          • 九、整数溢出
                                            • 十、为函数main提供返回值
                                            领券
                                            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档