学习《c++_primer》之第五章

5.1 节练习

练习 5.1

空语句是最简单的语句,空语句由一个单独的分号构成。如果在程序的某个地方,语法上需要一条语句但是逻辑上不需要,此时应该使用空语句,空语句什么也不做。

一种常见的情况是,当循环的全部工作在条件部分就可以完成时,我们通常会用到空语句。使用空语句时最好加上注释,从而令代码的阅读者知道这条语句是有意省略内容的。

练习 5.2

块是指用花括号括起来的语句和声明的序列,也称为复合语句。一个块就是一个作用域,在块中引入的名字只能在块内部以及嵌套在块中的子块里访问。如果在程序的某个地方,语法上需要一条语句,但是逻辑上需要多条语句,此时应该使用块。块不需要以分号结束。

例如,循环体必须是一条语句,但是我们通常需要在循环体内做很多事情,此时就应该把多条语句用花括号括起来,从而把语句序列转变成块。

练习 5.3

原文的 while 循环使用了块,其形式是:

while ( val <= 10 ) 
{ 
 sum += val; 
 ++val; 
} 

利用逗号运算符改写之后的形式如下所示:

while ( val <= 10 ) 
 sum += val, ++val; 

很明显,改写之后的代码不够清晰,可读性降低了。

5.2 节练习

练习 5.4

(a)是非法的,它的原意是希望在 while 语句的控制结构当中定义一个string::iterator 类型的变量 iter,然后判断 iter 是否到达了 s 的末尾,只要还没有到达末尾就执行循环体的内容。但是该式把变量的定义和关系判断混合在了一起,如果要使用 iter 与其他值比较,必须首先为 iter 赋初值。修改后的程序应该是:

string::iterator iter=s.begin(); 
while (iter != s.end()) 
{ 
 ++iter; 
 /*...*/ 
}

(b)是非法的,变量 status 定义在 while 循环控制结构的内部,其作用域仅限于 while 循环。if 语句已经位于 while 循环的作用域之外,status 在 if 语句内是一个未命名的无效变量。要想在 if 语句中继续使用 status,需要把它定义在while 循环之前。修改后的程序应该是:

bool status; 
while (status = find(word)) { /* ... */ } 
if (!status) { /* ... */ }

附上书上原话:

img

5.3 节练习

练习 5.5

#include <iostream>

using namespace std;

int main()
{
	int grade;
	cout << "请输入您的成绩:";
	cin >> grade;
	if (grade < 0 || grade > 100)
	{
		cout << "该成绩不合法" << endl;
		return -1;
	}
	if (grade == 100) // 处理满分的情况
	{
		cout << "等级成绩是:" << "A++" << endl;
		return -1;
	}
	if (grade < 60) // 处理不及格的情况
	{
		cout << "等级成绩是:" << "F" << endl;
		return -1;
	}
	int iU = grade / 10; // 成绩的十位数
	int iT = grade % 10; // 成绩的个位数
	string score, level, lettergrade;
	// 根据成绩的十位数字确定 score 
	if (iU == 9)
		score = "A";
	else if (iU == 8)
		score = "B";
	else if (iU == 7)
		score = "C";
	else
		score = "D";
	// 根据成绩的个位数字确定 level 
	if (iT < 3)
		level = "-";
	else if (iT > 7)
		level = "+";
	else
		level = "";
	// 累加求得等级成绩
	lettergrade = score + level;
	cout << "等级成绩是:" << lettergrade << endl;
	return 0;
}

输出如下:

img

练习 5.6

#include <iostream> 
using namespace std;
int main()
{
	int grade;
	cout << "请输入您的成绩:";
	cin >> grade;
	if (grade < 0 || grade > 100)
	{
		cout << "该成绩不合法" << endl;
		return -1;
	}
	if (grade == 100) // 处理满分的情况
	{
		cout << "等级成绩是:" << "A++" << endl;
		return -1;
	}
	if (grade < 60) // 处理不及格的情况
	{
		cout << "等级成绩是:" << "F" << endl;
		return -1;
	}
	int iU = grade / 10; // 成绩的十位数
	int iT = grade % 10; // 成绩的个位数
	string score, level, lettergrade;
	// 根据成绩的十位数字确定 score 
	score = (iU == 9) ? "A"
		: (iU == 8) ? "B"
		: (iU == 7) ? "C" : "D";
	// 根据成绩的个位数字确定 level 
	level = (iT < 3) ? "-"
		: (iT > 7) ? "+" : "";
	// 累加求得等级成绩
	lettergrade = score + level;
	cout << "等级成绩是:" << lettergrade << endl;
	return 0;
}

输出如下:

img

练习 5.7

(a) if 语句的循环体应该是一条语句,需要以分号结束,

程序修改为:

if (ival1 != ival2) 
    ival1 = ival2;
else ival1 = ival2 = 0;

(b) if 语句的循环体只能是一条语句,本题从逻辑上来说需要做两件事,一是修改 minval 的值,二是重置 occurs 的值,所以必须把这两条语句放在一个块里。

程序修改为:

if (ival < minval) 
{ 
    minval = ival; 
    occurs = 1; 
} 

(c) ival 是定义在 if 语句中的变量,其作用域仅限于第一个 if 语句,要想在第二个 if 语句中也使用它,就必须把它定义在两个 if 语句的外部。

程序修改为:

int ival; 
if (ival = get_value()) 
    cout << "ival = " << ival << endl; 
if (!ival) 
    cout << "ival = 0\n"; 

(d) 程序的原意是判断 ival 的值是否是 0,原题使用赋值运算符的结果是把 0赋给了 ival,然后检验 ival 的值,这样使得条件永远不会满足。

程序修改为:

if (ival == 0) 
    ival = get_value();

练习 5.8

悬垂 else 是指当程序中的 if 分支多于 else 分支时,如何为 else 寻找与之匹配的 if 分支的问题。

C++规定,else 与离它最近的尚未匹配的 if 匹配,从而消除了二义性。

练习 5.9

#include <iostream> 
using namespace std;
int main()
{
	unsigned int vowelCnt = 0;
	char ch;
	cout << "请输入一段文本:" << endl;
	while (cin >> ch)
	{
		if (ch == 'a')
			++vowelCnt;
		if (ch == 'e')
			++vowelCnt;
		if (ch == 'i')
			++vowelCnt;
		if (ch == 'o')
			++vowelCnt;
		if (ch == 'u')
			++vowelCnt;
	}
	cout << "您输入的文本中共有 " << vowelCnt << "个元音字母" << endl;
	return 0;
}

输出如下:

img

练习 5.10

#include <iostream> 
using namespace std;
int main()
{
	unsigned int aCnt = 0, eCnt = 0, iCnt = 0, oCnt = 0, uCnt = 0;
	char ch;
	cout << "请输入一段文本:" << endl;
	while (cin >> ch)
	{
		switch (ch)
		{
		case 'a':
		case 'A':
			++aCnt;
			break;
		case 'e':
		case 'E':
			++eCnt;
			break;
		case 'i':
		case 'I':
			++iCnt;
			break;
		case 'o':
		case 'O':
			++oCnt;
			break;
		case 'u':
		case 'U':
			++uCnt;
			break;
		}
	}
	cout << "元音字母 a 的数量是:" << aCnt << endl;
	cout << "元音字母 e 的数量是:" << eCnt << endl;
	cout << "元音字母 i 的数量是:" << iCnt << endl;
	cout << "元音字母 o 的数量是:" << oCnt << endl;
	cout << "元音字母 u 的数量是:" << uCnt << endl;
	return 0;
}

输出如下:

img

练习 5.11

#include <iostream> 
using namespace std;
int main()
{
	unsigned int aCnt = 0, eCnt = 0, iCnt = 0, oCnt = 0, uCnt = 0;
	unsigned int spaceCnt = 0, tabCnt = 0, newlineCnt = 0;
	char ch;
	cout << "请输入一段文本:" << endl;
	while (cin.get(ch))
	{
		switch (ch)
		{
		case 'a':
		case 'A':
			++aCnt;
			break;
		case 'e':
		case 'E':
			++eCnt;
			break;
		case 'i':
		case 'I':
			++iCnt;
			break;
		case 'o':
		case 'O':
			++oCnt;
			break;
		case 'u':
		case 'U':
			++uCnt;
			break;
		case ' ':
			++spaceCnt;
			break;
		case '\t':
			++tabCnt;
			break;
		case '\n':
			++newlineCnt;
			break;
		}
	}
	cout << "元音字母 a 的数量是:" << aCnt << endl;
	cout << "元音字母 e 的数量是:" << eCnt << endl;
	cout << "元音字母 i 的数量是:" << iCnt << endl;
	cout << "元音字母 o 的数量是:" << oCnt << endl;
	cout << "元音字母 u 的数量是:" << uCnt << endl;
	cout << "空格的数量是:" << spaceCnt << endl;
	cout << "制表符的数量是:" << tabCnt << endl;
	cout << "换行符的数量是:" << newlineCnt << endl;
	return 0;
}

输出如下:

img

注意

读入字符的语句应该使用 cin.get(ch),而不能使用 >>,因为后者会忽略本题所要统计的特殊符号。

如果写成 while (cin >> ch),则输出结果是这样:

img

练习 5.12

我们的设定是一个字符只会被统计一次。如果用户输入的序列是 xxxxxfflxxx,则统计结果是 ff:1 次、fl:0 次、fi:0 次。如果用户输入的序列是xxxxxfiffffflxxx,则统计结果是 ff:2 次、fl:1 次、fi:1 次。

满足题意的程序如下所示:

#include <iostream> 
using namespace std;
int main()
{
	unsigned int ffCnt = 0, flCnt = 0, fiCnt = 0;
	char ch, prech = '\0';
	cout << "请输入一段文本:" << endl;
	while (cin >> ch)
	{
		bool bl = true;
		if (prech == 'f')
		{
			switch (ch)
			{
			case 'f':
				++ffCnt;
				bl = false;
				break;
			case 'l':
				++flCnt;
				break;
			case 'i':
				++fiCnt;
				break;
			}
		}
		if (!bl)
			prech = '\0';
		else
			prech = ch;
	}
	cout << "ff 的数量是:" << ffCnt << endl;
	cout << "fl 的数量是:" << flCnt << endl;
	cout << "fi 的数量是:" << fiCnt << endl;
	return 0;
}

输出如下:

img

练习 5.13

switch 语句有几个语法要点:必须在必要的地方使用 break;语句,应该把变量定义在块作用域内,case 标签只能有一个值且不能是变量。

(a)的错误是在每个 case 分支中都缺少了 break;语句,造成的后果是一旦执行了前面的 case 分支,必定还会继续执行接下来的其他 case 分支。举例说明,如果ch 的内容是字符'a',则 aCnt、eCnt 和 iouCnt 的值都会增加;如果 ch 的内容是字符'e',则 eCnt 和 iouCnt 的值都会增加,这显然与程序的预期是不相符的。

修改后的程序如下所示:

unsigned aCnt = 0, eCnt = 0, iouCnt = 0;
char ch = next_text();
switch (ch) {
case 'a':
    aCnt++;
    break;
case 'e':
    eCnt++;
    break;
default:
    iouCnt++;
    break;
}

(b)的错误是在 case 分支中定义并初始化了变量 ix,同时在 default 分支中使用了该变量,此时如果控制流跳过 case 分支而直接到达 default 分支,则会试图使用未经初始化的变量,因而该程序无法通过编译。解决办法是,把 ix 的定义放置在 switch 语句之前。

修改后的程序如下所示:

unsigned index = some_value();
int ix;
switch (index) {
case 1:
	ix = get_value();
	ivec[ix] = index;
	break;
default:
	ix = ivec.size() - 1;
	ivec[ix] = index;
}

附上书上原话:

img

插句题外话

在Visual Studio 2019 中:

如果写成这样,则编译会报错

#include <iostream>
#include <vector>

using namespace std;

int main()
{
	unsigned index = 2;
	vector<int> ivec(5);
	int ix;
	switch (index) {
	case 1:
		//int ix = 10;
		ivec[ix] = index;
		break;
	default:
		ix = ivec.size() - 1;
		ivec[ix] = index;
	}

	for (auto i : ivec)
		cout << i << " ";
	cout << endl;

	return 0;
}

img

但是写成这样(即在定义ix的同时对其初始化)就可以编译通过了:

#include <iostream>
#include <vector>

using namespace std;

int main()
{
	unsigned index = 2;
	vector<int> ivec(5);
	int ix = 0;
	switch (index) {
	case 1:
		//int ix = 10;
		ivec[ix] = index;
		break;
	default:
		ix = ivec.size() - 1;
		ivec[ix] = index;
	}

	for (auto i : ivec)
		cout << i << " ";
	cout << endl;

	return 0;
}

输出如下:

img

在Dev-C++ 5.11中:

写成这样(即在定义ix的时候不对其进行初始化),是不报错的:

#include <iostream>
#include <vector>

using namespace std;

int main()
{
	unsigned index = 2;
	vector<int> ivec(5);
	int ix;
	switch (index) {
	case 1:
		//int ix = 10;
		ivec[ix] = index;
		break;
	default:
		ix = ivec.size() - 1;
		ivec[ix] = index;
	}

	for (auto i : ivec)
		cout << i << " ";
	cout << endl;

	return 0;
}

输出如下:

img

在Visual Studio 2019中,写成这样,会报如下错误:

#include <iostream>
#include <vector>

using namespace std;

int main()
{
	unsigned index = 2;
	vector<int> ivec(5);
	switch (index) {
	case 1:
		int ix = 10;
		ivec[ix] = index;
		break;
	default:
		ix = ivec.size() - 1;
		ivec[ix] = index;
	}

	for (auto i : ivec)
		cout << i << " ";
	cout << endl;

	return 0;
}

img

回归正题

(c)的错误是在同一个 case 标签中放置了多个值,而 C++规定一个 case 标签只能对应一个值。

修改后的程序如下所示:

unsigned evenCnt = 0, oddCnt = 0;
int digit = get_num() % 10;
switch (digit) {
case 1:
case 3:
case 5:
case 7:
case 9:
	oddcnt++;
	break;
case 2:
case 4:
case 6:
case 8:
case 10:
	evencnt++;
	break;
} 

(d)的错误是使用变量作为 case 标签的内容,C++规定,case 标签的内容只能是整型常量表达式。

修改后的程序如下所示:

const unsigned ival = 512, jval = 1024, kval = 4096;
unsigned bufsize;
unsigned swt = get_bufCnt();
switch (swt) {
case ival:
	bufsize = ival * sizeof(int);
	break;
case jval:
	bufsize = jval * sizeof(int);
	break;
case kval:
	bufsize = kval * sizeof(int);
	break;
}

5.4 节练习

练习 5.14

#include <iostream> 
#include <string> 
using namespace std;
int main()
{
	// 定义 3 个字符串变量,分别表示:
   // 当前操作的字符串、前一个字符串、当前出现次数最多的字符串
	string currString, preString = "", maxString;
	// 定义 2 个整型变量,分别表示:
   // 当前连续出现的字符串数量、当前出现次数最多的字符串数量
	int currCnt = 1, maxCnt = 0;
	while (cin >> currString) // 检查每个字符串
	{
		// 如果当前字符串与前一个字符串一致,更新状态
		if (currString == preString)
		{
			++currCnt;
			if (currCnt > maxCnt)
			{
				maxCnt = currCnt;
				maxString = currString;
			}
		}
		// 如果当前字符串与前一个字符串不一致,重置 currCnt 
		else
		{
			currCnt = 1;
		}
		// 更新 preString 以便于下一次循环使用
		preString = currString;
	}
	if (maxCnt > 1)
		cout << "出现最多的字符串是:" << maxString
		<< ",次数是:" << maxCnt << endl;
	else
		cout << "每个字符串都只出现了一次" << endl;
	return 0;
}

输出如下:

img

插句题外话

自己写的

#include <iostream>

using namespace std;

int main()
{
	int num = 1, res = 1;
	string word, preword, maxword;
	while (cin >> word)
	{
		if (preword == word)
		{
			num++;
		}
		else
		{
			if (num > res)
			{
				res = num;
				maxword = preword;
			}
			preword = word;
			num = 1;
		}
	}
	res == 1 ? cout << "no answer" << endl : cout << maxword << " " << res << endl;
	return 0;
}

输出如下:

img

练习 5.15

(a)的错误是在 for 语句中定义了变量 ix,然后试图在 for 语句之外继续使用ix。因为 ix 定义在 for 语句的内部,所以其作用域仅限于 for 语句。在 if 语句中 ix 已经失效,因此程序无法编译通过。

修改后的程序如下:

int ix; 
for (ix = 0; ix != sz; ++ix) { /* ... */ } 
if (ix != sz) 
// ...

(b)的错误有两个,一是变量 ix 未经初始化就直接使用,二是 for 语句的控制结构缺少一句话,在语法上是错误的。

修改后的程序如下:

int ix; 
for (ix = 0; ix != sz; ++ix) { /* ... */ } 

(c)的错误是一旦进入循环,程序就会无休止地执行下去。也就是说,当初始情况下 ix != sz 时,由题意可知 ix 和 sz 一直同步增长,循环的终止条件永远不会满足,所以该循环是一个死循环。

修改后的程序如下:

for (int ix = 0; ix != sz; ++ix) { /* ... */ }

练习 5.16

不赘叙了

练习 5.17

#include <iostream> 
#include <vector> 
using namespace std;
int main()
{
	vector<int> v1 = { 0, 1, 1, 2 };
	vector<int> v2 = { 0, 1, 1, 2, 3, 5, 8 };
	vector<int> v3 = { 3, 5, 8 };
	vector<int> v4 = { 3, 5, 0, 9, 2, 7 };
	auto it1 = v1.cbegin(); // 定义 v1 的迭代器
	auto it2 = v2.cbegin(); // 定义 v2 的迭代器
	auto it3 = v3.cbegin(); // 定义 v3 的迭代器
	auto it4 = v4.cbegin(); // 定义 v4 的迭代器
	// 设定循环条件:v1 和 v2 都尚未检查完
	while (it1 != v1.cend() && it2 != v2.cend())
	{
		// 如果当前位置的两个元素不相等,则肯定没有前缀关系,退出循环
		if (*it1 != *it2)
		{
			cout << "v1 和 v2 之间不存在前缀关系" << endl;
			break;
		}
		++it1; // 迭代器移动到下一个元素
		++it2; // 迭代器移动到下一个元素
	}
	if (it1 == v1.cend()) // 如果 v1 较短
	{
		cout << "v1 是 v2 的前缀" << endl;
	}
	if (it2 == v2.cend()) // 如果 v2 较短
	{
		cout << "v2 是 v1 的前缀" << endl;
	}
	return 0;
}

输出如下:

img

程序的输出结果是:v1 是 v2 的前缀。如果更新程序使其处理 v3 和 v4,则程序将显示:v3 和 v4 之间不存在前缀关系。

插句题外话

自己写的

#include <iostream>
#include <vector>

using namespace std;

bool isequal(vector<int> vshort, vector<int> vlong)
{
	for (int index = 0; index != vshort.size(); index++)
		if (vshort[index] != vlong[index])
			return false;
	return true;
}


int main()
{
	vector<int> v1;
	vector<int> v2;
	int n, size1, size2;

    cout << "输入第一组数据的长度:";
    cin >> size1;
    cout << "输入第一组数据:";
    for (int i = 0; i < size1; ++i)
    {
        cin >> n;
        v1.push_back(n);
    }

    cout << "输入第二组数据的长度:";
    cin >> size2;
    cout << "输入第二组数据:";
    for (int i = 0; i < size2; ++i)
    {
        cin >> n;
        v2.push_back(n);
    }

    if (size1 <= size2)
        isequal(v1, v2) ? cout << "v1是v2的前缀" << endl
        : cout << "v1 和 v2 之间不存在前缀关系" << endl;
	else
        isequal(v1, v2) ? cout << "v2是v1的前缀" << endl
        : cout << "v1 和 v2 之间不存在前缀关系" << endl;
	return 0;
}

输出如下:

img

练习 5.18

(a)的含义是每次循环读入两个整数并输出它们的和。因为 do-while 语句的循环体必须是一条语句或者一个语句块,所以在本题中应该把循环体的内容用花括号括起来。

修改后的程序是:

do {
    int v1, v2;
    cout << "Please enter two numbers to sum:";
    if (cin >> v1 >> v2)
        cout << "Sum is: " << v1 + v2 << endl;
} while (cin);

(b)的含义是当 get_response 的返回值不为 0 时执行循环体。因为 do-while语句不允许在循环条件内定义变量,所以该程序是错误的。

修改后的程序是:

int ival;
do {
    ival = get_response();
} while (ival);

(c) 的含义是当 get_response 的返回值不为 0 时执行循环体。因为出现在do-while 语句条件部分的变量必须定义在循环体之外,所以该程序是错误的。

修改后的程序是:

int ival;
do {
    ival = get_response();
} while (ival);

练习 5.19

#include <iostream>
#include <string>
using namespace std;

int main()
{
    do {
        cout << "请输入两个字符串" << endl;
        string str1, str2;

        // 检查输入是否有效
        if (!(cin >> str1 >> str2)) {
            // 输入失败时退出循环
            cout << "输入结束或发生错误。" << endl;
            break;
        }

        // 比较字符串长度并输出结果
        if (str1.size() < str2.size())
            cout << "长度较小的字符串是:" << str1 << endl;
        else if (str1.size() > str2.size())
            cout << "长度较小的字符串是:" << str2 << endl;
        else
            cout << "两个字符串等长" << endl;

    } while (true); // 循环控制交由输入校验处理
    return 0;
}

输出如下:

img

在自己写这题的时候,思考了一个问题,可以看看这篇博客 使用while循环分别对两个vector进行赋值,该怎么做

5.5 节练习

练习 5.20

#include <iostream> 
#include <string> 
using namespace std;
int main()
{
	string currString, preString;
	bool bl = true;
	cout << "请输入一组字符串:" << endl;
	while (cin >> currString)
	{
		if (currString == preString)
		{
			bl = false;
			cout << "连续出现的字符串是:" << currString << endl;
			break;
		}
		preString = currString;
	}
	if (bl)
		cout << "没有连续出现的字符串" << endl;
	return 0;
}

输出如下:

img

练习 5.21

#include <iostream> 
#include <string> 
using namespace std;
int main()
{
	string currString, preString;
	bool bl = true;
	cout << "请输入一组字符串:" << endl;
	while (cin >> currString)
	{
		if (!isupper(currString[0]))
		{
			preString = "";
			continue;
		}
		if (currString == preString)
		{
			bl = false;
			cout << "连续出现的以大写字母开头的字符串是:" << currString << endl;
			break;
		}
		preString = currString;
	}
	if (bl)
		cout << "没有连续出现的以大写字母开头的字符串" << endl;
	return 0;
}

练习 5.22

int sz;
do {
	sz = get_size();
} while (sz <= 0);

5.6 节练习

练习 5.23

#include <iostream> 
using namespace std;
int main()
{
	cout << "请依次输入被除数和除数:" << endl;
	int ival1, ival2;
	cin >> ival1 >> ival2;
	if (ival2 == 0)
	{
		cout << "除数不能为 0" << endl;
		return -1;
	}
	cout << "两数相除的结果是:" << ival1 / ival2 << endl;
	return 0;
}

输出如下:

img

练习 5.24

#include <iostream> 
#include <stdexcept> 
using namespace std;
int main()
{
	cout << "请依次输入被除数和除数:" << endl;
	int ival1, ival2;
	cin >> ival1 >> ival2;
	if (ival2 == 0)
	{
		throw runtime_error("除数不能为 0");
	}
	cout << "两数相除的结果是:" << ival1 / ival2 << endl;
	return 0;
}

在Visual Studio 2019中运行,弹出了这个:

img

在Dev C++中运行,输出如下:

img

练习 5.25

#include <iostream> 
#include <stdexcept> 
using namespace std;
int main()
{
	cout << "请依次输入被除数和除数:" << endl;
	int ival1, ival2;
	while (cin >> ival1 >> ival2)
	{
		try
		{
			if (ival2 == 0)
			{
				throw runtime_error("除数不能为 0");
			}
			cout << "两数相除的结果是:" << ival1 / ival2 << endl;
		}
		catch (runtime_error err)
		{
			cout << err.what() << endl;
			cout << "需要继续吗(y or n)?";
			char ch;
			cin >> ch;
			if (ch != 'y' && ch != 'Y')
				break;
		}
	}
	return 0;
}

输出如下:

img

posted @ 2024-11-17 00:21  hisun9  阅读(30)  评论(0)    收藏  举报