IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> C++指针和数组 -> 正文阅读

[数据结构与算法]C++指针和数组

目录:

3、数组与指针

4、指针的潜在危险

``

3.数组与指针

3.1一维数组与指针

定义指针变量指向数组首元素,以指针变量名代替数组名,实现数组操作。(指针所指位置不变)
指针变量从前到后依次指向数组各元素,通过指针的取内容运算得到对应元素。(指针所指位置不断变化)
例:用下列数据初始化一维数组,并通过指针变量求元素的最大值。
8.2 6.5 3 9.7 12 2.8 7.6 15 10.3
定义实型指针p指向数组b的首元素,max 表示最大值;
以p 代替b,通过循环语句输出数组的各元素;
指针p 从第二个元素开始遍历数组,遍历过程中将比max 大的元素赋给max。

#include<iostream>
using namespace std;
int main(void) 
{
	double b[]={8.2,6.5,3,9.7,12,2.8,7.6,15.6,10.3};
	double *p=b,max=b[0];	           //p指向b的首元素,b即&b[0]                           
	cout<<"数组为:\n";
	for(int i=0;i<9;i++)
    { 	                                    // 输出数组
		cout<<p[i]<<'\t'; 	 // 指针变量名p代替数组名b
		if((i+1)%5==0)cout<<'\n';
	}
	cout<<endl;
    p++; 	// p 指向b[1],等同于p =&b[1]
	for(int i=0;i<9;i++)
    {  // 求最大元素
		if(*p>max)max=*p;  // *p是指针p所指元素的值
		p++; 	// p 后移一个元素
	}
	cout<<"元素的最大值为:"<<max<<"\n";
	return 0;
}

l指针变量只能代表其所指向的数组。当指针不再指向数组的首元素时,不能代表原数组。

3.2二维数组与指针

3.2.1元素指针与行指针

指向元素的指针简称元素指针。(指向元素)
二维数组元素指针与一维数组元素指针的定义方式相同,使用方法相似。如:

float b[3][4],*p;  // 定义二维数组b 和元素指针p
p=&b[0][0];        // p指向二维数组首元素 

行指针定义:

指向二维数组某行的指针简称行指针。(指向一行)
二维数组数组名b 是b[0]的地址(&b[0]),b[0]是二维数组的第一行,即b是指向二维数组第1行的行指针;且b[0]是由5 个元素组成的一维数组,故行指针也称为指向一维数组的指针。

数据类型 (*指针变量名)[二维数组列数];

数据类型与所指向的二维数组的数据类型相同;
下标为二维数组的列数,通常为整型常量表达式;
必须用“()”将指针变量名括起来。
如:

float (*p2)[5];

二维数组行地址、元素地址和元素之间的关系
对行地址进行取值运算可以得到元素地址;
对元素地址进行取值运算可得到二维数组的元素。

分类表示方法备注
行指针p2+i,&p2[i]下标为i行的行地址
元素指针*(p2+i)+j,p2[i]+j下标为i行j列的元素地址
元素* (* (p2+i)+j),*(p2[i]+j),p2[i] [j]下标为i行j列的元素

3.2.2元素指针使用二维数组

将二维数组作为一个行数×列数的一维数组
定义元素指针指向二维数组的第一行第一列元素;
一维数组名为指针变量名;
一维数组大小为二维数组的行数×列数。

3.2.3行指针使用二维数组

定义行指针指向二维数组的首行
行指针变量名代替二维数组名,实现二维数组的操作;
行指针取值运算得到元素地址,然后元素地址取值运算得到元素,完成对二维数组的操作。
注意元素指针与行指针使用的区别,指针所指的位置。

例:通过指针输出下列二维数组,并求各元素的和。
2 3 1 4 10
6 2 5 8 3
7 8 9 6 12

行指针p1:&b[0];p1+1:&b[1]
元素指针p2:&b[0] [0];p2+1:&b[0] [1]

#include<iostream>
using namespace std;
int main(void) 
{
	int b[3][5]={{2,3,1,4,10},{6,2,5,8,3},{7,8,9,6,12}},sum=0;
	int (*p1)[5]=b,*p2=&b[0][0],i,j;//p1=&b[0][0];p2=b;   语法错误
	for(i=0;i<3;i++)
    { // 输出二维数组
		for(j=0;j<5;j++)
			cout<<p1[i][j]<<'\t';
		cout<<endl;
	}
	for(i=0;i<3*5;i++)            // 求二维数组各元素的和
		sum+=p2[i];
	cout<<"和为:"<<sum<<'\n';
	return 0;
}

3.2.4字符数组与指针

字符型指针变量指向字符串
定义时用字符串对其初始化;
(1)用字符串对指针变量赋值。
如:

char *s1=" C++ Program",*s2; 
s2=" This is a string."; 

(2)直接引用字符型指针变量所指的字符数组

#include <iostream>
#include <cstring>
using namespace std;
int main(void) {
char *s1,*s2; 
s2=" This is a string."; 
char str[50],*s3=str;
cin.getline(s3,50);        // 输入字符数组cin.getline(字符数组名,字符个数,结束标志)
cout<<s3;                                                       // 输出字符数组
strcpy(s3,s1);                                                 // 复制字符数组
*s3=*s2;  
return 0;
}

结合前面讲过的二级指针,我们来看看使用的实例。

#include<iostream>
#include<string>
using namespace std;
int main()
{
	char *name[3]={"China","Japan","England"};//name数组中的元素分别定义为指向三个字符串的指针
	char **p;  //name[0]:China,
	for(int i=0;i<3;i++)
	{
		p=name+i;
		cout<<"name["<<i<<"]point to";
		cout<<*p<<endl;
	}
    return 0;
}

例:设计一个程序,将字符串中的字符逆序。如将“I am a student.”逆序为“.tneduts a ma I”。
数组str 存储字符串,指针s1 指向首元素,s2 指向尾元素。
当s1在s2前面时,将s1 和s2 所指的元素互换;然后s1 后移一个元素,s2 前移一个元素。

#include<iostream>
#include<string>
using namespace std;
int main(void) {
char str[100],*s1=str,*s2=str,t;
cout<<"请输入一个字符串:";
cin.getline(s1,100);
cout<<"输入的字符串是:";
cout<<s2<<endl; 			// s2等同str、s1

while(*s2) s2++;       // s2 指向结束标志(循环条件的含义?)
s2--;                                                 //前移一位,指向尾元素
while(s1<s2) {                                           // 当s1 在s2 前面时
t=*s1,*s1=*s2,*s2=t;                  // 交换s1 和s2 所指元素
s1++,s2--;   // s1 后移、s2前移(指针前有没有*的区别?)
}
cout<<"逆序后的字符串是:";
cout<<str<<endl; 
return 0;
}

指针使用数组
搞清指针所指位置
分清操作对象是指针本身还是指针所指内存空间
分清所使用的是元素指针还是行指针
充分理解指针的含义:地址;代表所指向的数组。

3.2.5指针数组

各元素为指针变量的数组。
普通数组中存储的是普通数据(数值),指针数组中存储的是地址。
定义:

存储类型 数据类型* 数组名[数组大小];

如:

float *p1[5];
#include <iostream>
using namespace std;
int main(void) {
double d[4]={1.0,1.1,1.2,1.3};
double * p[4];
for(int i=0;i<4;i++)
	p[i]=&d[i];
for(int i=0;i<4;i++)
	cout<<*p[i]<<',';
return 0;
}

4.指针的潜在危险

指针的让我们对内存的操作有了很大的自由性,同时也带来了潜在的危险。

4.1产生的原因:

1.定义指针变量的同时未对其进行初始化:指针在被定义的时候,如果程序不对其进行初始化的话,它会指向随机区域,因为任何指针变量(除了static修饰的指针变量)在被定义的时候是不会被置空的,它的默认值是随机的。

2.指针所指向的内存空间被释放时,却没有对该指针变量的值(即该指针原来指向的内存空间的地址)进行置空:我们在用库函数malloc开辟内存空间时,要检查返回值是否为空,如果为空,则开辟失败;如果不为空,则指针指向的是开辟的内存空间的首地址。指针指向的内存空间在用free()或者delete(注意delete只是一个操作符,而free()是一个函数)释放后,如果程序员没有对其置空或者其他的赋值操作,就会使其成为一个野指针。

3.指针操作超越变量作用域.

4.2危害:

上述问题在于,指针指向的内存空间已经无效了,而该指针变量的值(即该指针原来指向的内存空间的地址)没有被置空,解引用一个非空的无效指针是一个未被定义的行为,也就是说不一定导致段错误,野指针很难定位到是哪里出现的问题,在哪里这个指针就失效了,不好查找出错的原因。所以调试起来会很麻烦,有时候会需要很长的时间。

4.3规避的方法:

1.在定义一个指针时同时初始化为NULL;

int *p=NULL;

2.释放指针指向的内存空间时,将指针重置为NULL。

free(p1);			//只释放了p1指向的堆区空间  并没有将指针p1置为空
p1 = NULL;

3.使用时不要超出变量作用域,如使用数组时:

int a[3];
int*p=a;
cout<<p[5];
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-12-08 14:04:10  更:2021-12-08 14:06:01 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 2:34:00-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码