指针与数组的对比
c程序中,指针和数组在不少地方可以相互替换着用,让人产生一种错觉,以为两者是等价的

数组要么在静态存储区被创建(如全局数组),要么在栈上被创建。数组名对应着(而不是指向)一块内存,其地址与容量在生命周期内保持不变,只有数组的内容可以改变

指针可以随时指向任意类型的内存块,它的特征是“可变”,所以我们常用指针来操作动态内存。指针远比数组灵活,但也更危险。

修改内容
字符数组a的容量是6个字符,其内容为hello。a的内容可以修改,例如a[0]='x'.指针p指向常量字符串“world”(位于静态存储区,内容为world),常量字符串的内容是不可以被修改的。从语法上看,编译器并不觉得语句p[0]='x'有什么不妥,但是该语句企图修改常量字符串的内容而导致运行错误

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <stdio.h>
  
int main()
{
 char a[] = "hello";
 a[0] = 'x';
  
 printf("%s\n", a);
  
 char *p = "wrold";
 p[0] = 'x';
 printf("%s\n", p);
  
 return 0;
}

内容复制与比较
不能对数组名进行直接复制与比较。若想把数组a的内容复制给数组b,不能用语句 b = a,否则将产生编译错误。应该用标准库函数strcpy进行复制。同理,比较b和a的内容是否相同,应该用标准库函数strcmp进行比较

语句p = a并不能把a的内容复制指针p,而是把a的地址赋给了p。要想复制a的内容,可以先用库函数malloc为p申请一块容量为strlen(a)1个字符的内存,再用strcpy进行字符串复制。同理,语句if(p == a)比较的不是内容而是地址,应该用库函数strcmp来比较

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
  
int main()
{
 char a[] = "hello";
 char b[10];
  
 strcpy(b, a); //不能用b = a
  
 int len = strlen(a);
 char *p = (char *)malloc((len + 1) * sizeof(char));
  
 strcpy(p, a);
  
 if (strcmp(p, a) == 0) {
  printf("p和a是相等的!\n");
 }
  
 free(p);
 return 0;
}

计算内存容量
用运算符sizeof可以计算出数组的容量(字节数)。sizeof(a)的值是12.指向p指向a,但是sizeof(p)的值却是4.这是因为sizeof(p)得到的是一个指针变量的字节数(32bit机器内存地址为32bit),相当于sizeof(char *),而不是p所指的内存容量。

注意当数组作为函数的参数进行传递时,该数组自动退化为同类型的指针。不论数组a的容量是多少,sizeof(a)始终等于sizeof(char *)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
  
void funC(char *a);
  
int main()
{
 char a[] = "hello";
 char *p = a;
  
 printf("%d\n", sizeof(a)); // 6字节
 printf("%d\n", sizeof(p)); // 4字节
  
 funC(a);
 return 0;
}
  
void funC(char *a)
{
 printf("%d\n", sizeof(a)); // 4字节而不是6字节
}

运算结果:

指针参数是如何传递内存的
如果函数的参数是一个指针,不要指望用该指针去申请动态内存。示例中,Test函数的语句GetMemory(str, 200)并没有使str获得期望的内存,str依旧是NULL,为什么?

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
  
void GetMemory(char *p, int num)
{
 p = (char*)malloc(sizeof(char) * num);
}
  
char* getMemory(char *p, int num)
{
 p = (char *)malloc(sizeof(char) * num);
 return p;
}
  
int main()
{
 char *str = NULL;
 str = getMemory(str, 200);
 strcpy(str, "hello world!"); //运行错误
 printf("%s", str);
 free(str);
 return 0;
}

错误

原因
问题出在函数GetMemory中。编译器总是要为函数的每个参数制作临时副本,指针参数p的副本是_p,编译器使_p = p.如果函数体内的程序修改了_p的内容,就导致参数p的内容作相应的修改。这就是指针可以用作输出参数的原因。在本例中,_p申请了新的内存,只是把_p所指的内存地址改变了,但是p丝毫未变。所以函数GetMemory并不能输出任何东西。事实上,每执行一次GetMemory就会泄漏一块内存,因为没有用free释放内存

改进
我们可以用函数返回值来传递动态内存,这种方法更简单,见示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
  
void GetMemory(char *p, int num)
{
 p = (char*)malloc(sizeof(char) * num);
}
  
char* getMemory(char *p, int num)
{
 p = (char *)malloc(sizeof(char) * num);
 return p;
}
  
int main()
{
 char *str = NULL;
 str = getMemory(str, 200);
 strcpy(str, "hello world!"); //运行错误
 printf("%s\n", str);
 free(str);
 return 0;
}

注意:
用函数返回值来传递动态内存这种方法虽然好用,但是常常有人把return语句用错了。这里强调不要用return语句返回指向”栈内存“的指针,因为该内存在函数结束时自动消亡。

示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
  
void GetMemory(char *p, int num)
{
 p = (char*)malloc(sizeof(char) * num);
}
  
char* getMemory(char *p, int num)
{
 p = (char *)malloc(sizeof(char) * num);
 return p;
}
  
char* getArray(void)
{
 char p[] = "hello world!";
 return p; // 编译器提出警告
}
int main()
{
 char *str = NULL;
 str = getArray();
 printf("%s\n", str); // str指向的内容是垃圾
 free(str);
 return 0;
}

杜绝“野指针”
"野指针"不是NULL指针,是指向“垃圾”内存的指针。人们一般不会错用NULL指针,因为用if语句很容易判断。但是“野指针”是很危险的,if语句对它不起作用。“野指针”的成因主要有两种:

指针变量没有初始化。任何指针变量刚被创建时不会自动成为NULL指针,它的缺省值是随机的,它会乱指一气。所以,指针变量在创建的同时应该被初始化,要么将指针设置为NULL,要么让它指向合法的内存,例如:

1
2
char *p = NULL;
char *str = (char *)malloc(sizeof(char) * 100);

指针p被free或者delete之后,没有置为NULL,让人误以为p是个合法的指针
    指针操作超越了变量的作用范围

内存耗尽怎么办
如果在申请动态内存时找不到足够大的内存块,malloc函数将返回NULL指针,宣告内存申请失败。通常有三种方式处理“内存耗尽”问题

判断指针是否为NULL,如果是则马上用return语句终止本函数。例如:

1
2
3
4
5
6
7
char* getPoint()
{
 char *p = malloc(sizeof(char) * 100);
 if (p == NULL) {
  return null;
 }
}

判断指针是否为NULL,如果是则马上用exit(1)终止整个程序的运行(我经常用也是推荐做法):

1
2
3
4
5
6
7
char* getPoint()
{
 char *p = malloc(sizeof(char) * 100);
 if (p == NULL) {
  exit(1);
 }
}

为new和malloc设置异常处理函数

[转]C语言常见错误总结1的更多相关文章

  1. C语言常见错误中英文对照表

    C语言常见错误中英文对照表(网络搜索及经验积累不断更新中) 常见错误中英文对照表 fatal error C1003:  error count exceeds number; stopping co ...

  2. C语言常见错误笔记

    1. 职业化的程序员起码要具备两点: 1)基本的软件技能 2)不犯低级的错误 2. 修改函数的形参是没用的,函数本身占用的存储单元在堆栈中分配,入口参数的值会在函数入口处拷贝到堆栈中,一旦函数返回,其 ...

  3. 【原】GO 语言常见错误

    1. Scan error on column index 4: converting string "" to a int: strconv.ParseInt: parsing ...

  4. C语言初学者代码中的常见错误与瑕疵(23)

    见:C语言初学者代码中的常见错误与瑕疵(23)

  5. 一个超复杂的间接递归——C语言初学者代码中的常见错误与瑕疵(6)

    问题: 问题出处见 C语言初学者代码中的常见错误与瑕疵(5) . 在该文的最后,曾提到完成的代码还有进一步改进的余地.本文完成了这个改进.所以本文讨论的并不是初学者代码中的常见错误与瑕疵,而是对我自己 ...

  6. C语言初学者代码中的常见错误与瑕疵(5)

    问题: 素数 在世博园某信息通信馆中,游客可利用手机等终端参与互动小游戏,与虚拟人物Kr. Kong 进行猜数比赛. 当屏幕出现一个整数X时,若你能比Kr. Kong更快的发出最接近它的素数答案,你将 ...

  7. C语言初学者代码中的常见错误与瑕疵(19)

    见:C语言初学者代码中的常见错误与瑕疵(19)

  8. C语言初学者代码中的常见错误与瑕疵(14)

    见:C语言初学者代码中的常见错误与瑕疵(14) 相关链接:http://www.anycodex.com/blog/?p=87

  9. 分数的加减法——C语言初学者代码中的常见错误与瑕疵(12)

    前文链接:分数的加减法——C语言初学者代码中的常见错误与瑕疵(11) 重构 题目的修正 我抛弃了原题中“其中a, b, c, d是一个0-9的整数”这样的前提条件,因为这种限制毫无必要.只假设a, b ...

随机推荐

  1. 前端开发:CSS3

    CSS介绍: CSS能够使页面具有美观一致的效果,并且能够让内容与格式分离,利于扩展 所以,CSS解决了下面两个问题: 1. 将HTML页面的内容与格式分离: 2. 提高web开发的工作效率. CSS ...

  2. 2018/2/27 Activiti教程之创建流程篇(与Springboot整合版)一

    因为电脑还在托运中,现在手上这台垃圾电脑实在是没法玩微服务,所以趁着这两天玩玩Activiti吧. 说实话,在学习Activiti中走了N多弯路,最大的原因就是网上没有一个完整(好)的教程,甚至连官方 ...

  3. [NOIP2004] 普及组

    不高兴的津津 纯模拟 #include<cmath> #include<cstdio> #include<iostream> using namespace std ...

  4. NOIP2012提高组D1T3 开车旅行

    n<=100000个山,每个山有高度,从一个山到另一个山代价为高度差,有A和B两人一起开车,A每次选前进方向的次近山,B选最近,保证山高度不同且如果代价相同的山低的代价算小,每次旅行先A走,然后 ...

  5. jsp内置对象之response、out、config、exception、pageContext。

    本文是对Jsp内置对象的response.out.config.exception.pageContext知识点的详细总结. response对象 Response内置对象和request内置对象是相 ...

  6. Ubuntu 16.04升级4.7.0内核后导致Compiz奔溃,问题:compiz[4852]: segfault at 48 ip 00007f88cae087f0 sp 00007ffce354c268 error 4 in libscale.so

    由于硬件的驱动支持问题,升级4.7.0的内核主要是为了能使用Intel HD Graphics 630驱动,但是也出现了相关问题,比如Compiz的特效导致桌面上如果有多个相同程序启动,然后再次点击时 ...

  7. windows 7 忘記密碼,用“带命令行的安全模式”

    net user administrator /active:yes net user tester /add net localgroup administrators tester /add

  8. 文本分类——NaiveBayes

    前面文章已经介绍了朴素贝叶斯算法的原理,这里基于NavieBayes算法对newsgroup文本进行分类測试. 文中代码參考:http://blog.csdn.net/jiangliqing1234/ ...

  9. JavaSE入门学习5:Java基础语法之keyword,标识符,凝视,常量和变量

    一keyword keyword概述:Java语言中有一些具有特殊用途的词被称为keyword.keyword对Java的编译器有着特殊的意义.在程 序中应用时一定要谨慎. keyword特点:组成k ...

  10. cmd启动Oracle服务和监听服务

    启动数据库服务 net start oracleserviceorcl 启动数据库监听 lsnrctl start