美文网首页
可爱的指针(一)

可爱的指针(一)

作者: SeanC52111 | 来源:发表于2020-11-21 23:14 被阅读0次

    可爱的指针(一)

    基本介绍

    在互联网时代,几乎人人都会使用互联网来浏览信息。自互联网2.0以后,网页也变得越来越丰富。一个网页可以包含各种形式的信息,比如文字、图片、视频等。网页及其中的图片、视频都可以通过网址来获得。比如链接https://www.nasa.gov/sites/default/files/styles/full_width_feature/public/thumbnails/image/potw2045a.jpg对应一张NASA的星云图。
    [图片上传失败...(image-f2f3ba-1605971654846)]
    给定一个链接(或者称之为地址),浏览器会根据地址向对应的服务器获取图片(内容)。指针也是类似的概念。我们可以把网络地址看成指向网络资源的指针。

    在计算机内存中,每一个资源都对应了一个地址。每当我们用程序语言定义一个变量,计算机都会在内存中开辟一个特定大小的存储空间,用来存储对应变量的值。
    [图片上传失败...(image-57e600-1605971654846)]
    每一个变量都有三个要素:(1)变量名,(2)变量的地址,(3)变量的值。在上面的图中,变量c的地址为0x0012FF74,变量c的值为76。不难发现,变量和上面例子当中的NASA星云图是很像的。和指向星云图的指针(网址)类似,我们可以把某个变量的地址称为指向该变量的指针。

    有趣的是,计算机是可以通过变量的地址(指针)访问或改变变量的内容。抛开指针,我们可以通过cout<<c<<endl;来打印变量c的值。在使用指针来打印变量之前,我们首先需要获得变量的指针(地址)。C++语言中,可以通过&c获得变量c的地址(指针)。如果我们想把&c存放在一个变量里,我们就需要定义一个指针变量 int* pointerpointer作为指针变量,它是用来存储指针的,pointer本身作为一个变量,也有一个对应的地址,这个地址和它本身存储的地址是不同的)。pointer = &c称作指向变量c的指针。

    int c = 76; //定义int类型变量c,并赋值76
    int *pointer; //定义名字为pointer的指针变量,*表示pointer类型为指针类型
    pointer = &c; //将变量c的地址赋值给指针变量pointer
    

    这里,int *中的int称为指针类型的基类型,基类型可以帮助程序确定指针指向的变量在内存中所占空间的大小。

    有了指针变量之后,如何通过指针变量访问指针变量指向的变量内容呢?(如何利用pointer访问变量c的值76呢?)可以使用指针运算符*实现。使用*pointer,我们可以获得pointer所指向的存储单元的内容。这里,pointer所指向的存储单元的内容是变量c(注意,内容是变量c而不是76,因为我们可以实现操作*pointer = 72其含义等于c = 72)。

    Example 1:

    void main(){
      int c;
      int *pointer;
      c = 76;
      pointer = &c;
      cout<<*pointer<<endl;
    }
    

    这里,cout<<*pointer<<endl;相当于cout<<c<<endl;所以结果打印出76。注意,在使用指针变量时,一定要先对其进行赋值,然后才能使用指针变量访问或者改变指向变量的值。

    Example 2:

    void main(){
      int akey = 0,b = 0;
      int *p = NULL, *q = NULL;
      akey = 66;
      p = &akey;
      q = &b;
      *q = *p;
      cout<<"b = "<<b<<endl;
      cout<<"*q = "<<*q<<endl;
    }
    

    这个结果会打印出

    b = 66
    *q = 66
    

    程序的逻辑很简单,最关键的一行就是*q = *p;。可以把这一行理解为b = akey;,即把akey的值赋值给b

    Example 3:

    void main(){
      int *p1, *p2, *p;
      int a,b;
      cin>>a>>b;
      p1 = &a; p2 = &b;
      if (a < b){
        p = p1;
        p1 = p2;
        p2 = p;
      }
      cout<<"a = "<<a<<", b = "<<b<<endl;
      cout<<"max = "<<*p1<<", min = "<<*p2<<endl;
    }
    

    这里,我们输入3 5,会打印出

    a = 3, b = 5
    max = 5, min = 3
    

    这个程序的核心逻辑在如果a < b,则将小数a的指针p1和大数b的指针p2交换。交换后,p1会指向大数,p2会指向小数。(注意,只是交换了指针,并未交换指针所指向的内容。)

    Discussion of *&a

    定义整型变量a:int a = 3;&a代表我把它取得整型变量a的地址;*&a表示取地址&a处相应的内容(变量a);所以,*&a等价于a

    Discussion of &*pointer

    定义整型变量a:int a = 3;并且定义一个指向变量a的指针变量pointerint *pointer = &a;*pointer等价于整型变量a&*pointer等价于&a(*pointer)++等价于a++

    Discussion of iPtr++

    假设iPtr所代表的地址是0x00000100
    iPtr指向一个整型元素(占4字节),则iPtr++等于iPtr+1*4=0x00000104
    iPtr指向一个实型元素(占4字节),则iPtr++等于iPtr+1*4=0x00000104
    iPtr指向一个字符元素(占1字节),则iPtr++等于iPtr+1*1=0x00000101

    指针与数组

    指向数组元素的指针的定义、赋值、使用与基本变量完全相同。我们可以看如下的例子:

    Example 4

    void main(){
      int a[5] = {1,2,3,4,5};
      int *p = &a[3];
      cout<<*p<<endl;
      *p = 100;
      cout<<a[3]<<endl;
    }
    

    程序的的核心在int *p = &a[3];*p = 100;。这里,*p = 100;相当于a[3] = 100;。所以打印出:

    4
    100
    

    以上是指向数组元素的指针的使用,而我们需要关注的是指向数组的指针。

    在C++中,数组名代表数组首元素的地址。定义int a[10],则数组名a代表数组a[10]中第一个元素a[0]的地址;即a&a[0]等价;若有int *p,则p = &a[0]p = a等价。(注意,a是地址常量,不能给a赋值)

    在定义int a[10]存放10个int类型的连续空间之后,我们可以使用a + n来获得数组a中第n+1个元素的地址。(a+1是数组a[10]中第二个元素a[1]的地址。)同时,指向数组元素的指针可以用做下标:[]*的作用相同,比如p[i]*(p+i)等价。(使用*(p+i)要注意数组下标越界的情况!)

    Discussion

    如果定义int a[5] = {1,2,3,4,5}; int *p;,假设当前i=3, a[4] = 4,则t = *p--相当于t = a[i--],先做*p运算。

    • *++p相当于a[++i],先将p自加,然后再做*运算;
    • *--p相当于a[--i],先将p自减,然后再做*运算;
    • *p++相当于a[i++],先做*运算,在做p自加;
    • *p--相当于a[i--],先做*运算,在做p自减;

    Example 5

    利用指针实现数组a的输入和输出:

    int main(){
      int *p, i, a[10];
      p = a;
      for(i = 0; i < 10; i++)
        cin>>*p++;
      p = a;
      for(i = 0; i < 10; i++)
        cout<<*p++;
      return 0;
    }
    

    输入1 2 3 4 5 6 7 8 9 10
    输出1 2 3 4 5 6 7 8 9 10

    Example 6

    void main(){
     int a[5] = {1,2,3,4,5};
     int *p = &a[3];
     *p = 100;
     cout<<*p++<<endl;
     cout<<*p--<<endl;
     cout<<*--p<<endl;
    }
    

    这个例子主要看*操作和++或者--的优先级。*p++表明先计算*p,之后再p++,此时输出100*p等价于a[4];*p--先输出a[4]的值:5,再自减,此时*p等价于a[3]*--p首先执行--p,此时*p等价于a[2],再输出a[2]的值:3。故最后的输出为

    100
    5
    3
    

    以上是指针和一维数组之间的一些问题。下一篇将会简单介绍指针和二维数组之间的一些操作。

    相关文章

      网友评论

          本文标题:可爱的指针(一)

          本文链接:https://www.haomeiwen.com/subject/ceeyiktx.html