首页 > 其他分享 >【cplusplus教程翻译】指针(Pointers)

【cplusplus教程翻译】指针(Pointers)

时间:2023-05-27 15:56:30浏览次数:40  
标签:教程 const int Pointers ++ myvar foo cplusplus 指针

在前面的章节中,变量被解释为计算机内存中的位置,可以通过其标识符(名称)访问这些位置。这样,程序就不需要关心内存中数据的物理地址;只要需要引用变量,它就简单地使用标识符。
对于C++程序来说,计算机的内存就像一系列的存储单元,每个存储单元的大小都是一个字节,并且每个存储单元都有一个唯一的地址。这些单字节存储单元的排序方式允许大于一个字节的数据表示占据具有连续地址的存储单元。
这样,每个单元都可以通过其唯一地址轻松地定位在存储器中。例如,具有地址1776的存储器单元总是紧跟在具有地址1775的单元之后并且在具有1777的单元之前,并且正好是776之后的一千个单元并且正好是2776之前的一千个单元。
当一个变量被声明时,存储其值所需的内存被分配到内存中的一个特定位置(其内存地址)。通常,C++程序不会主动决定存储其变量的确切内存地址。幸运的是,这项任务留给了程序运行的环境——通常是一个在运行时决定特定内存位置的操作系统。然而,对于程序来说,能够在运行时获得变量的地址以访问相对于它处于特定位置的数据单元可能是有用的。

取地址符

变量的地址可以通过在变量的名称前面加一个与号(&)来获得,即运算符的地址。例如:foo = &myvar;
这将把变量myvar的地址分配给foo;通过在变量myvar的名称前面加上operator(&)的地址,我们不再将变量本身的内容分配给foo,而是将其地址分配给它。
内存中变量的实际地址在运行时之前是未知的,但为了帮助澄清一些概念,我们假设myvar在运行时被放置在内存地址1776中。考虑下面的代码

myvar = 25;
foo = &myvar;
bar = myvar;

内存模型如下

首先,我们把25赋值给myvar变量,这个变量的地址是1776
第二个语句把myvar的地址赋值给foo,最后一个语句把myvar变量赋值给bar
第二三个语句的区别就是取地址符
存变量地址的变量被称为指针,指针威力非常大,可以用于底层编程,下面介绍怎么声明和使用指针

解引用操作符

指针可以认为是图里的边,考虑指令集里的访存指令,这个操作被叫做解引用baz = *foo;

请注意解引用的区别

baz = foo;   // baz equal to foo (1776)
baz = *foo;  // baz equal to value pointed to by foo (25) 

把&和*认为是操作符进行思考,赋值也是操作符

myvar = 25;
foo = &myvar;

声明指针

由于指针能够直接引用它所指向的值,因此指针指向char时与指向int或float时具有不同的属性。一旦想要解引用,就需要知道类型。为此,指针的声明需要包括指针要指向的数据类型。
声明语法type * name,type是指向数据的类型,注意指针类型是单独的一种类型,和指向数据的类型不同

int * number;
char * character;
double * decimals;

上面这三个变量指向的类型不同,但是这三个变量的类型是相同的(都是地址,32位或64位),指针变量的大小是相同,但是指向的类型大小可能不同,能支持的操作也不同。注意这个*和解引用的不同

// my first pointer
#include <iostream>
using namespace std;

int main ()
{
  int firstvalue, secondvalue;
  int * mypointer;

  mypointer = &firstvalue;
  *mypointer = 10;
  mypointer = &secondvalue;
  *mypointer = 20;
  cout << "firstvalue is " << firstvalue << '\n';
  cout << "secondvalue is " << secondvalue << '\n';
  return 0;
}

指针变量可以随便改变赋值

// more pointers
#include <iostream>
using namespace std;

int main ()
{
  int firstvalue = 5, secondvalue = 15;
  int * p1, * p2;

  p1 = &firstvalue;  // p1 = address of firstvalue
  p2 = &secondvalue; // p2 = address of secondvalue
  *p1 = 10;          // value pointed to by p1 = 10
  *p2 = *p1;         // value pointed to by p2 = value pointed to by p1
  p1 = p2;           // p1 = p2 (value of pointer is copied)
  *p1 = 20;          // value pointed to by p1 = 20
  
  cout << "firstvalue is " << firstvalue << '\n';
  cout << "secondvalue is " << secondvalue << '\n';
  return 0;
}

注意声明int * p1, * p2;int * p1, p2;声明的变量不同,前面是两个指针,后面是一个指针一个整型变量

指针和数组

数组的概念与指针的概念有关。事实上,数组的工作方式非常类似于指向其第一个元素的指针,而且,实际上,数组总是可以隐式转换为正确类型的指针。例如,考虑以下两个声明:

int myarray [20];
int * mypointer;

然后可以赋值mypointer = myarray;,请注意指针是可以随便赋值的,数组可以认为是一个常量,编译器就确定了,恒指向20个整型元素

// more pointers
#include <iostream>
using namespace std;

int main ()
{
  int numbers[5];
  int * p;
  p = numbers;  *p = 10;
  p++;  *p = 20;
  p = &numbers[2];  *p = 30;
  p = numbers + 3;  *p = 40;
  p = numbers;  *(p+4) = 50;
  for (int n=0; n<5; n++)
    cout << numbers[n] << ", ";
  return 0;
}

指针和数组的操作也是相同的,考虑常量
请注意[] offset操作符,等价于解引用和指针加减法

a[5] = 0;       // a [offset of 5] = 0
*(a+5) = 0;     // pointed to by (a+5) = 0 

指针初始化

声明的时候也可以初始化,这是一个好习惯

int myvar;
int * myptr;
*myptr = &myvar; // 这句话不合法,其实这个两边的类型不同

第二行不是解引用操作符,空格并不影响语法

int myvar;
int *foo = &myvar;
int *bar = foo;

指针运算

指针只支持加减法,偏移量和指向的数据类型有关
char永远是1字节,short int long这些和系统有关,最好写成int32 int16这种

char *mychar;
short *myshort;
long *mylong;

加一的偏移量是不同的,注意优先级顺序

*p++   // same as *(p++): increment pointer, and dereference unincremented address
*++p   // same as *(++p): increment pointer, and dereference incremented address
++*p   // same as ++(*p): dereference pointer, and increment the value it points to
(*p)++ // dereference pointer, and post-increment the value it points to 

注意可读性*p++ = *q++;等价于```c++
*p = *q;
++p;
++q;

# 指针和常量
```c++
int x;
int y = 10;
const int * p = &y;  //可以非常转成常
x = *p;          // ok: reading p
*p = x;          // error: modifying p, which is const-qualified 

常量指针用于函数参数,可以和常引用类似

// pointers as arguments:
#include <iostream>
using namespace std;

void increment_all (int* start, int* stop)
{
  int * current = start;
  while (current != stop) {
    ++(*current);  // increment value pointed
    ++current;     // increment pointer
  }
}

void print_all (const int* start, const int* stop)
{
  const int * current = start;
  while (current != stop) {
    cout << *current << '\n';
    ++current;     // increment pointer
  }
}

int main ()
{
  int numbers[] = {10,20,30};
  increment_all (numbers,numbers+3);
  print_all (numbers,numbers+3);
  return 0;
}

常量指针只是不能改指向的值,但是指针本身可以自增

int x;
      int *       p1 = &x;  // non-const pointer to non-const int
const int *       p2 = &x;  // non-const pointer to const int
      int * const p3 = &x;  // const pointer to non-const int
const int * const p4 = &x;  // const pointer to const int 

const和指针的语法肯定很棘手,识别最适合每种使用的情况往往需要一些经验。无论如何,尽早获得指针(和引用)的常量是很重要的,但如果这是你第一次接触常量和指针的混合,你不应该太担心掌握一切。更多的用例将出现在接下来的章节中。
常量指针和指针常量的声明区别关键是const和*的顺序,类型的顺序无关

const int * p2a = &x;  //      non-const pointer to const int
int const * p2b = &x;  // also non-const pointer to const int

指针和字符串字面量

字符串字面量本质上是字符数组,可以用字符数组初始化,也可以用指针初始化,注意是否位常量const char * foo = "hello";,可以认为是无名变量,```c++
*(foo+4)
foo[4]

这样用是没问题的,都是字符o
# 指针的指针
多级指针用法,一定要注意复杂度
```c++
char a;
char * b;
char ** c;
a = 'z';
b = &a;
c = &b;

链式法则

注意指针运算

指针空类型的指针

void类型的指针是一种特殊类型,void表示不定长度、不定解引用属性
void类型有很大的灵活性,可以进行类型转换,指向各种类型,不过要想解引用,还是要额外信息进行强转,最好使用模板

// increaser
#include <iostream>
using namespace std;

void increase (void* data, int psize)
{
  if ( psize == sizeof(char) )
  { char* pchar; pchar=(char*)data; ++(*pchar); }
  else if (psize == sizeof(int) )
  { int* pint; pint=(int*)data; ++(*pint); }
}

int main ()
{
  char a = 'x';
  int b = 1602;
  increase (&a,sizeof(a));
  increase (&b,sizeof(b));
  cout << a << ", " << b << '\n';
  return 0;
}

sizeof也是一个操作符,编译器就可以确定

无效指针和空指针

指针意味着地址,所有有有效无效之分

int * p;               // uninitialized pointer (local variable)

int myarray[10];
int * q = myarray+20;  // element out of bounds 

赋值的时候可以随便赋值,只有解引用的时候才能判断地址是否合法
空指针是一种特殊类型,可以有两种方式

int * p = 0;
int * q = nullptr;

NULL也是类似操作,只不过NULL不是关键字,也不是显式0,和头文件定义有关
注意空指针和void指针的区别,空指针是不指向任何地方,void指针是指向地方的类型不定

函数指针

函数指针可以用来作为另一个函数的参数(避免过多的if else,模板特化),函数指针的声明语法和正常函数声明一样,只不过把函数名用括号及*包括起来就行,注意和typedef等一起使用,类成员函数指针只能由类对象调用,注意声明语法

// pointer to functions
#include <iostream>
using namespace std;

int addition (int a, int b)
{ return (a+b); }

int subtraction (int a, int b)
{ return (a-b); }

int operation (int x, int y, int (*functocall)(int,int))
{
  int g;
  g = (*functocall)(x,y);
  return (g);
}

int main ()
{
  int m,n;
  int (*minus)(int,int) = subtraction;

  m = operation (7, 5, addition);
  n = operation (20, m, minus);
  cout <<n;
  return 0;
}

直接用函数名就可以初始化

总结

指针其实是底层存储模型的抽象,把各种内存都抽象成连续的字节序列,注意虚拟内存和物理内存的区别。

标签:教程,const,int,Pointers,++,myvar,foo,cplusplus,指针
From: https://www.cnblogs.com/xiaoweing/p/17436708.html

相关文章

  • win10安装配置mmdetection教程
    一、安装Anaconda并创建虚拟环境1、官网下载Anaconda安装包安装即可2、创建新的虚拟环境condacreate-nopenmmlabpython=3.8condaactivateopenmmlab二、配置Pytorch环境1、查看并确定自己电脑的CUDA版本打开nvidia设置面板,点击【帮助】-【系统信息】选择【组件】在......
  • 【城南 · LlamaIndex 教程】一文看懂LlamaIndex用法,为LLMs学习私有知识
    我是卷了又没卷,薛定谔的卷的AI算法工程师「陈城南」(全网平台同名)~担任某大厂的算法工程师,带来最新的前沿AI知识,分享AI有趣工具和实用玩法,包括ChatGPT、AI绘图等,欢迎大家交流~交流「cchengnan113」备注「AI交流」可进裙知乎「陈城南」公众号「陈城南」本文是对LlamaIndex......
  • OneKey虚拟信用卡开通教程
    充值错代币我们无法追回,请注意充值正确的币种,目前只支持USDC。注册OneKeyCard 点击邀请链接https://card.onekey.so/?i=3I8VT3 进入注册界面,点击【signinwithGoogle】,使用谷歌账户注册或者可点击官网链接:https://card.onekey.so 进入注册界面,点击【signinwithG......
  • 【K8s入门推荐】K8s1.24版本部署全教程,轻松掌握技巧kubeadm丨Kubernetes丨容器编排丨
    通过kubeadm方式极速部署Kubernetes1.24版本前言在Kubernetes的搭建过程中,繁琐的手动操作和复杂的配置往往会成为制约部署效率的关键因素。而使用kubeadm工具可以避免这些问题,大大提高集群的部署效率和部署质量。本文将为大家详细介绍如何使用kubeadm工具快速搭建Kubernetes1.24......
  • 【2023最新】超详细图文保姆级教程:App开发新手入门(6)
    9.编译正式版 本章我们简单介绍一下如何设置应用的桌面图标及应用的启动页 通过之前章节的学习,我们已经完成了一个简单应用的开发,本部分章节主要目的是为本系列教程进行一个整体性的收尾,简介讲解一下如何编译一个可用于上架应用市场的正式版安装包。(本章内容超级简单......
  • GitHub教程
    1.概述1.1Git和代码托管中心代码托管中心的任务:维护远程库局域网环境下:GitLab服务器外网环境下:GitHub码云1.2本地库和远程库团队内部协作跨团队协作2.Git命令行操作2.1本地库操作命令:gitinit效果:注意:.git目录中存放的是本地库相关的子目......
  • 【2023最新】超详细图文保姆级教程:App开发新手入门(2)
    上章节我们已经成功的创建了一个App项目,接下来我们讲述一下,如何导入项目、编辑代码和提交项目代码。 Let’sGo! 4.项目导入 当用户创建一个新的应用时,YonStudio开发工具会自动导入模板项目的默认代码,不需要手动进行代码导入。那么当我们不是创建应用,而是需要导入一个已经存......
  • 【2023最新】超详细图文保姆级教程:App开发新手入门(3)
    上文回顾,我们已经完成了一个应用项目创建、导入、代码更新、代码同步和代码提交,本章继续我们的新手开发之旅,讲述一下如何将开发完成的应用进行编译,生成可供他人安装、可上架的应用安装包。6应用打包 应用打包,简单来说就是将编写的代码,通过工具的打包编译机制,打包编译生成对应的手......
  • 【2023最新】超详细图文保姆级教程:App开发新手入门(4)
    之前章节我们已经完成了一个应用项目的导入、代码更新和代码提交和应用打包编译,本章继续讲述一下,如何在开发过程中进行代码的同步联机调试。7代码真机调试7.1纯静态CSS页面样式查看代码调试有多种方式,如果是查看纯粹的静态样式,可以使用浏览器打开对应页面(html或stml文件),或者直接......
  • 【2023最新】超详细图文保姆级教程:App开发新手入门(5)
    上文回顾,我们已经完成了一个应用的真机调试,本章我们来了解一下如何引入YonBuilder移动开发的(原生)移动插件,并利用移动插件完成一个简单的视频播放器。8.「移动插件」的使用 8.1什么是「移动插件」? 用通俗的话来解释,YonBuilder移动开发内的「移动插件」,是指使用原生语言(androi......