数组指针的用法,指针数组如何赋值
数组指针的用法,指针数组如何赋值详细介绍
本文目录一览: C语言:简述一下“数组和指针的关系”?
数组是一类同类型变量的集合,类似于属于上的集合的概念,数字也是有集合的,比如整数集,实数集等。
数组也是一个集合,数组的名字是集合的名字,数组后面的方括号里的数字,表示这个集合的大小,数组名前面的类型说明符,表示这个数组是什么类型的集合,比如:
int a[5], a是数组的名字,因为我们后面要用到数组,所以我们必须要取一个名字,要不然程序也不知道,你调用的数组是哪个不是?[5]方括号里的数字5,表示这个数组的大小是五个数的集合,数组名最前面的int是类型说明符,说明数组这个集合里的数是什么类型的,在C语言里,基本类型就那么几个,我们除了用基本类型外,还可以使用我们自己设置的结构体等类型,相当于数学里面,集合的说明,用来说明这个集合什么实数集合,还是整数集合的类似意义。
指针,我们直接可以用地址来代替指针这个名字,指针就是地址,你把任何变量看成一个人,每个人肯定都有自己住的地方,变量也一样,变量在内存上,一定也有地址,而指针就是地址。C语言里可以有两种调用变量的形式,一种直接使用变量的名字来调用,比如 int a=10;
意思就是把10这个数给变量a,除了这样用以为,还能通过指针来使用,指针是地址,需要用到取地址符&,比如 int *c, a; c=&a;像上面的语句,c是一个指针,怎么识别c是指针呢?因为c在声明的时候使用了指针声明符*,用来说明c是一个指针,而a则是一个变量,因为没有使用指针声明符进行声明,c=&a,这个语句即把a的地址,给了c,因为c是指针,所以c需要的是变量的地址,而不是变量的名字,这个时候在赋值号=的后面使用了取地址符&,用来取出变量a的地址给C,当我们要用a的时候,我们可以通过指针c来访问变量a,通过c上的地址,来找到a所在的位置,再取出a的值,这里需要使用解地址符 *,假设有个变量d需要变量a一样的值,我们需要使用d=(*c);这语句表示通过c的地址,访问a,取出a的值赋值给变量d,这里也有一个*符号,但是这个符号不是指针声明符了,而是解地址符了,那么怎么判断两种用途的*符号的作用呢?很简单,指针声明符值在变量声明,或者指针声明的时候使用,其他地址都不是指针声明符号,而是解地址符,怎么判断一个语句是声明还是其他语句呢?很简单,不管是什么声明,前面都需要类型说明符,比如 int a,*c。这是一个声明,声明一个变量a,类型是int型,一个指针c,存放地址,存放的是一个int型变量的地址。对,你没看错,指针也需要类型说明符,因为C语言里,有好多类型,他们大小不一样,需要说明,然后使用他们的过程,编译器会帮你解决的,你不用关心实现的问题。
好,上面大致说明了,数组是什么,指针是什么?那么数组和指针的关系是什么呢?乍一看,貌似没有任何关系,对吧,但是还是有的,因为数组也是放在内存上的啊,显然数组也是有地址的,即有指针的对吧,所以我们应该可以使用数组的指针对吧,int a[5]这是一个数组对吧,那么它的指针要怎么设定呢?答案是这样的, int [] *c,这是什么意思呢?很简单,c前面有指针说明符吧,说明c是一个指针,那么c这个指针是面向什么变量呢?是面向int [],那么int []是什么?[]是数组的特征,[]的前面有个int,说明c这个指针应该是指向一个int型的数组的对吧,所以这是一个指向数组的指针对吧,但是问题来了,数组是一个集合啊,比如上面说的int a[5],
这个数组是由五个int型的数组成的集合,你这个指针是指向哪一个数啊?如果你指向一个数,而指向一个整体,我怎么用?我们最常用的需求是,能访问数组里的每一个数啊对吧。这里有了需求,我们就有一个约定俗成的规定,那就是什么规定呢?数组的名字给它一个另外的功能,什么功能呢?就是数组的名字本身就是一个指针,那么要确定是哪个数的呢?我们默认是数组最开头的指针,即a[5]中的数组名字a是一个指针,代表a[0]的地址,这是一个默认,因为我们有需求,为了能快速简便的通过指针来使用数组里的数,而不是使用数组整个地址,因为没有意义,通过这个设定之后,我们有了进一步的需求,那么我们a[1]的元素怎么办?我们也想通过指针来使用啊,很简单,直接使用++或+即可,a+1,数组名a代表a[0]的地址,那么我们a+1就用来表示a[1]的地址,一点都不过分,虽然看起来有点变扭,但是符合我们的需求,而且简便快速,同理也可以使用++,--的用法,但是地址的* ,/就没有意义了,所以指针的 *, /是不能使用的,了解了约定俗成的这个规定后,百分99的程序都可以看懂了,希望能理解我说的,如果不理解,多看几遍我说的吧,然后对照着使用数组和指针的程序理解一下,用多了,看多了,自然就懂了。
指针就是指针,数组就是数组;在建立关系之前,指针与数组毫无关系。
一旦一个与某数组元素类型兼容的指针指向了这个数组的某个元素,这个指针就与这个数组建立了关系,之后就可以用指针来完成对这个数组的全部操作,其作用类似于数组名且比数组名更具灵活性。
由于指针是变量,数组名是常量,所以与数组建立了关系的指针可以用前++和后++操作指针,有不少操作要比直接用数组名来得方便且据说时效要高。
以上全是个人理解与使用感受,供参考。
一、概念
数组:数组是用于储存多个相同类型数据的集合。
指针:指针相当于一个变量,但是它和不同变量不一样,它存放的是其它变量在内存中的地址。
二、赋值、存储方式、求sizeof、初始化等
1.赋值
同类型指针变量可以相互赋值,数组不行,只能一个一个元素的赋值或拷贝
2.存储方式
数组:数组在内存中是连续存放的,开辟一块连续的内存空间。数组是根据数组的下进行访问的,多维数组在内存中是按照一维数组存储的,只是在逻辑上是多维的。
数组的存储空间,不是在静态区就是在栈上。
指针:指针很灵活,它可以指向任意类型的数据。指针的类型说明了它所指向地址空间的内存。
指针:由于指针本身就是一个变量,再加上它所存放的也是变量,所以指针的存储空间不能确定。
3.求sizeof
数组:
数组所占存储空间的内存:sizeof(数组名)数组的大小:sizeof(数组名)/sizeof(数据类型)
指针:
在32位平台下,无论指针的类型是什么,sizeof(指针名)都是4,在64位平台下,无论指针的类型是什么,sizeof(指针名)都是8。
关于指针和数组求sizeof,我在之前的博客中写过,现将连接贴上:https://blog.csdn.net/cherrydreamsover/article/details/81589838
4.初始化
数组:
(1)char a[]={"Hello"};//按字符串初始化,大小为6.(2)char b[]={'H','e','l','l'};//按字符初始化(错误,输出时将会乱码,没有结束符)(3)char c[]={'H','e','l','l','o','\0'};//按字符初始化1234这里补充一个大家的误区,就是关于数组的创建和销毁,尤其是多维数组的创建与销毁。(1)一维数组:int* arr = new int[n];//创建一维数组delete[] arr;//销毁(2)二维数组:int** arr = new int*[row];//这样相当于创建了数组有多少行for(int i=0;i
<row;i++){arr[i] = new int[col]; 到这里才算创建好了} 释放for(int i="0;i<row;i++){delete[]" arr[i];}delete[] arr;
指针:
//(1)指向对象的指针:(()里面的值是初始化值)int *p=new int(0) ; delete p;//(2)指向数组的指针:(n表示数组的大小,值不必再编译时确定,可以在运行时确定)int *p=new int[n]; delete[] p;//(3)指向类的指针:(若构造函数有参数,则new Class后面有参数,否则调用默认构造函数,delete调用析构函数)Class *p=new Class; delete p;//(4)指针的指针:(二级指针)int **pp=new (int*)[1];pp[0]=new int[6];delete[] pp[0];12345678910这里我们区分两个重要的概念:指针数组、数组指针。
(1)指针数组:它实际上是一个数组,数组的每个元素存放的是一个指针类型的元素。
int* arr[8];//优先级问题:[]的优先级比*高//说明arr是一个数组,而int*是数组里面的内容//这句话的意思就是:arr是一个含有8和int*的数组1234
请点击输入图片描述
(2)数组指针:它实际上是一个指针,该指针指向一个数组。
int (*arr)[8];//由于[]的优先级比*高,因此在写数组指针的时候必须将*arr用括号括起来//arr先和*结合,说明p是一个指针变量//这句话的意思就是:指针arr指向一个大小为8个整型的数组。1234
请点击输入图片描述
三、传参
数组:
数组传参时,会退化为指针,所以我们先来看看什么是退化!(1)退化的意义:C语言只会以值拷贝的方式传递参数,参数传递时,如果只拷贝整个数组,效率会大大降低,并且在参数位于栈上,太大的数组拷贝将会导致栈溢出。(2)因此,C语言将数组的传参进行了退化。将整个数组拷贝一份传入函数时,将数组名看做常量指针,传数组首元素的地址。
1.一维数组的传参
#include
//传参方式正确//用数组的形式传递参数,不需要指定参数的大小,因为在一维数组传参时,形参不会真实的创建数组,传的只是数组首元素的地址。(如果是变量的值传递,那么形参就是实参的一份拷贝)void test(int arr[]){}//传参方式正确//不传参数可以,传递参数当然也可以void test(int arr[10]){}//传参方式正确//一维数组传参退化,用指针进行接收,传的是数组首元素的地址void test(int *arr){}//传参方式正确//*arr[20]是指针数组,传过去的是数组名void test2(int *arr[20]){}//传参方式正确//传过去是指针数组的数组名,代表首元素地址,首元素是个指针向数组的指针,再取地址,就表示二级指针,用二级指针接收void test2(int **arr){}int main(){int arr[10] = {0};int *arr2[20] = {0};test(arr);test2(arr2);}2.二维数组的传参
//传参正确//表明二维数组的大小,三行五列void test(int arr[3][5]){}//传参不正确//二维数组的两个方括号,不能全部为空,也不能第二个为空,只能第一个为空void test(int arr[][]){}//传参正确//可以写成如下这样传参形式,但是不能写int arr[3][]void test(int arr[][5]){}//传参不正确//arr是一级指针,可以传给二维数组,但是不能正确读取void test(int *arr){}//传参不正确//这里的形参是指针数组,是一维的,可以传参,但是读取的数据不正确void test(int* arr[5]){}//传参正确//传过去的是二维数组的数组名,即数组首元素的地址,也就是第一行的地址,第一行也是个数组,用一个数组指针接收void test(int (*arr)[5]){}//传参不正确//可以传参,但是在读取的时候会有级别不同的问题void test(int **arr){}int main(){int arr[3][5] = {0};test(arr);}指针:
1.一级指针传参
当函数参数部分是一级指针时,可以接受什么参数例如:test(int*p)
(1)可以是一个整形指针(2)可以是整型变量地址(3)可以是一维整型数组数组名
#include
void print(int *p, int sz){int i = 0;for(i=0; i
<sz; i++){printf("%d\n", *(p+i));}}int main(){int arr[10]="{1,2,3,4,5,6,7,8,9};int" *p="arr;int" sz="sizeof(arr)/sizeof(arr[0]);//一级指针p,传给函数print(p," sz);return 0;}2.二级指针传参
即当函数参数部分是二级指针时,可以接受什么参数例如:test(int**p)
(1)二级指针变量(2)一级指针变量地址(3)一维指针数组的数组名
#include
void test(int** ptr){printf("num = %d\n", **ptr);}int main(){int num = 10;int*p = #int **pp = &p;test(pp);test(&p);return 0;}四、函数指针、函数指针数组、函数指针数组的指针
1.函数指针
void test(){printf("hehe\n");}//pfun能存放test函数的地址void (*pfun)();函数指针的形式:类型(*)( ),例如:int (*p)( ).它可以存放函数的地址,在平时的应用中也很常见。
2.函数指针数组
形式:例如int (*p[10])( );因为p先和[ ]结合,说明p是数组,数组的内容是一个int (*)( )类型的指针函数指针数组在转换表中应用广泛
3.函数指针数组的指针
指向函数指针数组的一个指针,也就是说,指针指向一个数组,数组的元素都是函数指针
void test(const char* str){printf("%s\n", str);}int main(){//函数指针pfunvoid (*pfun)(const char*) = test;//函数指针的数组pfunArrvoid (*pfunArr[5])(const char* str);pfunArr[0] = test;//指向函数指针数组pfunArr的指针ppfunArrvoid (*(*ppfunArr)[10])(const char*) = &pfunArr;return 0;}
c语言指针用法有哪些 如何应用
在下文中,我将会告诉大家c语言的指针都有哪些用法。
c语言指针用法 int t 定义整型变量
int *p:p为指向整型数据的指针变量。
int a[n]:定义整型数组a,它有n个元素。
int *p[n]:定义指针数组p,它由n个指向整形数据的指针元素组成。
int (*p)[n]:p为指向含n个元素的一维数组的指针变量。
int f():f为返回整型函数值的函数。
int *f():p为返回一个指针的函数,该指针指向整型数据。
int (*f)():p为指向函数的指针,该函数返回一个整形值。
int **p:p是一个指针变量,它指向一个指向整形数据的指针变量。
c语言的应用 驱动程序,无论是操作系统或者单片机,对硬件的驱动除了汇编(比较繁琐),都是用C语言来编写。
系统服务,由于操作系统都提供了C语言的API,并且C语言的执行效率比较高,所以用C语言来写系统服务是最适合不过的。
应用软件,由于C语言没有成熟的开发框架,所以不适合开发大型应用程序。但也有一些对效率要求比较高的程序使用C语言开发,如Git。
应用范围广泛,具备很强的数据处理能力,不仅仅是在软件开发上,而且各类科研都需要用到C语言,适于编写系统软件,三维,二维图形和动画。具体应用比如单片机以及嵌入式系统开发。
C语言是面向过程语言,C语言通过windows平台下编译的,是直接运行在windows平台下的,而Java始终是运行在他的虚拟机之上的;所以理论上C语言能做一些相对于比较底层的工作,像Java就不能编写Windows病毒。
数组指针怎么使用
可以这样来
#include
#include
#include
int main()
{
char *d[4];
d[0]=malloc(1*sizeof(char));
strcpy(d[0],"web");
printf("%s",d[0]);
return 0;
}
此时*d[]为一个指向指针的指针
d[0]=malloc(1*sizeof(char));
此句话的意思是,分配1个字节的内存空间给d[0],也就意味着d这个数组的第一个元素[0]内保存的是一个1字节空间的地址,因为d[0]指向一个地址!
你可以这样理解
对于普通的数组(例如 char d[4];),在内存中是这样的
变量:内存地址:内容
d[0]:0x400000h:'w'
d[1]:0x400001h:'e'
d[2]:0x400002h:'b'
d[3]:0x400003h:'\0'
而在此处 char *d[4]; *d[0] = malloc(1*sizeof(char));,是这样的'
d[0]:0x400000h:0x400100h
d[1]:0x400001h:0x400101h
d[2]:0x400002h:0x400102h
d[3]:0x400003h:0x400103h
而在内存0x400100h处
才保存着"web\0"
不好意思,我上面说错了,应该是
d[0]:0x400000h:0x400100h
d[1]:0x400001h:NULL
d[2]:0x400002h:NULL
d[3]:0x400003h:NULL
而在内存0x400100h处
0x400100h:'w'
0x400101h:'e'
0x400102h:'b'
0x400103h:'\0'
对应的,你的代码执行后,内存变量应该是这样的
d[0]:0x400000h:0x400100h
d[1]:0x400001h:0x400104h
d[2]:0x400002h:0x400108h
d[3]:0x400003h:NULL
在内存0x400100处:
0x400100h:'w'
0x400101h:'e'
0x400102h:'b'
0x400103h:'\0'
0x400104h:随机字符
......
0x400108h:随机字符
以上内存地址是以字节方式内存对齐的,用一些编译器编译后可能地址会有些问题(比如d[0]到d[1]不止差了4,可能是8),但总体框架就是这样!
指针数组怎么使用
在C语言和C++等语言中,数组元素全为同一类指针变量的数组称为指针数组,指针数组中的元素都必须具有相同的存储类型、指向相同数据类型的指针变量。指针数组比较适合用来指向若干个字符串,使字符串处理更加方便、灵活。一维指针数组的定义形式为:“类型名 *数组标识符[数组长度]”。
C语言 指针数组的使用
char
*name[4];是一个指针数组,每个成员name[i]都是一个字符指针。
printf函数打印字符串时传的参数就是一个字符指针。
所有printf("%s",
name[4]);
--针对你的问题
printf(“%d”,
name[4]);打印出来的是一个字符串的地址信息。
printf("%d",
*name[4]);打印出来的是name[4]指针所指向的字符对象的值。
P是指针数组,每个指针指向了A的二维起始地址p[i]=a[i];这里是赋值的地址,不是数据。p[0]就是a[0]地址,p[0][0]就是a[0][0]
一维数组a[]:a=a[0]
二维数组a[2][2]的储存原理:
a[]储存各行第一个的地址,,a[0][n]的n为读取地址:a[0]+n*sizeof(int)的值。
*p[]:由于[]的优先级高于*,所以int *p[3]说明p是一个内含三个指针元素的数组,每个元素都指向int的指针。由于程序中p[i]=a[i],所以*p[0]=a[0][0]。(指向各数组的第一个)
p[0]与p[1]有p[0]指向的数组隔开。
而int (*p)[2]是说明p指向一个内含三个int类型值的数组。(这是指向一维数组)
备注是//...或/*...*/
int **a可以表达一个二维数组。为什么呢?
你可以这么理解 * p[3]表达的是一个数组指针,这个指针 p[0]表示的是数组存的第一个地址。而数组实际上是一串连续地址的块。每一个小块存着一个内容。每次访问数组时,你为什么可以用数组名+下标访问呢? //比如a[i];
实际上就是 访问*a+i; * 这个符号可以表示你存的是变量的地址。而数组地址的第一位为int类型变量存的地址(你可以直接使用数组的地址加上单位变量的空间的大小去访问下个元素)。在这个程序中int *p[3] 表示可以存三个int 类型的地址。而p正好把二维数组的三个一维数组的开头的int类型变量的地址给存起来了。
给你举个例子
我现在输出的是地址,是不是连续的?
用*运算符获取地址的内容。
实际上p[i]与 *(p+i)是表达的是同一个意思。只不过只有数组可以定义一个连续的空间
(数组的第一个地址是随机的其他的是连续的。)单独用指针的话会随机分配的
数组的指针可以存三个地址。当然可以访问二维数组了。
分析指针数组与数组指针的区别和应用
这两个名字不同当然所代表的意思也就不同。从英文解释或中文全称看就比较容易理解。
指针数组:array of pointers,即用于存储指针的数组,也就是数组元素都是指针
数组指针:a pointer to an array,即指向数组的指针
还要注意的是他们用法的区别,下面举例说明。
int* a[4] 指针数组
表示:数组a中的元素都为int型指针
元素表示:*a[i] *(a[i])是一样的,因为[]优先级高于*
int (*a)[4] 数组指针
表示:指向数组a的指针
元素表示:(*a)[i]
注意:在实际应用中,对于指针数组
typedef int* pInt;
pInt a[4];
这跟上面指针数组定义所表达的意思是一样的,只不过采取了类型变换。
代码演示如下:
#include
using namespace std; int main(){int c[4]={1,2,3,4};int *a[4]; //指针数组int (*b)[4]; //数组指针b=&c;//将数组c中元素赋给数组afor(int i=0;i<4;i++){a[i]=&c[i];}//输出看下结果cout<<*a[1]<
<endl; 输出2就对cout<<(*b)[2]<<endl; 输出3就对return 0;}
注意:定义了数组指针,该指针指向这个数组的首地址,必须给指针指定一个地址,容易犯的错得就是,不给b地址,直接用(*b)[i]=c[i]给数组b中元素赋值,这时数组指针不知道指向哪里,调试时可能没错,但运行时肯定出现问题,使用指针时要注意这个问题。但为什么a就不用给他地址呢,a的元素是指针,实际上for循环内已经给数组a中元素指定地址了。但若在for循环内写*a[i]=c[i],这同样会出问题。总之一句话,定义了指针一定要知道指针指向哪里,不然要悲剧。
怎么正确地给指向数组的指针赋值
array相当于&array[0],也就是第一个元素的地址。所以第一种情况要用&array,第二种情况直接用array
用同类型的值进行赋值.
比如
int (*p)[5];可以用同类型的
int (*q)[5];给它赋值.
也可以
int a[10][5];然后用任意一行首地址赋值.
如
p=a;p=&a[1];等等.
这个是最基础的用法, 也可以强制转换, 实现一维数组 或者其他连续内存, 以数组方式访问.
c++指针数组怎么用
指针数组就是数组中每个元素都指向一个实体的地址,比如int型的指针数组,每一个元素代表一个int型数组的地址。
指针数组,是指一个数组里面装着指针;
一下是指针数组的使用案例:
#include "stdafx.h"
#include
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
int *ptr[5]; //指针数组
int p = 5, p2 = 8, *page, *page2;
page = &p;
page2 = &p2;
ptr[0] = &p;
ptr[1] = page2;
cout << *ptr[0] << endl;
cout << *page << endl;
cout << *ptr[1] << endl;
return 0;
}
运行结果:
指针数组如何赋值
如果没有描述错误的话,指针数组用法如下:
int * p[3];
int i;
int a,b,c;
p[0]=&a;
p[1]=&b;
p[2]=&c;
但根据我的经验,通常情况下没有必要使用指针数组的,而是使用指向数组的指针,所以怀疑题主描述错误。
如果是指向数组的指针的话,对数组初始化如下 :
int *p;
int a[3];
int i;
p=&a[0];
for(i=0;i<3;i++){
*p++=0;
}