PowerPoint 演示文稿

Similar documents
内 容 提 要 指 针 持 久 动 态 内 存 分 配 字 符 串 ( 字 符 数 组 ) 2

Microsoft PowerPoint - 5. 指针Pointers.ppt [兼容模式]

没有幻灯片标题

C++ 程序设计 告别 OJ1 - 参考答案 MASTER 2019 年 5 月 3 日 1

《计算概论》课程 第十九讲 C 程序设计语言应用

第五讲 数组 一维数组 二维数组 数组作为函数参数

Microsoft PowerPoint - 4. 数组和字符串Arrays and Strings.ppt [兼容模式]

新版 明解C++入門編

《C语言程序设计》第2版教材习题参考答案

untitled

《C语言程序设计》教材习题参考答案

新版 明解C言語入門編

chap07.key

新・明解C言語入門編『索引』

试卷代号 ~1075 座位号 E 口 国家开放大学 ( 中央广播电视大学 )20]5 年秋季学期 " 开放本科 " 期末考试 C 十十语言程序设计 试题 同二二十斗 2016 年 1 月 巴叫一 1. 下面的保留字 ( ) 不能作为函数的返回类型 A. void B. int C. new D. l

试卷代号 :1075 座位号 rn 国家开放大学 ( 中央广播电视大学 )2015 年秋季学期 " 开放本科 " 期末考试 c+ 十语言程序设计试题 2016 年 1 月 t 问一 Urr-f 斗 士 1 1. 下面的保留字 ( ) 不能作为函数的返回类型 A. void B. int C. new

运算符重载 为什么要 运算符重载 那些运算符可以重载, 哪些不可以 如何实现运算符重载 实现方式 : 成员函数与非成员函数 类型转换 怎样实现对象与基本数据类型数据的运算 2

Generated by Unregistered Batch DOC TO PDF Converter , please register! 浙江大学 C 程序设计及实验 试题卷 学年春季学期考试时间 : 2003 年 6 月 20 日上午 8:3

一 关于内存 数据存储 变量分类 局部变量 全局变量 静态变量 请看下面代码, 分析变量类型? int pi = 3; int Area(int r, int *sum) { int b; static int c =0; b = pi * r * r; c += b; *sum = c; retu

新・解きながら学ぶC言語

C++ 程序设计 告别 OJ2 - 参考答案 MASTER 2019 年 5 月 3 日 1

北京大学

C++ 程序设计 OJ9 - 参考答案 MASTER 2019 年 6 月 7 日 1

OOP with Java 通知 Project 2 提交时间 : 3 月 14 日晚 9 点 另一名助教 : 王桢 学习使用文本编辑器 学习使用 cmd: Power shell 阅读参考资料

Microsoft Word - 《C语言开发入门》课程教学大纲-2.doc

PowerPoint 演示文稿

2015年计算机二级(C语言)模拟试题及答案(三)

第4章 栈和队列

Microsoft PowerPoint - 01_Introduction.ppt

Microsoft PowerPoint - 8. 运算符重载 Operator Overloading.pptx

2013 C 1 # include <stdio.h> 2 int main ( void ) 3 { 4 int cases, a, b, i; 5 scanf ("%d", & cases ); 6 for (i = 0;i < cases ;i ++) 7 { 8 scanf ("%d %d

C 1 # include <stdio.h> 2 int main ( void ) { 4 int cases, i; 5 long long a, b; 6 scanf ("%d", & cases ); 7 for (i = 0;i < cases ;i ++) 8 { 9

Microsoft PowerPoint - C语言课件-9-结构体.pptx

文件

C/C++程序设计 - 字符串与格式化输入/输出

第六讲 数组、指针与字符串

CC213

Microsoft PowerPoint - string_kruse [兼容模式]

《C语言程序设计》教材习题参考答案

网C试题(08上).doc

C 1

JAVA 单元 2.1 四则运算机 ( 一 ) 单元教学进度设计 教学环节 教学内容 教师学生活动活动 反馈 反馈课前作业完成情况 反馈加分 1. 下面哪些是合法的变量名? ( ) A.2variable 答案 :DEG B..variable2 解答 : C.._whatavariable A:/

c语言面试题总结.doc

NOWOER.OM m/n m/=n m/n m%=n m%n m%=n m%n m/=n 4. enum string x1, x2, x3=10, x4, x5, x; 函数外部问 x 等于什么? 随机值 5. unsigned char *p1; unsigned long *p

CC213

实验目的 (1) 熟练掌握顺序 分支 循环三种结构 (2) 会使用流程控制结构编写程序 第三章程序的流程控制 实验要求 (1) 掌握 if-else swith-case 的使用 (2) 掌握 while do-while for 的使用 (3) 掌握分支嵌套和循环嵌套 (4) 分析理解如何避免死循

OOP with Java 通知 Project 4: 4 月 18 日晚 9 点 关于抄袭 没有分数

C++ 程序设计 OJ1 - 参考答案 MASTER 2019 年 5 月 3 日 1

试卷代号 :1253 座位号 E 口 国家开放大学 ( 中央广播电视大学 )2014 年秋季学期 " 开放本科 " 期末考试 C 语言程序设计 A 试题 2015 年 1 月 E 四! 五 总分! 一 单选题 ( 每小题 2 分, 共 20 分 ) 1. 由 C 语言源程序文件编译而成的目标文件的默

期中考试试题讲解

エスポラージュ株式会社 住所 : 東京都江東区大島 東急ドエルアルス大島 HP: ******************* * 关于 Java 测试试题 ******

C/C++语言 - 分支结构

Microsoft Word - V1_ _王翔会计习题课二.docx

Ps22Pdf

新・解きながら学ぶJava

FY.DOC

Microsoft PowerPoint - 3. 函数Functionl.ppt [兼容模式]

Microsoft PowerPoint - 08 指针

C++ 程序设计 OJ4 - 参考答案 MASTER 2019 年 5 月 30 日 1

untitled

《C语言程序设计》期末程序设计练习题库.docx

2015年计算机二级(C语言)模拟试题及答案(四)

C++ 程序设计 OJ2 - 参考答案 MASTER 2019 年 5 月 3 日 1

技能竞赛C试题

C/C++ System Program Documentation

第4章 栈和队列

求出所有的正整数 n 使得 20n + 2 能整除 2003n n 20n n n 20n n 求所有的正整数对 (x, y), 满足 x y = y x y (x, y) x y = y x y. (x, y) x y =

( CIP) /. :, ( ) ISBN TP CIP ( 2005) : : : : * : : 174 ( A ) : : ( 023) : ( 023)

C/C++ - 文件IO

3.1 num = 3 ch = 'C' 2

主要内容 函数的定义 声明与调用 函数间的参数传递 函数嵌套与内联函数 形参带缺省值的函数与函数重载 数据的作用域 预编译处理与多文件结构 C++ 系统函数 2

从问题到程序——指针

OOP with Java 通知 Project 4: 4 月 19 日晚 9 点

<4D F736F F D205A572D2D A1AAA1AAD4ACE7F42D43D3EFD1D4CAB5D1B5BDCCB3CC2E646F6378>

c_cpp

<4D F736F F D20CBB6CABFD1D0BEBFC9FAC8EBD1A7BFBCCAD4B4F3B8D92D383835B3CCD0F2C9E8BCC62E646F63>

C/C++语言 - C/C++数据

C++ 程序设计 OJ4 - 参考答案 MASTER 2017 年 5 月 21 日 1

Microsoft Word - 把时间当作朋友(2011第3版)3.0.b.06.doc

C/C++ - 字符串与字符串函数

PowerPoint Presentation

nooog

Monthly Report 2010_12

_汪_文前新ok[3.1].doc

Microsoft PowerPoint - 0 C复习.ppt [兼容模式]

02

PowerPoint 演示文稿

没有幻灯片标题

面向对象进阶 对象的生存期 静态成员 友元关系 常对象与常成员 对象数组与对象指针 向量类 :vector 字符串类 :string 2

技能竞赛C试题

Guava学习之Resources

C++ 程序设计 实验 1 - 参考答案 MASTER 2017 年 5 月 21 日 1

北京大学

ch02_Basic

幻灯片 1

第 10 章结构体和枚举教案 一 授课题目 ( 教学章 节或主题 ) 10.1 结构体及结构体变量 10.2 结构体数组 二 教学时间安排 上课 2 学时 三 教学目的 要求 知识目标 : 学习结构体变量和结构体数组的定义和引用能力目标 : 掌握结构体的初步应用情感目标 : 通过结构体编程实践获得成

untitled

C/C++ - 函数

第5章 递归 (Recurve)

Guava学习之CharSequenceReader

Transcription:

第六讲 指针与字符串 为什么指针 持久动态内存分配 字符串 ( 字符数组 )

指针 什么是指针 指针的定义与运算 指针与一维 ( 二维 ) 数组 指针数组 指针与引用 指针与函数 2

指针定义 什么是指针 指针变量, 简称指针, 用来存放其它变量的内存地址 指针的定义 类型标识符 * 指针变量名 声明一个指针类型的变量, 星号后面可以留空格 类型标识符表示该指针所指向的对象的数据类型, 即该指针所代表的内存单元所能存放的数据的类型 Tips: 变量为什么要声明? 1) 分配内存空间 ; 2) 限定变量能参与的运算及运算规则 Tips: 内存空间的访问方式 :1) 变量名 ; 2) 内存地址, 即指针 3

指针运算 指针的两个基本运算 提取变量的内存地址 :& 提取指针所指向的变量的值 :* 地址运算符 :& & 变量名 // 提取变量在内存中的存放地址 例 : int x=3; int * px; // 定义指针 px px=&x; // 将 x 的地址赋给指针 px 此时, 我们通常称这里的 px 是指向 x 的指针 注意 : 指针的类型必须与其指向的对象的类型一致 4

指针运算 指针运算符 : * * 指针变量名 // 提取指针变量所指向的对象的值 例 : int x; int * px; // 声明指针变量, 星号后面可以有空格! px=&x; *px = 3; // 等价于 x = 3, 星号后面不能有空格! ex06_pointer_01.cpp 在使用指针时, 我们通常关心的是指针指向的元素! 初始化 : 声明指针变量时, 可以赋初值 例 : int x = 3; int * px = &x; // 指针的值只能是某个变量的地址 5

空指针 void 类型的指针 void * 指针名 void 类型的指针可以指向任何类型的对象的地址 不允许使用 void 指针操纵它所指向的对象! 通过显式类型转换, 可以访问 void 类型指针所指向的对象 例 : int x = 3; int * px; void * pv; pv = &x; // OK, void 型指针指向整型变量 px = (int *)pv; // OK, 使用 void 型指针时需要强制类型转换 6

指针赋值 指针可能的取值 一个有效的指针只有三种取值 : (1) 一个对象的地址 ; (2) 指向某个对象后面的对象 ; (3) 值为 0 或 NULL( 空指针 ) int x=3; int * px=&x; Int * py=&x+1; int * pi; pi=0; // OK pi=null; // OK 没有初始化或赋值的指针是无效的指针, 引用无效指针会带来难以预料的问题! 指针赋值 : 只能使用以下四种类型的值 (1) 0 或者值为 0 的常量, 表示空指针 ; (2) 类型匹配的对象的地址 ; (3) 同类型的另一有效指针 ; (4) 另一个对象的下一个地址 ( 相对位置 ) 7

指针与常量 指向常量的指针 const 类型标识符 * 指针名 const int a = 3; int * pa = &a; // ERROR const int * cpa = &a; // OK 指向 const 对象 ( 常量 ) 的指针必须用 const 声明! 这里的 const 限定了指针所指对象的属性, 不是指针本身的属性! const int a = 3; int b = 5; const int * cpa = &a; // OK *cpa = 5; // ERROR cpa = &b; // OK *cpa = 9; // ERROR b = 9; // OK ex06_pointer_const.cpp 指向 const 的指针所指对象的值并不一定不能修改! 允许把非 const 对象的地址赋给指向 const 的指针 ; 但不允许使用指向 const 的指针来修改它所指向的对象的值! 8

指针与常量 常量指针, 简称常指针 常量指针 : 指针本身的值不能修改 类型标识符 * const 指针名 int a = 3, b = 5; int * const pa = &a; // OK pa = &b; // ERROR 指向 const 对象的 const 指针 const 类型标识符 * const 指针名 指针本身的值不能修改, 其指向的对象的值也不能修改 9

指针算术运算 指针可以和整数或整型变量进行加减运算, 且运算规则与指针的类型相关! int * pa; int k; pa + k --> pa 所指的当前位置之后第 k 个元素的地址 pa - k --> pa 所指的当前位置之前第 k 个元素的地址 在指针上加上或减去一个整型数值 k, 等效于获得一个新指针, 该 指针指向原来的元素之后或之前的第 k 个元素 指针的算术运算通常是与数组的使用相联系的 一个指针可以加上或减去 0, 其值不变 int * pa; int k; pa++ --> pa 所指的当前位置之后的元素的地址 pa-- --> pa 所指的当前位置之前的元素的地址 10

指针算术运算 short * pa // 每个元素占两个字节 pa-2 pa-1 pa pa+1 pa+2 pa+3 *(pa-2) *(pa-1) *pa *(pa+1) *(pa+2) *(pa+3) 11

指针算术运算 int * pb // 每个元素占四个字节 pb-1 pb *(pb-1) *pb pb+1 pb+2 *(pb+1) *(pb+2) 12

指针与数组 C++ 中, 指针与数组密切相关 : 由于数组元素在内存中是连续存放的, 因此使用指针可以非常方便地处理数组元素! int a[]={0,2,4,8}; int * pa; pa = a; // OK pa = &a[0]; // OK, 与上式等价 *pa = 3;// OK, 等价于 a[0]=3 *(pa+2) = 5; // OK, 等价于 a[2]=5 *(a+2) = 5; // OK, 等价于 a[2]=5 在 C++ 中, 数组名就是数组的首地址! 当数组名出现在表达式中时, 会自动转化成指向第一个数组元素的指针! 思考 :pa = a+1, 则 *pa =? 13

一维数组与指针 一维数组与指针 在 C++ 中, 引用数组元素有以下三种方式 : (1) 数组名与下标, 如 :a[0] (2) 数组名与指针运算, 如 :*(a+1) (3) 指针, 如 :int * pa=a; *pa ex06_pointer_array01.cpp int a[]={0,2,4,8}; int * pa = a; *pa = 1; // 等价于 a[0]=1 *(pa+2) = 5; // 等价于 a[2]=5 *(a+2) = 5; // OK, 等价于 a[2]=5 *(pa++) = 3; // OK, 等价于 a[0]=3; pa = pa+1; *(a++) = 3; // ERROR! a 代表数组首地址, 是常量指针! *(pa+1) = 10; // 思考 : 修改了哪个元素的值? 指针的值可以随时改变, 即可以指向不同的元素 ; 数组名是常量指针, 值不能改变 14

举例 例 : 使用三种方法输出一个数组的所有元素 // 第一种方式 : 数组名与下标 for (int i=0; i<n; i++) cout << a[i] << "," ; // 第二种方式 : 数组名与指针运算 for (int i=0; i<n; i++) cout << *(a+i) << "," ; // 第三种方式 : 指针 for (int * pa=a; pa<a+n; pa++) cout << *pa << "," ; // 第三种方式 : 指针 for (int * pa=a, i=0; i<n; i++) cout << pa[i] << "," ; 若 pa 是指针,k 是整型数值, 则 *(pa+k) 可以写成 pa[k] *(pa-k) 可以写成 pa[-k] ex06_pointer_array02.cpp 15

一维数组与指针 一维数组 a[n] 与指针 pa=a 数组名 a 是地址常量, 数组名 a 与 &a[0] 等价 ; a+i 是 a[i] 的地址,a[i] 与 *(a+i) 等价 ; 数组元素的下标访问方式也是按地址进行的 ; 可以通过指针 pa 访问数组的任何元素, 且更加灵活 ; pa++ 或 ++pa 合法, 但 a++ 不合法 ; *(pa+i) 与 pa[i] 等价, 表示第 i+1 的元素 ; a[i] <=> pa[i] <=> *(pa+i) <=> *(a+i) 16

指针数组 指针数组 : 数组的元素都是指针变量 指针数组的声明 : 类型标识符 * 指针数组名 [n] int a[]={0,2,4,8}; int b[]={1,3,5,7}; int c[]={2,3,5,8}; int *pa[3]={a,b,c}; // 声明一个有三个元素的指针数组 // pa[0]=a, pa[1]=b, pa[2]=c 上面的 pa 代表指针数组, 不是普通的指针! 17

二维数组 C++ 中, 二维数组是按行顺序存放在内存中的, 可以理解为一维数组组成的一维数组 例 :int A[2][3]={{1,2,3},{7,8,9}}; 可以理解为 : A A[0] A 00 A 01 A 02 A[1] A 10 A 11 A 12 A[0][0] A[0][1] A[0][2] A[1][0] A[1][1] A[1][2] A[0] A[1] ( 第一行的首地址 ) ( 第二行的首地址 ) A[0], A[1] 称为行数组 18

二维数组与指针 int A[2][3]={{1,2,3},{7,8,9}}; int * pa = A[0]; // 行数组 A[0] 的首地址, 等价于 pa=&a[0][0] int * pa1 = A[1]; // 行数组 A[1] 的首地址, 即 &A[1][0] *pa = 11; // 等价于 A[0][0]=11 *(pa+2) = 12; // 等价于 A[0][2]=12 *(pa+4) = 13; // 等价于 A[1][1]=13 19

二维数组与指针 对于二维数组 A, 虽然 A A[0] 都是数组首地址, 但二者指向的对象不同 : A[0] 是一维数组的名字, 它指向的是行数组 A[0] 的首元素, 即 A[0][0] *A[0] *(A[0]+1) A[0][0] A[0][1] 而 A 是一个二维数组的名字, 它指向的是它的首元素, 即 A[0] *A A[0] *(A+1) A[1] 20

二维数组与指针 int A[2][3]={{1,2,3},{7,8,9}}; int * p = A; // ERROR! int * p = A[0]; // OK 当 int * p 声明指针时,p 指向的是一个 int 型数据, 而不是一个地址, 因此, 用 A[0] 对 p 赋值是正确的, 而用 A 对 p 赋值是错误的! 如何用普通指针引用二维数组的元素? 设指针 p=&a[0][0], 则 A[i][j] <==> *(p+n*i+j) 例 : 二维数组与普通指针 ex06_pointer_array2d.cpp 21

指针与引用 引用与指针 int a = 3; int * pa = &a; // 指针 引用是变量的别名 ; 引用必须初始化, 且不能修改 ; 引用只针对变量, 函数没有引用 ; 传递大量数据时, 最好使用指针 ; 用引用能实现的功能, 用指针都能实现 引用作为函数参数的优点 传递方式与指针类似, 但可读性强 ; 函数调用比指针更简单 安全 ; int & ra = a; // 引用 22

指针作为函数参数 指针作为函数参数 以地址方式传递数据 形参是指针时, 实参可以是指针或地址 void split(double x, int * n, double * f) double x, x2; int x1; split(x, &x1, &x2) ex06_pointer_fun01.cpp 当函数间需要传递大量数据时, 开销会很大 此时, 如果数据是连续存放的, 则可以只传递数据的首地址, 这样就可以减小开销, 提高执行效率! 23

指针作为函数参数 指针作为函数参数的三个作用 使形参和实参指向共同的内存地址 ; 减小函数间数据传递的开销 ; 传递函数代码的首地址 ( 后面介绍 ) Tips: 如果在被调函数中不需要改变指针所指向的对象的值, 则可以将形参中的指针声明为指向常量的指针 24

指针型函数 当函数的返回值是地址时, 该函数就是指针型函数 指针型函数的定义 数据类型 * 函数名 ( 形参列表 ) { 函数体 } 25

指向函数的指针 在程序运行过程中, 不仅数据要占用内存空间, 函数也要在内存中占用一定的空间 函数名就代表函数在内存空间中的首地址 用来存放这个地址的指针就是指向该函数的指针 函数指针的定义 数据类型 (* 函数指针名 )( 形参列表 ) 这里的数据类型和形参列表应与其指向的函数相同 注 : 函数名除了表示函数的首地址外, 还包括函数的返回值类型, 形参个数 类型 顺序等信息 Tips: 可以象使用函数名一样使用函数指针 26

函数指针 函数指针需要赋值后才能使用 int Gcd(int x, int y); int Lcm(int x, int y); ex06_pointer_fun02.cpp int (* pf)(int, int); // 声明函数指针 pf = Gcd; // pf 指向函数 Gcd cout << " 最大公约数 :" << pf(a,b) << endl; pf = Lcm; // pf 指向函数 Lcm cout << " 最小公倍数 :" << pf(a,b) << endl; 27

持久动态存储分配 动态内存申请 --- new 动态内存释放 --- delete 动态数组的申请与释放 在被调函数中申请动态内存, 返回给主调函数 28

动态存储分配 若在程序运行之前, 不能够确切知道数组中元素的个数, 如果声明为很大的数组, 则可能造成浪费, 如果声明为小数组, 则可能不够用 此时需要动态分配空间, 做到按需分配 动态内存分配相关函数 申请内存空间 :new 释放内存空间 :delete 每个程序在执行时都会占用一块可用的内存, 用于存放动态分配的对象, 此内存空间称为自由存储区 (free store) 29

申请内存空间 申请单个存储单元 px = new 数据类型 ; px = new 数据类型 ( 初始值 ); 申请用于存放指定数据类型数据的内存空间, 若申请成功, 则返回该内存空间的地址, 并赋值给指针 px; 若申请不成功, 则返回 0 或 NULL delete px; // 释放由 new 申请的内存空间 注 :px 必须是由 new 操作的返回值! 30

动态内存数组 创建一维动态数组 px = new 数据类型 [ 数组长度 ]; ex06_pointer_new01.cpp px = new 数据类型 [ 数组长度 ](); // 赋初值 0 这里初始化时只能将全部元素设置为 0, 而不能象数组变量那样用初始化列表赋初值 (C++11 新标准已加入该功能 ) 31

动态内存数组 创建多维动态数组 px = new 数据类型 [n 1 ][n 2 ]...[n m ]; 注 : 此时 px 不是普通的指针, 而是 : (* px)[n 2 ]...[n m ] 动态数组的释放 delete[] px; // 释放由 new 建立的数组 32

动态内存举例 例 : 给定一个正整数 N, 求出 N 个最小的素数 int main() { int N; cout << "Input N: "; cin >> N; int * pa = new int[n]; // 申请内存空间 ex06_pointer_new02.cpp int i, flag, k=0, n=2; while (k < N) // 寻找 N 个素数 { flag = 0; for(i=2; i<n; i++) if (n % i == 0) {flag = 1; break;} if (flag==0) { pa[k] = n; k++; } } n++; for(i=0; i<k && pa[i]<=sqrt(n); i++) if (n % pa[i] == 0) {flag = 1; break;} 33

动态内存举例 例 : 将前面的例子写成函数形式 int * find_prime(int N) { int * pa = new int[n]; // 申请内存空间 } int i, flag, k=0, n=2; while (k < N) // 寻找 N 个素数 { flag = 0; for(i=2; i<n; i++) if (n % i == 0) {flag = 1; break;} if (flag==0) { pa[k] = n; k++; } n++; } return pa; 注 : 在主调函数中, 如果不再需要 pa, 则需用 delete[] 释放 34

字符串 ( 字符数组 ) ( C 语言 ) 字符串的表示 字符串输入输出 字符串操作 --- 相关函数 字符操作函数 35

字符串 字符串的表示 : 一维字符数组 char str[5]={'m','a','t','h','\0'}; char str[5]="math"; // OK char str[]="math"; // OK, 只能用于初始化 字符串以 "\0" 为结束标志 使用双引号时, 会自动在最后添加结束标志 字符串赋值 : 逐个赋值, 循环实现 char str[5]; str = "Math"; // ERROR: 一维数组, 不能直接赋值! 36

字符串输出 字符串的输出 法一 : 单个元素单个元素输出 ( 循环, 数组 ) 法二 : 整体输出 例 : char str[20]="c++ and Matlab"; for(int i=0;i<20;i++) if (str[i]!='\0') cout << str[i]; else break; char str[20]="c++ and Matlab"; cout << str << endl; ex06_str_cout.cpp 注 : 输出字符中不含 "\0" 37

字符串输入 字符串的输入 输入单个字符串时, 中间不能有空格 一次输入多个字符串时, 以空格隔开 例 : char str[5]; cin >> str; char str1[5], str2[5], str3[5]; cin >> str1 >> str2 >> str3; ex06_str_cin.cpp 输入数据 :How are you? 内存中变量状态如下 : str1: H o w \0 str2: a r e \0 str3: y o u? \0 char str[13]; cin >> str; // 输入数据 :How are you? 结果会怎样 38

整行输入 整行输入 cin.getline(str,n, 结束符 ); 连续读入多个字符 ( 可以有空格 ), 直到读满 N-1 个为止, 或遇到指定的结束符 ( 不存储结束符 ) 结束符可以省略, 默认为 '\n' ( 换行 ) 例 : char str[13]; cin.getline(str,13); ex06_str_getline.cpp 39

单个字符输入 单个字符的输入 getchar(); char ch; ch=getchar(); 40

字符串操作 字符串相关函数 ( 需包含头文件 cstring 和 cstdlib ) 函数 描述 用法 strlen 求字符串长度 strlen(str) strcat 字符串连接 strcat(dest,src) strcpy 字符串复制 strcpy(dest,src) strcmp 字符串比较 strcmp(str1,str2) atoi 将字符串转换为整数 atoi(str) atol 将字符串转换为 long atol(str) atof 将字符串转换为 double atof(str) itoa 将整数转换为字符串 itoa(int,str,raidx) ( 更多函数可参见 http://www.cppreference.com) 41

字符串操作 strlen(str) 返回字符串 str1 的长度 ( 不含结束符 ) strcat(str1,str2) 将 str2 的全部内容添加到 str1 中,str2 的内容保留 strncat(str1,str2,n) 将 str2 的内容添加到 str1 中, 至多添加 n 个字符 strcmp(str1,str2) 按字典顺序比较 str1 和 str2 的大小 如果 str1>str2, 则返回一个正数 ; str1<str2, 则返回一个负数 ; 相等则返回 0 42

字符串操作 strncmp(str1,str2,n) 按字典顺序比较 str1 和 str2 的前 n 个字符的大小 strcpy(str1,str2) 将 str2 的全部内容复制到 str1 中 ; str1 的长度应该不小于 str2 的长度 strncpy(str1,str2,n) 将 str2 的前 n 个字符复制到 str1 中 ; 若 n 大于 str2 的长度, 则复制全部内容 例 : ex06_str_fun.cpp int N = 20; char str1[n]; char str2[]="hello world!"; strncpy(str1,str2,n-1); // 实际复制字符个数不超过 str2 长度 43

字符串操作 x=atoi(str) x=atol(str) x=atof(str) 分别将 str 转化为整型 长整型和双精度型数据 str 须是由数字组成的字符串 例 : int x; double y; x=atoi("66"); // x=66 y=atof("14.5"); // y=14.5 ex06_str_atoi.cpp itoa(int,str,radix) 按指定的进制将一个整数转化为字符串 例 : char str[5]; itoa(66,str,16); // 按 16 进制转换 44

字符检测 需加入头文件 cctype 函数 描述 用法 isdigit 是否为数字 isdigit('3') isalpha 是否为字母 isalpha('a') isalnum 是否为字母或数字 isalnum('c') islower 是否为小写 islower('b') isupper 是否为大写 isupper('b') isspace 是否为空格 isspace(' ') tolower 将大写转换为小写 tolower('a') toupper 将小写转换为大写 toupper('a') 更多字符检测函数参见相关资料 注 : 以上检测和转换函数只针对单个字符, 而不是字符串! 45

字符与整数 字符与整型数据之间的转换 例 : char x='2'; int y=x; int z=x-'0'; cout << "x=" << x << endl; // x='2' 是字符 cout << "y=" << y << endl; // y=50 是整数 cout << "z=" << z << endl; // z=50-48=2 是整数 字符数据与整型数据之间的转换是通过 ASCII 码实现的 字符参加算术运算时, 自动转换为整数 注 :atoi 等只能作用在字符串上! 不能作用在字符上! 46

课后练习 课后练习 ( 自己练习 ) (1) 已知一个数组, 数组名为 x, 试用一条语句算出该数组的元素个数 ( 提示 : 使用 sizeof 函数 ) (2) 阅读下面的代码 int a[]={6,21,12,34,55}; int * pa = a, * pb; *pa = 5; *(pa+2) = 8; *(pa++) = *a + 42; pb = pa; *(++pb) = 45; 指出最后 a 的值,*pa 的值,*pb 的值 47

上机作业 1) 编写函数, 交换两个双精度变量的值, 分别用引用和指针实现 函数分别为 swap_ref 和 swap_point, 并在主函数中定义两个双精度变量, 从键盘接受输入, 并将交换后的值在屏幕上输出 ( 程序取名 hw06_01.cpp) void swap_ref(double & ra, double & rb); void swap_pointer(double * pa, double * pb); 2) 找出前 100 个素数, 存放在数组 p 中, 并分别使用下列方式在屏幕上输出 p, 每行输出 10 个 ( 程序取名 hw06_02.cpp) 方式一 : 数组名 + 下标运算 ; 方式二 : 数组名 + 指针运算 ; 方式三 : 指针 + 指针运算 48

上机作业 3) 编写函数, 计算两个矩阵的乘积 Z=X*Y, 其中 X R m p,y R p n, Z R m n 要求对任意正整数 m,p,n 都能实现矩阵相乘 ( 程序取名 hw06_03.cpp) void matrix_prod(double * px, double * py, double * pz, int m, int p, int n); 提示 : 这里 px, py, pz 分别指向 X[0][0], Y[0][0] 和 Z[0][0] 4) 给定两个一维数组 a 和 b, 其中 a 中的数据是无序的, 而 b 中的数据按升序排列试统计 a 的所有元素中, 大于 b 的第 k 个元素且小于第 k+1 个元素的数据个数 其中 a=[98,12,34,71,43,54,28,33,65,56], b=[10,30,50,80,100] 要求将结果存放在数组 c 中, 其中 c[k] 表示数组 a 中大于 b[k] 而小于 b[k+1] 的元素个数 ( 程序取名 hw06_04.cpp) 49

上机作业 5) 二进制转十进制 编写函数, 将一个用字符串表示的二进制数转化为十进制数, 如 10001 所对应的十进制数为 17, 在主函数中用 1100110011001100 测试 ( 程序取名 hw06_05.cpp) int bin2dec(const char * const str); 提示 : 将一个字符转化成数字, 可借助字符加减运算 ( 推荐 ) 或字符串函数 6) 字符易位破译 编写函数, 测试两个字符串是否字符异位相等, 即两个字符串中 包含的字母是相同的, 但次序可以不同, 如 silent 和 listen 是字符异位相等, 但 baac 与 abcc 不是 ( 程序取名 hw06_08.cpp) bool isanagram(const char * const str1, const char * const str2); 提示 : 可以先对字符串进行排序, 然后再比较 50

上机作业 7) 有 17 人围成一圈, 编号 1~17, 从 1 号开始报数, 报到 3 的倍数的人离开, 一直数下去, 直到最后只剩一人, 求此人编号 程序取名 hw06_07.cpp 51