二级指针的作用详解_二级指针有什么用

收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。
img
img

如果你需要这些资料,可以戳这里获取

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

欢迎转载,转载请注明原文地址:http://blog.csdn.net/majianfei1023/article/details/46629065

一、概念

在如下的A指向B、B指向C的指向关系中:

首先

C是"一段内容",比如你用malloc或者new分配了一块内存,然后塞进去"一段内容",那就是C了。C的起始地址是0x00000008。

B是一个指针变量,其中存放着C的地址,但是B也要占空间的啊,所以B也有地址,B的起始地址是0x00000004,但是B内存中存放的是C的地址,所以B里面的内容就是0x00000008。

那么到此为止都比较好理解:

B= 0x00000008;  //B的内容 
*B = “一段内容”;  //B解引用,也就是B指针指向的C的值
&B = 0x00000004;  //B取地址,B的地址是0x00000004
那么,再来看A:
A是二级指针变量,其中存放着B的地址0x00000004,A也有地址,是0x00000000;

*A = B= 0x00000008;  //A解引用也就是B的内容 
**A = *B = “一段内容”;  //B解引用,也就是B指针指向的C的值
A = &B = 0x00000004;  //A存的是B的地址,B的地址是0x00000004
&A = 0x00000000;  //A取地址

二、使用

二级指针作为函数参数的作用:在函数外部定义一个指针p,在函数内给指针赋值,函数结束后对指针p生效,那么我们就需要二级指针。

看看下面一段代码:有两个变量a,b,指针q,q指向a,我们想让q指向b,在函数里面实现。

1.先看看一级指针的实现

#include
 
using namespace std;
 
int a= 10;
int b = 100;
int *q;
 
void func(int *p)
{
    cout<<“func:&p=”<<&p<<“,p=”<<p<<endl;  //note:3
    p = &b;
    cout<<“func:&p=”<<&p<<“,p=”<<p<<endl;  //note:4
}
 
 
int main()
{
    cout<<“&a=”<<&a<<“,&b=”<<&b<<“,&q=”<<&q<<endl;  //note:1
    q = &a;
    cout<<“*q=”<<*q<<“,q=”<<q<<“,&q=”<<&q<<endl;  //note:2
    func(q);
    cout<<“*q=”<<*q<<“,q=”<<q<<“,&q=”<<&q<<endl;  //note:5
 
    system(“pause”);
    return 0;
}

这么写有什么问题?为什么*q不等于100?我们看一下输出便知:

&a=0032F000,&b=0032F004,&q=0032F228
*q=10,q=0032F000,&q=0032F228
func:&p=0018FD24,p=0032F000
func:&p=0018FD24,p=0032F004
*q=10,q=0032F000,&q=0032F228

我们看输出:

note:1->a,b,q都有一个地址.

note:2->q指向a.

note:3->我们发现参数p的地址变了,跟q不一样了,是的参数传递是制作了一个副本,也就是p和q不是同一个指针,但是指向的地址0x0032F000(a的地址)还是不变的.

note:4->p重新指向b.

note:5->退出函数,p的修改并不会对q造成影响。

结论:

编译器总是要为函数的每个参数制作临时副本,指针参数p的副本是 p,编译器使 p = q(但是&p != &q,也就是他们并不在同一块内存地址,只是他们的内容一样,都是a的地址)。如果函数体内的程序修改了p的内容(比如在这里它指向b)。在本例中,p申请了新的内存,只是把 p所指的内存地址改变了(变成了b的地址,但是q指向的内存地址没有影响),所以在这里并不影响函数外的指针q。

这就需要二级指针操作:

2.二级指针操作

#include
 
using namespace std;
 
int a= 10;
int b = 100;
int *q;
 
void func(int **p)  //2
{
    cout<<“func:&p=”<<&p<<“,p=”<<p<<endl;

img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上物联网嵌入式知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

如果你需要这些资料,可以戳这里获取

、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新**

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

如果你需要这些资料,可以戳这里获取

  • 19
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在二叉树的创建过程中,需要创建新的节点并将它加入到已有的二叉树中。根据二叉树的特点,每个节点都有左子树和右子树,因此需要向每个节点传递两个指针参数,分别指向左子树和右子树。如果直接传递指针参数,那么在函数中修改指针的值时会影响到原指针的值,进而影响到整个树的结构,这是不可取的。 因此,我们通常会传递指向指针的指针或者指向指针的引用,也就是所谓的“二级指针”或“指针的引用”。这样,在函数中修改指针的值时,只会影响到指针指向的地址,而不会影响到原指针的值,从而保证了整个树的结构不会被破坏。 具体来说,二级指针作用是,将一个指针的地址传递给另一个指针。在创建新节点时,我们需要将新节点的地址赋值给父节点的左子树或右子树指针,如果使用一级指针,那么就需要在函数中返回新节点的地址,再将其赋值给父节点的指针。而使用二级指针,则可以直接将新节点的地址赋值给父节点的指针,不需要在函数中返回新节点的地址。这样可以简化代码,提高效率。 举个例子,假设我们要在一棵二叉树中插入一个新节点,代码如下: ``` struct TreeNode { int val; TreeNode* left; TreeNode* right; TreeNode(int x) : val(x), left(nullptr), right(nullptr) {} }; void insert(TreeNode* root, int val) { if (root == nullptr) { root = new TreeNode(val); return; } if (val < root->val) { insert(root->left, val); } else { insert(root->right, val); } } ``` 注意,这里传递的是一级指针 `TreeNode* root`。在函数中,我们首先判断根节点是否为空,如果为空则创建新节点并返回。否则,根据节点的值大小递归地插入新节点到左子树或右子树中。但是,这段代码有一个问题,就是新节点并没有被正确地插入到树中。原因在于,`root` 是一个指针,传递过来的是指针的副本,函数中对 `root` 的修改并不会影响到原指针 `root`。因此,我们需要使用二级指针来解决这个问题。 改进后的代码如下: ``` void insert(TreeNode** root, int val) { if (*root == nullptr) { *root = new TreeNode(val); return; } if (val < (*root)->val) { insert(&((*root)->left), val); } else { insert(&((*root)->right), val); } } ``` 注意,这里传递的是二级指针 `TreeNode** root`。在函数中,我们首先判断根节点是否为空,如果为空则创建新节点并将其地址赋值给 `*root`。否则,根据节点的值大小递归地插入新节点到左子树或右子树中,并将新节点的地址赋值给 `(*root)->left` 或 `(*root)->right`。这样,就可以正确地将新节点插入到树中了。 总之,使用二级指针可以避免指针的值被修改,保证树的结构正确性,提高代码的效率和可读性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值