Bài giảng Lập trình C++ - Chương 14: File Processing
Outline
14.1 Introduction
14.2 The Data Hierarchy
14.3 Files and Streams
14.4 Creating a Sequential-Access File
14.5 Reading Data from a Sequential-Access File
14.6 Updating Sequential-Access Files
14.7 Random-Access Files
14.8 Creating a Random-Access File
14.9 Writing Data Randomly to a Random-Access File
14.10 Reading Data Sequentially from a Random-Access File
14.11 Example: A Transaction-Processing Program
14.12 Input/Output of Objects
account-number value 30 void ClientData::setAccountNumber( int accountNumberValue ) 31 { 32 accountNumber = accountNumberValue; 33 34 } // end function setAccountNumber 35 36 // get last-name value 37 string ClientData::getLastName() const 38 { 39 return lastName; 40 41 } // end function getLastName 42 43 // set last-name value 44 void ClientData::setLastName( string lastNameString ) 45 { 46 // copy at most 15 characters from string to lastName 47 const char *lastNameValue = lastNameString.data(); 48 int length = strlen( lastNameValue ); 49 length = ( length 4 5 using std::cerr; 6 using std::endl; 7 using std::ios; 8 9 #include 10 11 using std::ofstream; 12 13 #include 14 #include "clientData.h" // ClientData class definition 15 16 int main() 17 { 18 ofstream outCredit( "credit.dat", ios::binary ); 19 20 // exit program if ofstream could not open file 21 if ( !outCredit ) { 22 cerr ( &blankClient ), 34 sizeof( ClientData ) ); 35 36 return 0; 37 38 } // end main * 14.9 Writing Data Randomly to a Random-Access File Use seekp to write to exact location in file Where does the first record begin? Byte 0 The second record? Byte 0 + sizeof(object) Any record? (Recordnum - 1) * sizeof(object) * fig14_13.cpp(1 of 4) 1 // Fig. 14.13: fig14_13.cpp 2 // Writing to a random access file. 3 #include 4 5 using std::cerr; 6 using std::endl; 7 using std::cout; 8 using std::cin; 9 using std::ios; 10 11 #include 12 13 using std::setw; 14 15 #include 16 17 using std::ofstream; 18 19 #include 20 #include "clientData.h" // ClientData class definition 21 * fig14_13.cpp(2 of 4) 22 int main() 23 { 24 int accountNumber; 25 char lastName[ 15 ]; 26 char firstName[ 10 ]; 27 double balance; 28 29 ofstream outCredit( "credit.dat", ios::binary ); 30 31 // exit program if ofstream cannot open file 32 if ( !outCredit ) { 33 cerr > accountNumber; 44 client.setAccountNumber( accountNumber ); 45 * fig14_13.cpp(3 of 4) 46 // user enters information, which is copied into file 47 while ( client.getAccountNumber() > 0 && 48 client.getAccountNumber() > setw( 15 ) >> lastName; 53 cin >> setw( 10 ) >> firstName; 54 cin >> balance; 55 56 // set record lastName, firstName and balance values 57 client.setLastName( lastName ); 58 client.setFirstName( firstName ); 59 client.setBalance( balance ); 60 61 // seek position in file of user-specified record 62 outCredit.seekp( ( client.getAccountNumber() - 1 ) * 63 sizeof( ClientData ) ); 64 65 // write user-specified information in file 66 outCredit.write( 67 reinterpret_cast( &client ), 68 sizeof( ClientData ) ); 69 * fig14_13.cpp(4 of 4) 70 // enable user to specify another account number 71 cout > accountNumber; 73 client.setAccountNumber( accountNumber ); 74 75 } // end while 76 77 return 0; 78 79 } // end main * fig14_13.cppoutput (1 of 1) Enter account number (1 to 100, 0 to end input) ? 37 Enter lastname, firstname, balance ? Barker Doug 0.00 Enter account number ? 29 Enter lastname, firstname, balance ? Brown Nancy -24.54 Enter account number ? 96 Enter lastname, firstname, balance ? Stone Sam 34.98 Enter account number ? 88 Enter lastname, firstname, balance ? Smith Dave 258.34 Enter account number ? 33 Enter lastname, firstname, balance ? Dunn Stacey 314.33 Enter account number ? 0 * 14.10 Reading Data Sequentially from a Random-Access File read - similar to write Reads raw bytes from file into memory inFile.read( reinterpret_cast( &number ), sizeof( int ) ); &number: location to store data sizeof(int): how many bytes to read Do not use inFile >> number with raw bytes >> expects char * Upcoming program Output data from a random-access file Go through each record sequentially If no data (accountNumber == 0) then skip * fig14_14.cpp(1 of 3) 1 // Fig. 14.14: fig14_14.cpp 2 // Reading a random access file. 3 #include 4 5 using std::cout; 6 using std::endl; 7 using std::ios; 8 using std::cerr; 9 using std::left; 10 using std::right; 11 using std::fixed; 12 using std::showpoint; 13 14 #include 15 16 using std::setprecision; 17 using std::setw; 18 19 #include 20 21 using std::ifstream; 22 using std::ostream; 23 24 #include // exit protoyype 25 #include "clientData.h" // ClientData class definition 26 * fig14_14.cpp(2 of 3) 27 void outputLine( ostream&, const ClientData & ); 28 29 int main() 30 { 31 ifstream inCredit( "credit.dat", ios::in ); 32 33 // exit program if ifstream cannot open file 34 if ( !inCredit ) { 35 cerr ( &client ), 48 sizeof( ClientData ) ); 49 * fig14_14.cpp(3 of 3) 50 // read all records from file 51 while ( inCredit && !inCredit.eof() ) { 52 53 // display record 54 if ( client.getAccountNumber() != 0 ) 55 outputLine( cout, client ); 56 57 // read next from file 58 inCredit.read( reinterpret_cast( &client ), 59 sizeof( ClientData ) ); 60 61 } // end while 62 63 return 0; 64 65 } // end main 66 67 // display single record 68 void outputLine( ostream &output, const ClientData &record ) 69 { 70 output 6 7 using std::cout; 8 using std::cerr; 9 using std::cin; 10 using std::endl; 11 using std::ios; 12 using std::left; 13 using std::right; 14 using std::fixed; 15 using std::showpoint; 16 17 #include 18 19 using std::ofstream; 20 using std::ostream; 21 using std::fstream; 22 23 #include 24 25 using std::setw; 26 using std::setprecision; 27 28 #include // exit prototype 29 #include "clientData.h" // ClientData class definition * fig14_15.cpp(2 of 14) 30 31 int enterChoice(); 32 void printRecord( fstream& ); 33 void updateRecord( fstream& ); 34 void newRecord( fstream& ); 35 void deleteRecord( fstream& ); 36 void outputLine( ostream&, const ClientData & ); 37 int getAccount( const char * const ); 38 39 enum Choices { PRINT = 1, UPDATE, NEW, DELETE, END }; 40 41 int main() 42 { 43 // open file for reading and writing 44 fstream inOutCredit( "credit.dat", ios::in | ios::out ); 45 46 // exit program if fstream cannot open file 47 if ( !inOutCredit ) { 48 cerr > menuChoice; // receive choice from user 109 110 return menuChoice; 111 112 } // end function enterChoice 113 114 // create formatted text file for printing 115 void printRecord( fstream &readFromFile ) 116 { 117 // create text file 118 ofstream outPrintFile( "print.txt", ios::out ); 119 120 // exit program if ofstream cannot create file 121 if ( !outPrintFile ) { 122 cerr ( &client ), 137 sizeof( ClientData ) ); 138 139 // copy all records from record file into text file 140 while ( !readFromFile.eof() ) { 141 142 // write single record to text file 143 if ( client.getAccountNumber() != 0 ) 144 outputLine( outPrintFile, client ); 145 146 // read next record from record file 147 readFromFile.read( reinterpret_cast( &client ), 148 sizeof( ClientData ) ); 149 150 } // end while 151 152 } // end function printRecord 153 * fig14_15.cpp(8 of 14) 154 // update balance in record 155 void updateRecord( fstream &updateFile ) 156 { 157 // obtain number of account to update 158 int accountNumber = getAccount( "Enter account to update" ); 159 160 // move file-position pointer to correct record in file 161 updateFile.seekg( 162 ( accountNumber - 1 ) * sizeof( ClientData ) ); 163 164 // read first record from file 165 ClientData client; 166 updateFile.read( reinterpret_cast( &client ), 167 sizeof( ClientData ) ); 168 169 // update record 170 if ( client.getAccountNumber() != 0 ) { 171 outputLine( cout, client ); 172 173 // request user to specify transaction 174 cout > transaction; 177 178 // update record balance 179 double oldBalance = client.getBalance(); 180 client.setBalance( oldBalance + transaction ); 181 outputLine( cout, client ); 182 * fig14_15.cpp(9 of 14) 183 // move file-position pointer to correct record in file 184 updateFile.seekp( 185 ( accountNumber - 1 ) * sizeof( ClientData ) ); 186 187 // write updated record over old record in file 188 updateFile.write( 189 reinterpret_cast( &client ), 190 sizeof( ClientData ) ); 191 192 } // end if 193 194 // display error if account does not exist 195 else 196 cerr ( &client ), 214 sizeof( ClientData ) ); 215 216 // create record, if record does not previously exist 217 if ( client.getAccountNumber() == 0 ) { 218 219 char lastName[ 15 ]; 220 char firstName[ 10 ]; 221 double balance; 222 223 // user enters last name, first name and balance 224 cout > setw( 15 ) >> lastName; 226 cin >> setw( 10 ) >> firstName; 227 cin >> balance; 228 229 // use values to populate account values 230 client.setLastName( lastName ); 231 client.setFirstName( firstName ); 232 client.setBalance( balance ); 233 client.setAccountNumber( accountNumber ); 234 * fig14_15.cpp(11 of 14) 235 // move file-position pointer to correct record in file 236 insertInFile.seekp( ( accountNumber - 1 ) * 237 sizeof( ClientData ) ); 238 239 // insert record in file 240 insertInFile.write( 241 reinterpret_cast( &client ), 242 sizeof( ClientData ) ); 243 244 } // end if 245 246 // display error if account previously exists 247 else 248 cerr ( &client ), 266 sizeof( ClientData ) ); 267 268 // delete record, if record exists in file 269 if ( client.getAccountNumber() != 0 ) { 270 ClientData blankClient; 271 272 // move file-position pointer to correct record in file 273 deleteFromFile.seekp( ( accountNumber - 1 ) * 274 sizeof( ClientData ) ); 275 * fig14_15.cpp(13 of 14) 276 // replace existing record with blank record 277 deleteFromFile.write( 278 reinterpret_cast( &blankClient ), 279 sizeof( ClientData ) ); 280 281 cout > accountNumber; 311 312 } while ( accountNumber 100 ); 313 314 return accountNumber; 315 316 } // end function getAccount * 14.12 Input/Output of Objects I/O of objects Chapter 8 (overloaded >>) Only object's data transmitted Member functions available internally When objects stored in file, lose type info (class, etc.) Program must know type of object when reading One solution When writing, output object type code before real object When reading, read type code Call proper overloaded function (switch)
File đính kèm:
- Bài giảng Lập trình C++ - Chương 14 File Processing.ppt