C语言指针及C++引用

经过c与c++的学习,整理一下关于指针以及引用方面的问题

指针:

指针是什么?

指针本身是一个变量,它存储的是数据在内存中的地址而不是数据本身的值,指针类型、指针所指向的类型、指针的值或者叫指针所指向的内存区、指针本身所占据的内存区都是我们要考虑的问题

1、int a=0,p;      p =&a;

2、int  a=0;       *p=&a;

第一种定义方法然后将a的地址赋给p。第二种是在定义指针p的同时将a的地址赋给指针p。我理解为  int *  是定义指针的标志

将指针p去掉以后可以看出指针的类型:

(1)char*p;//指针的类型是char*

(2)int*(*ptr)[4];//指针的类型是int*(*)[4]

将*p去掉以后可以看出指针指向区域的内容是什么类型

(1)char*p;//指针的类型是char

(2)int*(*ptr)[4];//指针的类型是int*()[4]

我们可以通过*p来找到指针所指向的变量a的地址,然后对地址中的值操作。

printf("%p",p)  结果是p指向的a的地址
printf("%d",*p) 结果是变量a的值
printf("%d",&p) 结果是指针p的地址

引用了一个看到的例子:
  1.  int *p; //首先从P 处开始,先与*结合,所以说明P 是一个指针,然后再与int 结合,说明指针所指向的内容的类型为int 型.所以P是一个返回整型数据的指针
  2.  int p[3]; //首先从P 处开始,先与[]结合,说明P 是一个数组,然后与int 结合,说明数组里的元素是整型的,所以P 是一个由整型数据组成的数组
  3.  int *p[3]; //首先从P 处开始,先与[]结合,因为其优先级比*高,所以P 是一个数组,然后再与*结合,说明数组里的元素是指针类型,然后再与int 结合,说明指针所指向的内容的类型是整型的,所以P 是一个由返回整型数据的指针所组成的数组
  4.  int (*p)[3]; //首先从P 处开始,先与*结合,说明P 是一个指针然后再与[]结合(与"()"这步可以忽略,只是为了改变优先级),说明指针所指向的内容是一个数组,然后再与int 结合,说明数组里的元素是整型的.所以P 是一个指向由整型数据组成的数组的指针
  5.  int **p; //首先从P 开始,先与*结合,说是P 是一个指针,然后再与*结合,说明指针所指向的元素是指针,然后再与int 结合,说明该指针所指向的元素是整型数据.,二级指针一般用在数据结构编程中
  6.  Int (*p)(int); //从P 处开始,先与指针结合,说明P 是一个指针,然后与()结合,说明指针指向的是一个函数,然后再与()里的int 结合,说明函数有一个int 型的参数,再与最外层的int 结合,说明函数的返回类型是整型,所以P 是一个指向有一个整型参数且返回类型为整型的函数的指针
  7.  int *(*p(int))[3]; //从P 开始,先与()结合,说明P 是一个函数,然后进入()里面,与int 结合,说明函数有一个整型变量参数,然后再与外面的*结合,说明函数返回的是一个指针,,然后到最外面一层,先与[]结合,说明返回的指针指向的是一个数组,然后再与*结合,说明数组里的元素是指针,然后再与int 结合,说明指针指向的内容是整型数据.所以P 是一个参数为一个整数据且返回一个指向由整型指针变量组成的数组的指针变量的函数.
&是取地址运算符,*是间接运算符。&a 的运算结果是一个指针,指针的类型是a的类型加个*,指针所指向的类型是a的类型,指针所指向的地址就是a的地址。*p 的结果是p所指向的东西,这个东西有这些特点:它的类型是p指向的类型,它所占用的地址是p所指向的地址。这里提一个例子:Int **ptr=&pa; //&pa 在这句中*ptr 就是指针pa,但是*ptr占据了内存,所以被放到右边

数组的数组名可以看成一种指针arr[0] == *arr;arr[3] == *(arr+3)或者是char *str[3]={"Hehe!","Haha.","Helloworld"};

str 是一个三单元的数组,该数组的每个单元都是一个指针,这些指针各指向一个字符串。str 当作一个指针的话,它指向数组的第0号单元,它的类型是char **,它指向的类型是char *。str+1 也是一个指针,它指向数组的第1号单元,它的类型是char**,它指向的类型是char*。*str 也是一个指针,它的类型是char *,它所指向的类型是char,它指向的地址是字符串"Hehe!"的第一个字符的地址,即‘H‘的地址。

结构体和指针:struct NUM{  int x;  int y;}struct NUM num = {20,30};struct *ptr = # 访问成员的时候 ptr->x,ptr->yint *pstr = (int*)# 访问成员的时候 (*pstr)(*(pstr+1))

引用:

一个变量可取多个别名。

普通引用举例

答案:

由此可见a,b,c地址是同一个

下面再看一个例子

这道题输出a == b == 20,此时b的值不能被修改,但是可以通过修改a去修改b,还有,只有常引用可以引用const修饰的变量。

void S(int& a, int& b)//使用引用的话,不做临时拷贝,&的使用说明此处只是原参数的另一个名字而已,所以修改时直接在原参数的基础上修改变量的值

void S(int* a_p, int* b_p)//传入的是地址,因为地址是唯一的,所以指针通过地址的访问进而可修改其内容。

总结以下几点:

(1)引用就是起"外号",为一个标识符另外再取一个名字。&在这里不是求地址运算,而是起标识作用。

(2)类型标识符是指目标变量的类型,引用不光可以引用标识符,也可以引用立即数(右值、字面值、常量),但必须加const属性。

(3)声明引用时,必须同时对其进行初始化引用,否则编译错误。

(4)引用声明完毕后,相当于目标变量名有两个名称,即该目标原名称和引用名,且不能再把该引用名作为其他变量名的别名,即引用只能是一次性的,无法再更改(引用一旦成功后,它是个变量了这个身份至死不渝)。

(5)对引用求地址,就是对目标变量求地址。即引用名是目标变量名的一个别名。引用在定义上是说引用不占据任何内存空间,但是编译器在一般将

其实现为const指针,即指向位置不可变的指针,所以引用实际上与一般指针同样占用内存。

(6)不能建立引用的数组。因为数组是一个由若干个元素所组成的集合,所以无法建立一个由引用组成的集合,但是可以建立数组的引用。

(7)引用常见的使用用途:作为函数的参数、函数的返回值,但绝不能返回局部变量的引用。

(8)不能定义空引用,但"野引用"或"悬空引用"是存在的(引用了堆内存的数据,当堆内存释放后就不应该再使用)。

(9)引用可以当作函数的参数,它引用的对象就是函数的实参,所以引用可以达到指针的效果。
a、函数之间共享变量
b、提高参数的传递效率(比指针还要高)
c、可以获取参数

但是引用不可以完全取代指针。

引用和指针的区别和联系:

不同点:

1. 指针是一个实体,而引用仅是个别名;

2. 引用使用时无需解引用(*),指针需要解引用;

3. 引用只能在定义时被初始化一次,之后不可变;指针可变;

4. 引用没有 const,指针有 const;const修饰的指针不可变;

5. 引用不能为空,指针可以为空;

6. “sizeof 引用”得到的是所指向的变量(对象)的大小,而“sizeof 指针”得到的是指针本身(所指向的变量或对象的地址)的大小;

7. 指针和引用的自增(++)运算意义不一样;

8.从内存分配上看:程序为指针变量分配内存区域,而引用不需要分配内存区域。

9.指针通过某个指针变量指向一个对象后,对它所指向的变量间接操作。程序中使用指针,程序的可读性差;而引用本身就是目标变量的别名,对引用的操作就是对目标变量的操作。

相同点:两者都是地址的概念,指针指向一块儿内存,其内容为所指内存的地址;引用是某块儿内存的别名

 
 

原文地址:https://www.cnblogs.com/hjt1806/p/9451234.html

时间: 2024-10-15 17:21:06

C语言指针及C++引用的相关文章

第5课 C语言指针深入1

1.客户端两种主流的接口模型: #ifndef _SOCKETCLINET_H #endif _SOCKETCLINET_H #ifdef __cplusplus //如果用了C++的编译器,用C语言的规范来引用 extern "C" { #endif //socket客户端环境初始化 int socketclient_init(void** handle); //socket客户端报文发送 int socketclient_send(void* handle, unsigned ch

C语言指针的陷阱

C语言指针的陷阱   分类: C/Cpp 转自:http://blog.csdn.net/porscheyin/article/details/3461670 "C语言诡异离奇,陷阱重重,却获得了巨大成功!"--C语言之父Dennis M.Ritchie.Ritchie大师的这句话体现了C语言的灵活性以及广泛的使用,但也揭示了C是一种在应用时要时刻注意自己行为的语言.C的设计哲学还是那句话:使用C的程序员应该知道自己在干什么.有时用C写的程序会出一些莫名其妙的错误,看似根源难寻,但仔细

c++智能指针以及循环引用问题(转)

解决循环引用: 在知道存在循环引用的条件下,使用boost::weak_ptr,即弱引用来代替循环引用中的某个强引用,从而打破循环引用的环. 由于 C++ 语言没有自动内存回收机制,程序员每次 new 出来的内存都要手动 delete,比如流程太复杂,最终导致没有 delete,异常导致程序过早退出,没有执行 delete 的情况并不罕见,并造成内存泄露.如此c++引入 智能指针 . c++ 智能指针主要包括:unique_ptr,shared_ptr, weak_ptr, 这三种,其中auto

C语言指针理解patr1

今天在看<Head First C语言>中,书中的代码是*lat=*lat+1;我则写成了*lat++:结果是并不能得到lat指针的内容加一.之后发现原来在C语言中*解指针运算符 与++后自增运算符的优先级是相同的,*p++,后自增运算符++只作用于p,并不能作用于*p(如果*的优先级更高的话才行). 之后在调试中又发现了一个之前不了解的地方,在C语言中栈的存储是向低地址扩展,也就是先声明的变量在内存中反而内存地址是更大的. 1 #include <stdio.h> 2 void

C语言指针的深入理解

指针是C语言中广泛使用的一种数据类型.运用指针编程是C语言最主要的风格之一. 利用指针变量可以表示各种数据结构:能很方便地使用数组和字符串:并能象汇编语言一样处理内存地址,从而编出精练而高效的程序.指针极大地丰富了C语言的功能. 学习指针是学习C语言中最重要的一环,能否正确理解和使用指针是我们是否掌握C语言的一个标志. C语言为什么有指针以及指针有什么好处? 从大体上讲:指针是 C 语言区别于其他同时代高级语言的主要特征之一.其次因为数组或函数都是连续存放的.通过访问指针变量取得了数组或函数的首

彻底搞定C语言指针详解

1.语言中变量的实质 要理解C指针,我认为一定要理解C中“变量”的存储实质, 所以我就从“变量”这个东西开始讲起吧! 先来理解理解内存空间吧!请看下图: 内存地址→ 6 7 8 9 10 11 12 13 ----------------------------------------------------------------- ... | | | | | | | |.. ------------------------------- ---------------------------

8、C语言——指针

指针 一.指针变量定义 1.定义的格式: 类型名 *指针变量名 此处的*是一种标志,指针变量的标志 指针变量名是符合用户的标识符 前类型后分号是定义语句,除此以外都是执行语句 c语言有两种变量: 变量(普通变量)存储内容值,常量都是内容值,只能存放在普通变量中: 地址变量(指针变量)存储地址值: 注: 1)定义变量(普通变量.指针变量)都必须在前面有类型名 2)在定义指针变量时,指针变量名前的"*"表示现定义的是一个指针类型的恩,只是一个标志.变量.星号并不是指针变量名的 3)指针变量

C++函数的三种传递方式为:值传递、指针传递和引用传递

值传递: void fun(int x){ x += 5; //修改的只是y在栈中copy x,x只是y的一个副本,在内存中重新开辟的一块临时空间把y的值 送给了x:这样也增加了程序运行的时间,降低了程序的效率. } void main(void){ int y = 0; fun(y); cout<<\"y = \"<<y<<endl; //y = 0; } 指针传递: void fun(int *x){ *x += 5; //修改的是指针x指向的内

你需要知道关于C语言指针的一切

Everything you need to know about pointers in C 你需要知道关于C语言指针的一切 指针的定义 指针是内存地址. (嗯,简短的段落.) 开始 假设你声明一个名为foo的变量. int foo; 这个变量占用一些内存. 在当前主流的Intel处理器上,它占用四个字节的内存(因为int是四个字节宽). 现在让我们声明另一个变量. int *foo_ptr = &foo; foo_ptr被声明为指向int的指针.我们已经初始化它指向foo. 正如我所说,fo