我们知道,一个字段是与单个项目有关的单块信息。而记录由字段组成,是关于单个项目的完整信息集。例如,一组字段可能是一个人的姓名、年龄、地址和电话号码,而所有这些与一个人有关的字段则共同组成了一个记录。
在 C++ 中,结构体提供了一种将信息组织到字段和记录中的便捷方法。例如,下面的结构体声明可以用来创建一个包含某人信息的记录:
const int NAME_SIZE = 51, ADDR_SIZE = 51, PHONE_SIZE = 14;
struct Info
{
char name[NAME_SIZE];
int age;
char address1[ADDR_SIZE];
char address2[ADDR_SIZE];
char phone[PHONE_SIZE];
};
除了提供信息的组织结构之外,结构体还可以将信息打包成一个单元。例如,假设结构体变量 person 被声明如下:
一旦 person 的成员(或字段)填充了信息,则可以使用 write 函数将整个变量写入一个文件中:
第一个实参是 person 变量的地址。reinterpret_cast<char*> 转换操作符是必需的,因为 write 需要第一个实参是一个指向 char 的指针。如果将除 char 之外的其他任何东西的地址传递给 write 函数,则必须使用转换操作符使它看起来像是一个指向 char 的指针。第二个实参是 sizeof 运算符,它告诉 write 有多少个字节要写入文件。
下面的程序演示了这种技术:
//This program demonstrates the use of a structure variable to
//store a record of information to a file.
#include <iostream>
#include <fstream>
#include <cstring>
#include <cctype> // for toupper
using namespace std;
const int NAME_SIZE = 51, ADDR_SIZE = 51, PHONE_SIZE = 14;
struct Info
{
char name[NAME_SIZE];
int age;
char address1[ADDR_SIZE];
char address2[ADDR_SIZE];
char phone[PHONE_SIZE];
};
int main()
{
Info person; // Store information about a person
char response; // User response
string input; // Used to read strings
// Create file object and open file
fstream people("people.dat", ios::out | ios::binary);
if (!people)
{
cout << "Error opening file. Program aborting.\n";
return 0;
}
// Keep getting information from user and writing it to the file in binary mode
do {
cout << "Enter person information:\n";
cout << "Name: ";
getline (cin, input);
strcpy(person.name, input.c_str());
cout << "Age:";
cin >> person.age;
cin.ignore(); // Skip over remaining newline
cout << "Address line 1: ";
getline(cin, input);
strcpy(person.address1, input.c_str());
cout << "Address line 2: ";
getline(cin, input);
strcpy(person.address2, input.c_str());
cout << "Phone : ";
getline(cin, input);
strcpy(person.phone, input.c_str());
people.write(reinterpret_cast<char *>(&person), sizeof(person));
cout << "Do you want to enter another record? ";
cin >> response;
cin.ignore();
} while (toupper(response) == 'Y');
// Close file
people.close ();
return 0;
}
程序输出结果:
注意,由于结构可以包含混合的数据类型,所以当打开一个文件来存储它们时,应该总是使用 ios::binary 模式。
此程序允许通过填充 person 变量的成员来创建一个文件,然后把变量写入文件。要将 C 字符串读入一个数组,则程序首先应使用 getline 函数读取字符串对象,然后使用 strcpy 将 C 字符串移动到字符数组中。下面的程序打开文件并将每个记录读入 person 变量,然后在屏幕上显示该信息:
// This program demonstrates the use of a structure
// variable to read a record of information from a file.
#include <iostream>
#include <fstream>
using namespace std;
const int NAME_SIZE = 51, ADDR_SIZE = 51, PH〇NE_SIZE = 14; struct Info {
char name[NAME_SIZE];
int age;
char addressl[ADDR_SIZE];
char address2 [ADDR_SIZE];
char phone[PHONE_SIZE];
};
int main()
{
Info person; // Store person information
char response; // User response
// Create file object and open file for binary reading
fstream people("people.dat", ios::in | ios:ibinary);
if (!people)
{
cout << "Error opening file. Program aborting.\n";
return 0;
}
// Label the output
cout << "Here are the people in the file:\n\n";
// Read one structure at a time and echo to screen
people.read(reinterpret_cast<char *>(&person), sizeof (person));
while (!people.eof())
{
cout << "Name: ";
cout << person.name << endl;
cout << "Age: ";
cout << person.age << endl;
cout << "Address line 1: ";
cout << person.address1 << endl;
cout << "Address line 2: ";
cout << person.address2 << endl;
cout << "Phone: ";
cout << person.phone << endl;
cout << "\nStrike any key to see the next record.\n";
cin.get (response);
people.read(reinterpret_cast<char *>(&person), sizeof(person));
}
cout << "That's all the information in the file!\n";
people.close();
return 0;
}
程序输出结果:
注意,如果结构中包含指针,则使用本节介绍的技术将无法正确存储到磁盘。这是因为如果在程序的后续运行中将结构读入内存中,则不能保证所有程序变量都位于相同的内存位置。由于 string 类对象包含隐式指针,所以它们不能作为将要存储的结构的一部分。