关于各种循环语句,大家应该很熟悉了。凡是学习过C语言的或者有过编程经验的朋友都能熟练的应用循环语句了,所以,本章中我们就不仔细的介绍各种语法。快速浏览过去,就其中一些不常见的或者以前疏漏的知识点进行整理总结。
在C++11中,新增加了一种循环:基于范围的for循环。这简化了一种常见的循环任务:对数组(或容器类,如vector和array)的每个元素执行相同的操作,如下例所示:
double prices[5] = {4.99, 10.99, 6.87, 7.99, 8.49};
for (double x : prices)
cout << x << std::endl;
其中,x最初表示数组prices的第一个元素。显示第一个元素后,不断执行循环,而x依次表示数组的其他元素。因此,上述代码显示全部5个元素,每个元素占据一行。总之,该循环显示数组中每个值。
要修改数组的元素,需要使用不同的循环变量语法:
for (double &x : prices)
x = x * 0.80;
符号&表明x是一个引用变量。就这里而言,这种声明让接下来的代码能够修改数组的内容,而第一种语法不能。
还可结合使用基于范围的for循环和初始化列表:
for (int x : {3, 5, 2, 8, 6})
cout << x << " ";
cout << '\n';
了解了循环的工作原理之后,我们看一个循环完成的一项最常见、最重要的任务:逐字符地读取来自文件或键盘的文本。例如,读者可能想编写一个能够计算输入中的字符数、行数和字数的程序。尽管C++中的while循环与C语言中的while循环一样,但是由于C++的I/O工具不同,所以我们需要了解三种不同的while循环中的输入模式。
事实上,cin对象支持3种不同模式的单字符输入,其用户接口各不相同。
如果程序要使用循环来读取来自键盘的文本输入,则必须有办法知道何时停止读取。如何知道这一点呢?一种方法是选择某个特殊字符---有时被称为哨兵字符,将其作为停止标记。例如,下面程序中遇到#字符时停止读取输入。
#include <iostream>
提醒:由于ch是char类型,所以只能存储一个字符。我们在键盘通过cin输入一串字符串时,实际上字符串存储在输入流缓冲文件中,只有一个字符读入ch变量。所以才会有接下来的连续循环读入。int main()
{
using namespace std;
char ch;
int count = 0;
cout<<"Enter characters; enter # to-quit:\n";
cin>>ch;
while(ch!='#')
{
cout<<ch;
count++;
cin>>ch;
}
cout<<endl<<count<<" characters read\n";
return 0;
}
该程序计算读取的字符数,并回显这些字符,即在屏幕上显示读取的字符。按下键盘上的键不能自动将字符显示在屏幕上,程序必须通过回显输入字符来完成这项工作。
下面是该程序的运行情况:
Enter characters ; enter # to quit:
see ken run#really fast
seekenrun
9 characters read
上面为什么程序在输出时省略了空格呢?原因在cin。读取char值时,与读取其他基本类型一样,cin将忽略空格和换行符。因此输入中的空格没有被回显,也没有被包括在计数内。
通常,逐个字符读取输入的程序需要检查每个字符,包括空格、制表符和换行符。cin所属的istream类(在iostream中定义)中包含一个能够满足这种要求的成员函数。具体地说,成员函数cin.get(ch)读取输入中的下一个字符(即使它是空格),并将其赋值给变量ch。使用这个函数替代cin>>ch,可以修补上面程序丢失空格的问题。
#include <iostream>
其实使用诸如#等符号来表示输入结果很难令人满意,因为这样的符号可能就是合法输入的组成部分,其他符号(如@和%)也如此。如果输入来自于文件,则可以使用一种功能更强大的技术---检测文件尾(EOF)。C++输入工具和操作系统协同工作,来检测文件尾并将这种信息告知程序。int main()
{
using namespace std;
char ch;
int count = 0;
cout<<"Enter characters; enter # to-quit:\n";
cin.get(ch);
while(ch!='#')
{
cout<<ch;
count++;
cin.get(ch);
}
cout<<endl<<count<<" characters read\n";
return 0;
}
下面是该程序的运行情况:
Enter characters ; enter # to quit:
Did you use a #2 pencil?
Did you use a
14 characters read
现在,该程序回显了每个字符,并将全部字符计算在内,其中包括空格。输入仍被缓冲,因此输入的字符个数仍可能比最终到达程序的要多。
(3)文件尾条件
咋一看,读取文件中的信息似乎同cin和键盘输入没什么关系,但其实存在两个相关的地方。
首先,很多操作系统都支持重定向,允许用文件替换键盘输入。例如,假设在Windows中有一个名为gofish.exe的可执行程序和一个名为fishtale的文本文件,则可以在命令提示符模式下输入命令:gofish < fishtale 这样,程序将从fishtale文件而不是键盘获取输入。<符号是UNIX和Windows命令提示符模式的重定向运算符。
其次,很多操作系统都允许通过键盘来模拟文件尾条件。在Unix中,可以在行首按下Ctrl+D来实现;在Windows命令提示符模式下,可以在任意位置按Ctrl+Z和Enter。总之,很多PC编程环境都将Ctrl+Z视为模拟的EOF。
如果编程环境能够检测EOF,那么就可以在键盘输入中模拟EOF代替我们使用的#等字符。这一点很有用。
我们看一下具体是如何实现的:检测到EOF后,cin将两位(eofbit和failbit)都设置为1.可以通过成员函数eof()来查看eofbit是否被设置;如果检测到EOF,则cin.eof()将返回bool值true,否则返回false。同样,如果eofbit或failbit被设置为1,则fail()成员函数返回true,否则返回false。注意,eof()和fail()方法报告最近读取的结果:也就是说,它们在事后报告,而不是预先报告。因此应将cin.eof()或cin.fail()测试放在读取后。
还是前面的例子,这里我们使用EOF结尾符代替#符号:
#include <iostream>
需要注意的几点:int main()
{
using namespace std;
char ch;
int count = 0;
cout<<"Enter characters; enter # to-quit:\n";
cin.get(ch);
while(cin.fail() == false)
{
cout<<ch;
count++;
cin.get(ch);
}
cout<<endl<<count<<" characters read\n";
return 0;
}
下面是该程序的运行情况:
The green bird sings in the winter.<Enter>
The green bird sings in the winter.
Yes, but the crow files in the dawn.<Enter>
Yes, but the crow files in the dawn.
<CTRL>+<Z><ENTER>
73 characters read
另外,cin.get(char)的返回值是一个cin对象。然而,istream类提供了一个可以将istream对象(如cin)转换为bool值的函数;当cin出现在需要bool值的地方时,该转换函数将被调用。另外,如果最后一次读取成功了,则转换得到的bool值为true;否则为false。这意味着可以将上述while测试改写成这样:while(cin).这比!cin.fail()或!cin.eof()更通用,因为它可以检测到其他失败原因,如磁盘故障。第二,常见的字符输入做法
每次读取一个字符,直到遇到EOF的输入循环的基本设计如下:
cin.get(ch);
while(cin.fail() == false)
{
. . .
cin.get(ch);
}
可以在上述代码中使用一些简洁方法。
比如将上述while测试改写成这样: while (!cin.fail() )
这样,cin.get(char)只被调用一次,而不是两次:循环前一次,循环后一次。最后,由于cin.get(char)的返回值为cin,因此可以将循环精简成这种格式:
while(cin.get(ch))
{
. . .
}
如果ch是一个字符,则循环将显示它。如果ch为EOF,则循环将结束。
另外,除了当前所做的修改之外,关于cin.get()还有一个微妙而重要的问题。由于EOF表示的不是有效字符编码,因此可能不与char类型兼容。例如,在有些系统中,char类型是没有符号的,因此char变量不可能为EOF值(-1)。由于这个原因,如果使用cin.get()并测试EOF,则必须将返回值赋给int变量,而不是char变量。另外,如果将ch的类型声明为int,而不是char,则必须在显示ch时将其强制转换为char类型。
如下程序所示:
#include <iostream>
int main()
{
using namespace std;
int ch;
int count = 0;
while((ch = cin.get()) != EOF)
{
cout.put(char(ch));
++count;
}
cout << endl << count <<" characters read\n";
return 0;
}
那么应使用cin.get()还是cin.get(char)呢?使用字符参数的版本更符合对象方式,因为其返回值是istream对象。这意味着可以将它们拼接起来。例如下面的代码将输入中的下一个字符读入ch1中,并将接下来的一个字符读入到ch2中:
cin.get(ch1).get(ch2);
这是可行的,因为函数调用cin.get(ch1)返回一个cin对象,然后便可以通过该对象调用get(ch2)。
get()的主要用途是能够将stdio.h的getchar()和putchar()函数转换为iostream的cin.get()和cout.put()方法。只要用头文件iostream替换stdio.h,并使用相似的方法替换所有的getchar()和putchar()即可。
下面来讨论一下如何使用嵌套for循环处理二维数组。
其实,二维数组就是每个元素本身都是数组的数组。例如:假设要存储5个城市在4年间的最高温度。可以声明:int maxtemps[4][5];
该声明意味着maxtemps是一个包含4个元素的数组,其中每个元素都是一个由5个整数组成的数组。可以将maxtemps数组看作由4行组成,其中每一行有5个温度值。
假如要打印数组所有内容,可以用一个for循环来改变行,用另一个被嵌套的for循环来改变列:
for (int row = 0; row < 4; row++)
{
for (int col = 0; col < 5; ++col)
cout << maxtemps[row][col] << "\t";
cout<<endl;
}
(1) 初始化二维数组
格式这样:
int maxtemps[4][5] =
{
{96, 100, 87, 101, 105},
{96, 98, 91, 107, 104},
{97, 101, 93, 108, 107},
{98, 103, 95, 109, 108}
};
(2) 使用二维数组
还是通过例子说明:
#include <iostream>
const int Cities = 5;
const int Years = 4;
int main()
{
using namespace std;
const char* cities[Cities] =
{
"Gribble City",
"Gribbletown",
"New Gribble",
"San Gribble",
"Gribble Vista"
};
int maxtemps[Years][Cities] =
{
{96, 100, 87, 101, 105},
{96, 98, 91, 107, 104},
{97, 101, 93, 108, 107},
{98, 103, 95, 109, 108}
};
cout << "Maxinum temperatures for 2008 - 2011\n\n";
for (int city = 0; city < Cities; ++ city)
{
cout << cities[city] << " :\t";
for (int year = 0; year < Years; ++year)
cout << maxtemps[year][city] << "\t";
cout << endl;
}
return 0;
}
下面是该程序的输出:
Maxinum temperatures for 2008 - 2011
Gribble City: 96 96 97 98
Gribbletown: 10098101 103
New Gribble: 87 91 93 95
San Gribble: 101 107 108 109
Gribble Vista: 105 104 107 108
在输出中使用制表符比使用空格可使数据排列更有规则。然而,制表符设置不相同,因此输出的外观将随系统而异。
另外,在这个例子中,可以使用char数组的数组,而不是字符串指针数组。在这种情况下,声明如下:
char cities[Cities][25] =
{
"Gribble City",
"Gribbletown",
"New Gribble",
"San Gribble",
"Gribble Vista"
};
上述方法将全部5个字符串的最大长度限制为24个字符。指针数组存储5个字符串的地址,而使用char数组的数组时,将5个字符串分别复制到5个包含25个元素的char数组中。因此,从存储空间的角度说,使用指针数组更为经济;然而,如果要更改其中的任何一个字符串,则二维数组是更好的选择。令人惊讶的是,这两种方法使用相同的初始化列表,显示字符串的for循环代码页相同。
另外,还可以使用string对象数组,而不是字符串指针数组。在这种情况下,声明如下:
const string cities[Cities] =
{
"Gribble City",
"Gribbletown",
"New Gribble",
"San Gribble",
"Gribble Vista"
};
如果希望字符串是可修改的,应省略限定符const。使用string对象数组时,初始化列表和用于显示字符串的for循环代码与前两种方法中相同。在希望字符串是可修改的情况下,string类自动调整大小的特性使这种方法比使用二维数组更方便。