收藏 分享(赏)

数组指针与字符串.ppt

上传人:ysd1539 文档编号:8428109 上传时间:2019-06-25 格式:PPT 页数:115 大小:1.05MB
下载 相关 举报
数组指针与字符串.ppt_第1页
第1页 / 共115页
数组指针与字符串.ppt_第2页
第2页 / 共115页
数组指针与字符串.ppt_第3页
第3页 / 共115页
数组指针与字符串.ppt_第4页
第4页 / 共115页
数组指针与字符串.ppt_第5页
第5页 / 共115页
点击查看更多>>
资源描述

1、第六章 数组 指针与字符串,C+语言程序设计,本章主要内容,数组 指针 动态存储分配 指针与数组 指针与函数 字符串,数组的概念,数组是具有一定顺序关系的若干相同类型对象的集合体,组成数组的对象称为该数组的元素。 数组属于构造类型。,数 组,一维数组的声明与引用,一维数组的声明 类型说明符 数组名 常量表达式 ;例如: int a10; 表示 a 为整型数组,有10个元素:a0.a9,引用 必须先声明,后使用。 只能逐个引用数组元素,而不能一次引用整个数组 例如:a0=a5+a7-a2*3,数 组,例6.1一维数组的声明与引用,#include void main() int A10,B10;

2、int i;for(i=0;i10;i+)Ai=i*2-1;B10-i-1=Ai;,数 组,for(i=0;i10;i+)cout“A“i“=“Ai;cout“ B“i“=“Biendl; ,一维数组的存储顺序,数组元素在内存中顺次存放,它们的地址是连续的。,数组名字是数组首元素的内存地址。 数组名是一个常量,不能被赋值。,数 组,一维数组的初始化,可以在编译阶段使数组得到初值: 在声明数组时对数组元素赋以初值。 static int a10=0,1,2,3,4,5,6,7,8,9; 可以只给一部分元素赋初值。 例如:static int a10=0,1,2,3,4; 在对全部数组元素赋初值时

3、,可以不指定数组长度。 例如:static int a=1,2,3,4,5,数 组,#include void main() int i;static int f20=1,1; for(i=2;i20;i+) fi=fi-2+fi-1; for(i=0;i20;i+) if(i%5=0) coutendl;cout.width(12); coutfi; ,例:用数组来处理求Fibonacci数列问题,例:用数组来处理求Fibonacci数列问题,运行结果:1 1 2 3 5 8 13 21 34 5589 144 233 377 610987 1597 2584 4181 6765,多维数组的

4、声明,声明:数据类型 数组名常量表达式1常量表达式2; int a53;表示a为整型二维数组,其中第一维有5个下标(04),第二维有3个下标(02),数组的元素个数为15,可以用于存放5行3列的整型数据。 引用:数组名整型表达式1整型表达式2;,数 组,存储顺序 按行存放,上例中数组a的存储顺序为:,二维数组的声明 类型说明符 数组名常量表达式1常量表达式2 例如:float a34;,引用 例如:b12=a23/2,下标不要越界,二维数组的声明及引用,数 组,将所有数据写在一个内,按顺序赋值 例如:static int a34=1,2,3,4,5,6,7,8,9,10,11,12; 分行给二

5、维数组赋初值 例如:static int a34=1,2,3,4,5,6,7,8,9,10,11,12; 可以对部分元素赋初值 例如:static int a34=1,0,6,0,0,11;,二维数组的初始化,数 组,数组作为函数参数,数组元素作实参,与单个变量一样。 数组名作参数,形参和实参都应是数组名,类型要一样,传送的是数组首地址。对形参数组的改变会直接影响到实参数组。,数 组,例6-2 使用数组名作为函数参数,主函数中初始化一个34矩阵并将每个元素都输出,然后调用子函数,分别计算每一行的元素之和,将和直接存放在每行的第一个元素中,返回主函数之后输出各行元素的和。,数 组,#includ

6、e void RowSum(int A4, int nrow) int sum;for (int i = 0; i nrow; i+)sum = 0;for(int j = 0; j 4; j+)sum += Aij;cout “Sum of row “ i “ is “ sum endl;Ai0=sum; ,void main(void) int Table34 = 1,2,3,4,2,3,4,5,3,4,5,6;for (int i = 0; i 3; i+) for (int j = 0; j 4; j+)cout Tableij “ “;cout endl;RowSum(Table,3

7、);for (int i = 0; i 3; i+)cout Tablei0; ,运行结果: 1 2 3 4 2 3 4 5 3 4 5 6 Sum of row 0 is 10 Sum of row 1 is 14 Sum of row 2 is 18 10 14 18,对象数组(*),声明:类名 数组名元素个数; 访问方法: 通过下标访问数组名下标.成员名,数 组,对象数组的初始化,数组中每一个元素对象被创建时,系统都会调用该类构造函数初始化该对象。 通过初始化列表赋值。 例: Point A2=Point(1,2),Point(3,4); 如果没有为数组元素指定显式初始值,数组元素便使用

8、默认值初始化(调用默认构造函数)。,数 组,数组元素所属类的构造函数,若不声明构造函数,则采用默认构造函数。 各元素对象的初值要求为相同的值时,可以声明具有默认形参值的构造函数。 各元素对象的初值要求为不同的值时,需要声明带形参的构造函数。 当数组中每一个对象被删除时,系统都要调用一次析构函数。,数 组,例6-3 对象数组应用举例,/Point.h #if !defined(_POINT_H) #define _POINT_H class Point public:Point(); Point(int xx,int yy);Point();void Move(int x,int y);int

9、GetX() return X;int GetY() return Y;private:int X,Y; ; #endif,数 组,/6-2.cpp #include #include “Point.h“ Point:Point() X=Y=0;cout“Default Constructor called.“endl; Point:Point(int xx,int yy) X=xx;Y=yy;cout “Constructor called.“endl; Point :Point() cout“Destructor called.“endl; void Point :Move(int x,i

10、nt y) X=x; Y=y; ,/6-3.cpp #include #include “Point.h“ int main( ) cout“Entering main.“endl;Point A2;for(int i=0;i2;i+)Ai.Move(i+10,i+20);cout“Exiting main.“endl;return 0; ,运行结果: Entering main. Default Constructor called. Default Constructor called. Exiting main. Destructor called. Destructor called.

11、,关于内存地址,对象的访问方式 通过对象名访问 通过地址访问 地址运算符: 则&var 表示变量var在内存中的起始地址,声明(可以声明各种类型的指针) 例: int i;int *i_pointer=指向整型变量的指针,指针变量的概念,指针:内存地址,用于间接访问内存单元。 指针变量: 用于存放地址的变量。,引用 例1: i=3; (访问方式) 例2: *i_pointer=5;,指 针,指针变量的初始化,语法形式存储类型 数据类型 *指针名初始地址; 例: int a,*pa= 注意事项 用变量地址作为初值时,该变量必须在指针初始化之前已说明过,且变量类型应与指针类型一致。 可以用一个已赋

12、初值的指针去初始化另一个指针变量。 不要用一个内部 auto 变量去初始化 static 指针。,指 针,指针变量的赋值运算,指针变量名=地址 “地址”中存放的数据类型与指针的类型必须相符。 向指针变量赋的值必须是地址常量或变量,不能是普通整数。但可以赋值为整数0,表示空指针。 指针的类型是它所指向变量的类型,而不是指针本身数据值的类型,任何一个指针本身的数据值都是unsigned long int型。 允许声明指向 void 类型的指针。该指针可以被赋予任何类型对象的地址。 例: void *general;,指 针,例6-5 指针的声明、赋值与使用,#include void main(

13、) int *i_pointer; int i; i_pointer= ,指 针,程序运行的结果是: Output int i=10 Output int pointer i=10,例6-6 void类型指针的使用,void vobject; /错,不能声明void类型的变量 void *pv; /对,可以声明void类型的指针 int *pint; int i; void main() pv = ,指 针,指向常量的指针,不能通过该指针来改变所指对象的值,但指针本身可以改变,可以指向另外的对象。例1char *name1 = “John“; *name1=A; 例2 const char *

14、name1 = “John“; char s=“abc“; name1=s; *name1=1;,指 针,指针类型的常量,若声明指针常量,则指针本身的值不能被改变。例: char *const name2 = “John“; name2=“abc“;,指针变量的算术运算,指针与整数的加减运算 指针 p 加上或减去 n ,其意义是指针当前指向位置的前方或后方第 n 个数据的地址。 这种运算的结果值取决于指针指向的数据类型。 指针自增,自减运算 指向下一个或前一个数据。 例如:y=*px+ 相当于 y=*(px+) (*和+优先级相同,自右向左运算),指 针,pb-1,pb,pb+1,pb+2,*

15、(pb-1),*pb,*(pb+1),*(pb+2),long *pb,关系运算 指向相同类型数据的指针之间可以进行各种关系运算。 指向不同数据类型的指针,以及指针与一般整数变量之间的关系运算是无意义的。 指针可以和零之间进行等于或不等于的关系运算。例如:p=0或p!=0 赋值运算 向指针变量赋的值必须是地址常量或变量,不能是普通整数。但可以赋值为整数0,表示空指针。,指针变量的关系运算,指 针,指向数组元素的指针,声明与赋值 例: int a10, *pa;pa= 通过指针引用数组元素 *pa就是a0,*(pa+1)就是a1,. ,*(pa+i)就是ai. ai, *(pa+i), *(a+

16、i), pai都是等效的。 不能写 a+,因为a是数组首地址是常量。,指 针,例6-7,设有一个int型数组a,有10个元素。用三种方法输出各元素: 使用数组名和下标 使用数组名和指针运算 使用指针变量,指 针,main() int a10;int i;for(i=0; iai;coutendl;for(i=0; i10; i+)coutai; ,使用数组名和下标,main( ) int a10;int i;for(i=0; iai;coutendl;for(i=0; i10; i+)cout*(a+i); ,使用数组名指针运算,使用指针变量,main( ) int a10;int *p,i;

17、for(i=0; iai;coutendl;for(p=a; p(a+10); p+)cout*p; ,指针数组,每个数组的元素是指针变量。 声明形式: 类型名 *数组名数组长度 例:Point *pa2;pa由pa0,pa1两个指针组成,指 针,例6-8 利用指针数组存放单位矩阵,#include void main() int line1 =1,0,0; int line2 =0,1,0; int line3 =0,0,1; int *p_line3; p_line0=line1; p_line1=line2;p_line2=line3;,指 针,/输出单位矩阵cout“Matrix te

18、st:“endl;for(int i=0;i3;i+) for(int j=0;j3;j+) coutp_lineij“ “; coutendl; ,输出结果为: Matrix test: 1,0,0 0,1,0 0,0,1,例6-9 二维数组举例,#include void main( ) int array223=11,12,13,21,22,23;for(int i=0;i2;i+) cout*(array2+i)endl; for(int j=0;j3;j+)cout*(*(array2+i)+j)“ “; coutendl; ,指 针,在某次运行之后,程序的输出结果为: 0X0065

19、FDE0 11,12,13 0X0065FDEC 21,22,23,以指针作为函数参数,以地址方式传递数据,可以用来返回函数的处理结果。 实参是数组名时形参可以是指针变量。,指针与函数,例6.10,题目:读入三个浮点数,将整数部分和小数部分分别输出。 #include void splitfloat(float x, int *intpart, float *fracpart) *intpart = int(x); *fracpart = x - *intpart; ,指针与函数,void main(void) int i, n;float x, f;cout x;splitfloat(x,

20、,程序的运行结果:Enter three (3) floating point numbers 4.7 Integer Part is 4 Fraction Part is 0.7 8.913 Integer Part is 8 Fraction Part is 0.913 -4.7518 Integer Part is -4 Fraction Part is -0.7518,例: 输出数组元素的内容和地址,#include #include void Array_Ptr(long *P, int n) int i;cout “In func, address of array is “ un

21、signed long(P) endl;cout “Accessing array in the function using pointers“ endl;for (i = 0; i n; i+) cout “ Address for index “ i “ is “ unsigned long(P+i);cout “ Value is “ *(P+i) endl; ,指针与函数,void main(void) long list5 = 50, 60, 70, 80, 90;cout “In main, address of array is “ unsigned long(list) en

22、dl;cout endl;Array_Ptr(list,5); ,程序的运行结果:In main, address of array is 6684132In func, address of array is 6684132 Accessing array in the function using pointersAddress for index 0 is 6684132 Value is 50Address for index 1 is 6684136 Value is 60Address for index 2 is 6684140 Value is 70Address for in

23、dex 3 is 6684144 Value is 80Address for index 4 is 6684148 Value is 90,用指向常量的指针做形参,#include const int N=6; void print(const int *p,int n); void main( ) int arrayN;for(int i=0;iarrayi;print(array,N); ,指 针,void print(const int *p, int n) cout“*p;for(int i=1;in;i+)cout“.“*(p+i);cout“endl; ,指针型函数,当函数的返回

24、值是地址时,该函数就是指针型函数。 声明形式:数据类型 *函数名( ),指针与函数,声明形式数据类型 (*函数指针名)(形参表);含义: 数据指针指向数据存储区,而函数指针指向的是程序代码存储区。,指向函数的指针,指针与函数,例6-11 函数指针,#include void print_stuff(float data_to_ignore); void print_message(float list_this_data); void print_float(float data_to_print); void (*function_pointer)(float); void main() f

25、loat pi = (float)3.14159;float two_pi = (float)2.0 * pi;,指针与函数,print_stuff(pi);function_pointer = print_stuff;function_pointer(pi);function_pointer = print_message;function_pointer(two_pi);function_pointer(13.0);function_pointer = print_float;function_pointer(pi);print_float(pi); ,void print_stuff(f

26、loat data_to_ignore) cout“This is the print stuff function.n“; void print_message(float list_this_data) cout“The data to be listed is “ list_this_dataendl; void print_float(float data_to_print) cout“The data to be printed is “ data_to_printendl; ,程序运行结果: This is the print stuff function. This is the

27、 print stuff function. The data to be listed is 6.283180 The data to be listed is 13.000000 The data to be printed is 3.141590 The data to be printed is 3.141590,对象指针(*),声明形式 类名 *对象指针名; 例 Point A(5,10); Piont *ptr; ptr=,指 针,对象指针应用举例,int main() Point A(5,10);Point *ptr;ptr= ,指 针,错误的例子,class Fred; cla

28、ss Barney Fred x; ; class Fred Barney y;,指 针,正确的程序,class Fred; class Barney Fred *x; ; class Fred Barney y;,指 针,this指针(1),是隐含于每一个类的成员函数中的特殊指针。 明确地指出了成员函数当前所操作的数据所属的对象。 当通过一个对象调用成员函数时,系统先将该对象的地址赋给this指针,然后调用成员函数,成员函数对对象的数据成员进行操作时,就隐含使用了this指针。,指 针,this指针(2),例如:Point类的构造函数。 Point:Point(int xx,int yy)

29、X=xx; Y=yy; 相当于: this-X=xx; this-Y=yy;,指 针,指向类的非静态成员的指针,通过指向成员的指针只能访问公有成员 声明指向成员的指针 声明指向公有数据成员的指针 类型说明符 类名:*指针名; 声明指向公有函数成员的指针 类型说明符 (类名:*指针名)(参数表);,指 针,指向类的非静态成员的指针,指向数据成员的指针 说明指针应该指向哪个成员 指针名=&类名:数据成员名; 通过对象名(或对象指针)与成员指针结合来访问数据成员 对象名.* 类成员指针名 或: 对象指针名*类成员指针名,指 针,指向类的非静态成员的指针,指向函数成员的指针 初始化 指针名=类名:函数

30、成员名; 通过对象名(或对象指针)与成员指针结合来访问函数成员 (对象名.* 类成员指针名)(参数表) 或: (对象指针名*类成员指针名)(参数表),指 针,指向类的非静态成员的指针,例6-13 访问对象的公有成员函数的不同方式 void main() Point A(4,5); Point *p1= ,指 针,指向类的静态成员的指针,对类的静态成员的访问不依赖于对象。 可以用普通的指针来指向和访问静态成员。 例6-14 通过指针访问类的静态数据成员。 例6-15 通过指针访问类的静态函数成员。,指 针,例6-14通过指针访问类的静态数据成员,#include class Point publ

31、ic: Point(int xx=0, int yy=0) X=xx;Y=yy;countP+;Point(Point ,指 针,void main() int *count= ,例6-15通过指针访问类的静态函数成员,#include class Point public: static void GetC() cout“ Object id=“countPendl;private: int X,Y;static int countP; ; int Point:countP=0;,指 针,void main() void (*gc)()=Point:GetC; Point A(4,5); c

32、out“Point A,“A.GetX()“,“A.GetY();gc(); Point B(A); cout“Point B,“B.GetX()“,“B.GetY();gc(); ,动态申请内存操作符 new(*),new 类型名T(初值列表)功能:在程序执行期间,申请用于存放T类型对象的内存空间,并以初值列表赋以初值。结果值:成功:T类型的指针,指向新分配的内存 失败:0(NULL),动态存储分配,释放内存操作符delete(*),delete 指针P功能:释放指针P所指向的内存。P必须是new操作的返回值。,动态存储分配,例6-16 动态创建对象举例,#include class Poi

33、nt public:Point() X=Y=0; cout“Default Constructor called.n“;Point(int xx,int yy) X=xx; Y=yy; cout “Constructor called.n“; Point() cout“Destructor called.n“; int GetX() return X;int GetY() return Y;void Move(int x,int y) X=x; Y=y; private:int X,Y; ;,动态存储分配,int main() cout“Step One:“endl;Point *Ptr1=n

34、ew Point;delete Ptr1; cout“Step Two:“endl;Ptr1=new Point(1,2);delete Ptr1;return 0; ,运行结果: Step One: Default Constructor called. Destructor called. Step Two: Constructor called. Destructor called.,例6-17动态创建对象数组举例,#include class Point /类的声明同例6-16,略 ; int main() Point *Ptr=new Point2; Ptr0.Move(5,10);

35、 Ptr1.Move(15,20); cout“Deleting.“endl;delete Ptr; return 0; ,动态存储分配,运行结果: Default Constructor called. Default Constructor called. Deleting. Destructor called. Destructor called.,例6-18动态数组类,#include class Point /类的声明同例6-16 ; class ArrayOfPoints public:ArrayOfPoints(int n) numberOfPoints=n; points=ne

36、w Pointn; ArrayOfPoints() cout“Deleting.“endl;numberOfPoints=0; delete points; Point,void main() int number;coutnumber;ArrayOfPoints points(number); points.Element(0).Move(5,10); points.Element(1).Move(15,20); ,运行结果如下: Please enter the number of points:2 Default Constructor called. Default Construct

37、or called. Deleting. Destructor called. Destructor called.,动态创建多维数组,new 类型名T下标表达式1下标表达式2;如果内存申请成功,new运算返回一个指向新分配内存首地址的指针,是一个T类型的数组,数组元素的个数为除最左边一维外各维下标表达式的乘积。例如: char (*fp)3; fp = new char23;,char (*fp)3;,fp,fp+1,例6-19动态创建多维数组,#include void main( ) float (*cp)98;int i,j,k;cp = new float898;for (i=0;

38、i8; i+)for (j=0; j9; j+)for (k=0; k9; k+)*(*(*(cp+i)+j)+k)=i*100+j*10+k;,动态存储分配,for (i=0; i8; i+) for (j=0; j9; j+) for (k=0; k8; k+)coutcpijk“ “; coutendl;coutendl; ,动态存储分配函数,void *malloc( size ); 参数size:欲分配的字节数。 返回值: 成功,则返回void型指针。 失败,则返回空指针。 头文件: 和 ,动态存储分配,动态内存释放函数,void free( void *pointer ); 参数p

39、ointer:指针,指向需释放的内存。 返回值:无 头文件: 和 ,动态存储分配,浅拷贝与深拷贝(*),浅拷贝 实现对象间数据元素的一一对应复制。 深拷贝 当被复制的对象数据成员是指针类型时,不是复制该指针成员本身,而是将指针所指的对象进行复制。,浅拷贝与深拷贝,例6-20对象的浅拷贝,#include class Point /类的声明同例6-16/ ; class ArrayOfPoints /类的声明同例6-18/ ;,浅拷贝与深拷贝,void main() int number;cinnumber;ArrayOfPoints pointsArray1(number); pointsAr

40、ray1.Element(0).Move(5,10); pointsArray1.Element(1).Move(15,20); ArrayOfPoints pointsArray2(pointsArray1); cout“Copy of pointsArray1:“endl;cout“Point_0 of array2: “pointsArray2.Element(0).GetX()“, “pointsArray2.Element(0).GetY()endl;cout“Point_1 of array2: “pointsArray2.Element(1).GetX()“, “pointsAr

41、ray2.Element(1).GetY()endl;,pointsArray1.Element(0).Move(25,30); pointsArray1.Element(1).Move(35,40); cout“After the moving of pointsArray1:“endl;cout“Point_0 of array2: “pointsArray2.Element(0).GetX()“, “pointsArray2.Element(0).GetY()endl;cout“Point_1 of array2: “pointsArray2.Element(1).GetX()“, “p

42、ointsArray2.Element(1).GetY()endl; ,运行结果如下: Please enter the number of points:2 Default Constructor called. Default Constructor called. Copy of pointsArray1: Point_0 of array2: 5, 10 Point_1 of array2: 15, 20 After the moving of pointsArray1: Point_0 of array2: 25, 30 Point_1 of array2: 35, 40 Delet

43、ing. Destructor called. Destructor called. Deleting. 接下来程序出现异常,也就是运行错误。,拷贝前,拷贝后,例6-21对象的深拷贝,#include class Point /类的声明同例6-16 ; class ArrayOfPoints public:ArrayOfPoints(ArrayOfPoints,浅拷贝与深拷贝,ArrayOfPoints :ArrayOfPoints (ArrayOfPoints void main() /同例6-20 ,程序的运行结果如下: Please enter the number of points:

44、2 Default Constructor called. Default Constructor called. Default Constructor called. Default Constructor called. Copy of pointsArray1: Point_0 of array2: 5, 10 Point_1 of array2: 15, 20 After the moving of pointsArray1: Point_0 of array2: 5, 10 Point_1 of array2: 15, 20 Deleting. Destructor called.

45、 Destructor called. Deleting. Destructor called. Destructor called.,拷贝前,拷贝后,用字符数组存储和处理字符串,字符数组的声明和引用(见例6-19),例:static char str8=112,114,111,103,114,97,109,0; static char str8=p,r,o,g,r,a,m,0; static char str8=“program“; static char str=“program“;,字符串 字符串常量,例如:“china“ 没有字符串变量,用字符数组来存放字符串 字符串以0为结束标志。

46、字符数组的初始化,字符串,例6-22 输出一个字符串,#include void main() static char c10=I, ,a,m, ,a, ,b,o,y; int i;for(i=0;i10;i+)coutci;coutendl; ,运行结果:I am a boy,字符串,例6-23输出一个钻石图形,#include void main() static char diamond5= , ,*, ,*, ,*, *, , , ,*, ,*, ,*, , ,*;int i,j;for (i=0;i5;i+) for(j=0;j5;j+)coutdiamondij; coutendl; ,运行结果:,* * * * *,字符串,字符串的输入/输出,方法 逐个字符输入输出 将整个字符串一次输入或输出 例:char c=“China“; coutc; 注意 输出字符不包括 0。 输出字符串时,输出项是字符数组名,输出时遇到0结束。 输入多个字符串时,以空格分隔;输入单个字符串时其中不能有空格。,字符串,用字符数组存储和处理字符串,char a4, *p1, *p2; 错误的: a=“abc“; cinp1; 正确的: p1=“abc“; p2=a; cinp2;,

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 企业管理 > 管理学资料

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报