c++指针的应用及注意问题

  指针是c/c++学习一个比较让人头痛的问题,在程序设计过程中,指针也往往是产生隐含bug的原因。下面就来谈谈指针的应用以及需要注意的一些问题,里面也许就有你平时没有注意到的问题,希望能帮助各位读者理解好指针。

  一、我们先来回忆一下指针的概念吧,方便下面的介绍

  指针是存放地址值的变量或者常量。例如:int a=1;&a就表示指针常量(“&”表示取地址运算符,也即引用)。int *b,b表示的是指针变量(注意,是b表示指针变量而不是*b),*表示要说明的是指针变量。大家注意int *b[2]和int(*b)[2]是不同的,int *b表示一个指针数组,而int (*b)[2]表示含有两个元素的int指针,这里要注意运算优先级问题,有助于理解指针问题。

  在这里大概介绍基本概念就够了,至于具体使用方法,如赋值等,很多书都有介绍,我就不多说了。

  二、应用以及注意的问题

  1、 理解指针的关键所在——对指针类型和指针所指向的类型的理解

  ①、 指针类型:可以把指针名字去掉,剩下的就是这个指针

  例如:int *a;//指针类型为int *

  int **a;//指针类型为int **

  int *(*a)[8];//指针类型为 int *(*)[8]

  ②、 指针所指向的类型:是指编译器将把那一片内存所看待成的类型。这里只要把指针声明语句中的指针名字和名字右边的“*”号去掉就可以了,剩下的就是指针所指向的类型。

  我之所以把他们放在第一位,是因为弄清楚他们是学c/c++指针的重点,正确理解他们才能使你打好c/c++的编程基础。

  2、 指针的应用——传递参数。

  其实它可以相当于隐式的返回值,这就比return的方法更加灵活了,可以返回更多的值,看看下面的例子自然就明白了:

  #include "iostream.h"

  void example(int *a1,int &b1,int c1)

  {

  *a1*=3;

  ++b1;

  ++c1;

  }

  void main()

  {

  int *a;

  int b,c;

  *a=6;

  b=7;c=10;

  example(a,b,c);

  cout 《"*a="《*a<

  cout 《"b="<

  cout 《"c="<

  }

  输出:*a=18

  b=8

  c=10

  注意到没有,*a和b的值都改变了,而c没有变。这是由于a1是指向*a(=6)的指针,也即与a是指向同一个地址,所以当a1指向的值改变了,*a的值也就改变了。在函数中的参数使用了引用(int &b1),b1是b的别名,也可以把它当作特殊的指针来理解,所以b的值会改变。函数中的参数int c1只是在函数中起作用,当函数结束时候便消失了,

  所以在main()中不起作用。

  3、 关于全局变量和局部变量的一个问题

  先不废话了,先看看程序:

  #include “iostream.h”

  int a=5;

  int *example1(int b)

  {

  a+=b;

  return &a;

  }

  int *example2(int b)

  {

  int c=5;

  b+=c;

  return &b;

  }

  void main()

  {

  int *a1=example1(10);

  int *b1=example2(10);

  cout 《”a1=”《*a1<

  cout 《”b1=”《*b1<

  }

  输出结果:

  a1=15

  b1=4135

  *b1怎么会是4135,而不是15呢?是程序的问题?没错吧?

  由于a是全局变量,存放在全局变量的内存区,它一直是存在的;而局部变量则是存在于函数的栈区,当函数example2()调用结束后便消失,是b指向了一个不确定的区域,产生指针悬挂。

  下面是对example1()和example2()的反汇编(用TC++ 3.0编译):

  example1():

  push bp;入栈

  mov bp,sp

  mov ax,[bp+04];传递参数

  add [00AA],ax;相加

  mov ax,00AA ;返回了结果所在的地址

  pop bp;恢复栈,出栈

  ret;退出函数

  example2():

  push bp;入栈

  mov bp,sp

  sub sp,02

  mov word ptr [bp-02],0005

  mov ax,[bp-02];传递参数

  add [bp+04],ax;相加

  lea ax,[bp+04];问题就出在这里

  mov sp,bp

  pop bp;恢复栈,出栈

  ret;退出函数

  对比之后看出来了吧?ax应该是存储的是结果的地址。而在example2()中,返回的却是[bp+04]的内容,因此指针指向了一个不确定的地方,由此产生的指针悬挂。example1()中,ax返回了正确的结果的地址。

  4、 内存问题:使用指针注意内存的分配和边界。

  使用指针过程中应该给变量一个适当的空间,以免产生不可见的错误。

  请看以下代码:

  #include “iostream.h”

  void main()

  {

  char *a1;

  char *a2;

  cin 》a1;

  cin 》a2;

  cout 《”a1=”<

  cout 《”a2=”<

  }

  输入:abc

  123

  输出:

  a1=123

  a2=

  Null pointer assignment

  指针指向了“空”。解决办法就是分配适当的内存给这两个字符串。修正后的代码

  如下:

  #include “iostream.h”

  void main()

  {

  char *a1;

  char *a2;

  a1=new char [10];

  a2=new char [10];

  cin 》a1;

  cin 》a2;

  cout 《”a1=”< // cout 《”a2=”<

  delete(a1);注意,别忘了要释放内存空间

  delete(a2);

  }

  到此就能输出正确的结果了。

  分配了适当的内存之后要注意释放内参空间,同时还应该注意不要超出所分配的内存的大小,否则会有溢出现象产生,导致不可预料的结果。

  5、 关于特殊的指针——引用

  引用有时候应用起来要比指针要灵活,用它做返回的时候是不产生任何变量的副本的这样减小了内存的占用,提高执行的速度。引用使用起来要比指针好理解,比较直观。当引用作为参数时,不会改变参数的地址,因此可以作为左值。

  下面请看一个例子:

  #include “iostream.h”

  char ch[5]=”ABCD”;

  char &example(int b)

  {

  return ch;

  }

  void main()

  {

  cout 《”ch=”<

  example(2)=”c”;

  cout《”ch=”<

  }

  ch=ABCD

  ch=ABcD

  在实际编程过程中,可以灵活地引用或指针,尽量提高程序的可读性和执行效率。

  三、小结

  指针是学习c/c++的重点难点,主要原因是指针较为抽象,不容易理解。使用指针千万要明白让指针指向什么地方,如何让指针指向正确的地方。在深入系统底层之中需要应用到大量的指针,因此需要理解好指针的基本概念,例如:指针类型和指针所指向的类型。平时应该对留心观察,了解程序的工作过程,必要时候可以对程序进行反汇编,加深对指针的理解,这种方法同样适合学别的编程方面的知识。

  四、结束

  指针的应用是很广泛的,利用指针可以做很多事情,要成为一个好的程序员,必须对指针有比较深刻的了解。写本文的目的在于让大家对指针有更深一层的了解,提高指针的应用能力,内容大都是我在实际编程中遇到的问题。相信能给大家一定的帮助。

时间: 2025-01-20 14:14:16

c++指针的应用及注意问题的相关文章

C语言指针的长度和类型

如果考虑应用程序的兼容性和可移植性,指针的长度就是一个问题,在大部分现代平台上,数据指针的长度通常是一样的,与指针类型无关,尽管C标准没有规定所有类型指针的长度相同,但是通常实际情况就是这样.但是函数指针长度可能与数据指针的长度不同. 指针的长度取决于使用的机器和编译器,例如:在现代windows上,指针是32位或是64位长 测试代码: #include<stdio.h> #include<math.h> #include<stdlib.h> #include<s

指向多维数组的指针变量

问题描述 指向多维数组的指针变量 #include int main() { int a[3][4]={1,2,3,4,5,6,7,8,9,10,11,12}; int *p; for(p=a[0];p<a[0]+12;p++) { if((p-a[0])%4==0) printf(" "); printf("%4d",*p); } } 把for(p=a[0];p<a[0]+12;p++)改成for(p=a[0];p<a+3;p++)为什么是正确的

指针-c语言中字符数组初始化问题

问题描述 c语言中字符数组初始化问题 字符数组初始化1: char str[]=""123"";//不报错2: char str[4]; str=""123"";//不能将const char[4] to char[4]字符指针初始化1: char *str=""123"";//不报错2: char *str; str=""123"";//不报错求

Ubuntu系统鼠标指针上下跳动该怎么办?

  可能不少网友都遇到了这个问题,就是鼠标指针一闪一闪的,一会而消失一会儿出现的样子.这个经验告诉你怎么做可以消除这个问题,保证99.5%有效率. 1.打开系统设置 2.点击显示 3.然后切换到"未知显示器",你会发现未知显示器是开启的状态 4.现在你只需要把未知显示器设置为"关闭"状态就可以了.然后点击应用,系统会提示你是否保存设置,注意别点错了.

C#定义鼠标指针形状

在Windows应用程序中,通过设置控件的Cursor属性可以定义鼠标的显示形状.控件(如Button控件)的Cursor属性用于设置鼠标指针的类型,默认为Default. 语法: public virtual Cursor Cursor { get; set; } 属性值:一个Cursor,表示当鼠标指针位于控件上时显示的光标. Cursor:是一个值,该属性值如表所示. 表 Cursor属性的值 值 说明 AppStarting 获取在应用程序启动时显示的光标 Arrow 获取箭头光标 Cr

fgets函数执行完成后,文件指针如何移动?

用fgets执行之后,读取了文件中的一行,这时文件位置指针是自动指向文件的下一行的开头吗,还是指向当前行的结尾?如果一行的字符串没读取完会怎样? 实例结果: 如果一行的字符串没读取完会,下一次会接着上一次读取: 如果读完了,直接从下一行开头开始. 百度百科: 从文件结构体指针stream中读取数据,每次读取一行. 读取的数据保存在buf指向的字符数组中,每次最多读取bufsize-1个字符(第bufsize个字符赋'\0'),如果文件中的该行,不足bufsize个字符,则读完该行就结束. 如若该

彻底搞定C语言指针 第六篇

指向另一指针的指针 一.针概念: 早在本系列第二篇中我就对指针的实质进行了阐述 .今天我们又要学习一个叫做指向另一指针地址的指针.让我们先回顾一下指针的概念吧! 当我 们程序如下申明变量: short int i: char a: short int * pi: 程序会 在内存某地址空间上为各变量开辟空间,如下图所示. 内存地址→6 7 8 9 10 11 12 13 14 15 ------------------- --------------------------------------

c语言先用scanf初始化了一个字符指针,之后再定义字符数组出现内存不可读,在线等,急求

问题描述 c语言先用scanf初始化了一个字符指针,之后再定义字符数组出现内存不可读,在线等,急求 #include #include int main() { char*s; scanf("%s",s); //printf("%sn",s); //int n = strlen(s); //printf("%dn",n); char ret[56]; return 0; } 解决方案 s只是指针变量,没有分配内存 char*s; s = mall

一个关于指针和数组的问题

问题描述 一个关于指针和数组的问题 #define _CRT_SECURE_NO_WARNINGS#include ""stdlib.h""#include ""stdio.h""#include ""string.h"" int main(){ char buf1[100] = { 0 }; char buf2[100] = { 0 }; char *p1 = buf1; char *

ios-MFMailViewController选择指针

问题描述 MFMailViewController选择指针 使用xcode的MFMailViewController .设置了接收器. [mailComposer setToRecipients:emailAddresses]; 需要将接收器设置给NSArray中的指针,不知道应该怎么实现? 具体一点就是,需要根据用户的Email或者其他信息来访问一个人