DelphiBasics
  Home  |  Files
 Documents
 Tutorials

 Writing your first program
 Writing your second program
 Amending this program

 Delphi data types
   Numbers
   Text (strings and chars)
   Sets and enumerations
   Arrays
   Records

 Programming logic
   Looping
   SubRoutines
   Exception handling

 Dates and times

 Files

 Pointers

 Printing text and graphics

 Object Orientation basics
   Memory leaks!
   Inheritance
   Abstraction
   Interfaces
   An example class

 References

 Standard components

 Articles

 A brief history of Delphi

 Usability : file handling

 Usability : reference books

 Author links

  Files
Delphi file support
Delphi provides a number of different file access mechanisms. The oldest is in support of consoles, where the Read, ReadLn, Write and WriteLn routines have a syntax that omits the file name. With no file name, IO (Input and Output) is routed to the console.
 

 
Of greater importance to modern applications, are disk file operations. Disks such as hard disks, floppy disks, CDs and DVDs (the latter are treated as read only).
 
Delphi confusingly provides two basic sets of routines for file handling. The most Delphi like are covered by this article and this web site. The other type are thin wrappers around Windows APIs - and as such are platform specific. They also support text files less intuitively. They are not covered here.
 
Additionally, hidden away, Delphi provides a very elegant way of reading and writing complete text files. The TStringList class has methods for loading the list of strings from a text file. And for saving the list likewise. See the final section of this article.
 

Accessing files
There are a number of basic operations for handling both text and binary files. (The latter can hold non character data values). First, we must get a handle for a named file:
 
 var
   myFile : TextFile;
 
 begin
   AssignFile(myFile, 'Test.txt');

Here we are getting a handle to a text file, designated by the TextFile type (binary files are of type File). We ask Delphi to assign a file handle for a file called 'Test.txt' which will be assumed to be in the current directory (as given by the GetCurrentDir routine).
 
Next, we must open the file using this handle. This operation tells Delphi how we want to treat the file. There are 3 ways of opening the file:
 
  href='RTL.asp?Name=ReWrite'>ReWrite Opens a file as new - discards existing contents if file exists
  href='RTL.asp?Name=Reset'>Reset Opens a file for read and write access
  href='RTL.asp?Name=Append'>Append Opens a file for appending to the end (such as a log file)

We'll cover the access mechanisms for text and binary files separately. Meanwhile, when we have finished, we must close the file:
 
   CloseFile(myFile);


Reading and writing to text files
Text files are great for simple uses, such as where we record a processing log. Text files fall short when reading and writing structured data. They do support number to string and string to number formatting, but you are often better off defining your own record structure and using a typed binary file instead.
 
Here is a simple example of access to a text file:
 
 var
   myFile : TextFile;
   text   : string;
 
 begin
  // Try to open the Test.txt file for writing to
   AssignFile(myFile, 'Test.txt');
   ReWrite(myFile);
 
  // Write a couple of well known words to this file
   WriteLn(myFile, 'Hello');
   WriteLn(myFile, 'World');
 
  // Close the file
   CloseFile(myFile);
 
  // Reopen the file for reading
   Reset(myFile);
 
  // Display the file contents
   while not Eof(myFile) do
   begin
     ReadLn(myFile, text);
     ShowMessage(text);
   end;
 
  // Close the file for the last time
   CloseFile(myFile);
 end;

 The ShowMessage routine displays the following :
 
 Hello
 World

If we replaced the ReWrite routine with Append, and rerun the code, the existing file would then contain:
 
 Hello
 World
 Hello
 World

since append retains the existing file contents, appending after this anything new written to it.
 
Notice that we have used WriteLn and ReadLn to write to and read from the file. This writes the given text plus a carriage return and line feed to the text. The read reads the whole line up to the carriage return. We have read from the file until Eof (End Of File) is true. See also Eoln.
 
We can use Read and Write to read and write multiple strings to a file. More importantly, we can use these to write numbers as strings, with some useful formatting (see Write for further details):
 
 var
   myFile : TextFile;
   text   : string;
   i      : Integer;
 
 begin
  // Try to open the Test.txt file for writing to
   AssignFile(myFile, 'Test.txt');
   ReWrite(myFile);
 
  // Write a couple of well known words to this file
   Write(myFile, 'Hello ');
   Write(myFile, 'World');
 
  // Terminate this line
   WriteLn(myFile);
 
  // Write some numbers to the file as a single line
   for i := 2 to 4 do
     Write(myFile, i/2, '  ');
 
  // Terminate this line
   WriteLn(myFile);
 
  // Repeat the above, but with number formatting
   for i := 2 to 4 do
     Write(myFile, i/2:5:1);
 
  // Terminate this line
   WriteLn(myFile);
 
  // Close the file
   CloseFile(myFile);
 
  // Reopen the file for reading only
   Reset(myFile);
 
  // Display the file contents
   while not Eof(myFile) do
   begin
     ReadLn(myFile, text);
     ShowMessage(text);
   end;
 
  // Close the file for the last time
   CloseFile(myFile);
 end;

 Hello World
 1.00000000000000E+0000  1.50000000000000E+0000  2.00000000000000E+0000
 1.0  1.5  2.0


Reading and writing to typed binary files
Typed binary files are files that have a data type as the basic unit of writing and reading. You write, say, an Integer, or a Record to a file, and read the same unit of data back. Records are particularly useful, allowing us to store any mix of data types in the one file unit of data.
 
This is best illustrated with an example:
 
 type
   TCustomer = Record
     name : string[20];
     age  : Integer;
     male : Boolean;
   end;
 
 var
   myFile   : File of TCustomer; // A file of customer records
   customer : TCustomer;         // A customer record variable
 
 begin
  // Try to open the Test.cus binary file for writing to
   AssignFile(myFile, 'Test.cus');
   ReWrite(myFile);
 
  // Write a couple of customer records to the file
   customer.name := 'Fred Bloggs';
   customer.age  := 21;
   customer.male := true;
   Write(myFile, customer);
 
   customer.name := 'Jane Turner';
   customer.age  := 45;
   customer.male := false;
   Write(myFile, customer);
 
  // Close the file
   CloseFile(myFile);
 
  // Reopen the file in read only mode
   FileMode := fmOpenRead;
   Reset(myFile);
 
  // Display the file contents
   while not Eof(myFile) do
   begin
     Read(myFile, customer);
     if customer.male
     then ShowMessage('Man with name '+customer.name+
                      ' is '+IntToStr(customer.age))
     else ShowMessage('Lady with name '+customer.name+
                      ' is '+IntToStr(customer.age));
   end;
 
  // Close the file for the last time
   CloseFile(myFile);
 end;

 Man with name Fred Bloggs is 21
 Lady with name Jane Turner is 45

The code is very similar to that used for text files, except that we define a file of a certain type (record), and pass/receive record data when writing/reading.
 

Reading and writing to pure binary files
Pure binary files are a bit peculiar. You must use BlockRead and BlockWrite instead of Read and Write. These have the added benefit of greater performance than the Read and Write, but are really geared at writing just blocks of binary data.
 
Here is an example :
 
 var
   myFile    : File;
   byteArray : array[1..8] of byte;
   oneByte   : byte;
   i, count  : Integer;
 
 begin
  // Try to open the Test.byt file for writing to
   AssignFile(myFile, 'Test.byt');
   ReWrite(myFile, 4);  // Define a single 'record' as 4 bytes
 
  // Fill out the data array
   for i := 1 to 8 do
     byteArray[i] := i;
 
  // Write the data array to the file
   BlockWrite(myFile, byteArray, 2);  // Write 2 'records' of 4 bytes
 
  // Fill out the data array with different data
   for i := 1 to 4 do
     byteArray[i] := i*i;             // Value : 1, 4, 9, 16
 
  // Write only the first 4 items from the data array to the file
   BlockWrite(myFile, byteArray, 1);  // Write 1 record of 4 bytes
 
  // Close the file
   CloseFile(myFile);
 
  // Reopen the file for reading only
   FileMode := fmOpenRead;
   Reset(myFile, 1);  // Now we define one record as 1 byte
 
  // Display the file contents
  // Start with a read of the first 6 bytes. 'count' is set to the
  // actual number read
   ShowMessage('Reading first set of bytes :');
   BlockRead(myFile, byteArray, 6, count);
 
  // Display the byte values read
   for i := 1 to count do
     ShowMessage(IntToStr(byteArray[i]));
 
  // Now read one byte at a time to the end of the file
   ShowMessage('Reading remaining bytes :');
   while not Eof(myFile) do
   begin
     BlockRead(myFile, oneByte, 1);  // Read and display one byte at a time
     ShowMessage(IntToStr(oneByte));
   end;
 
  // Close the file for the last time
   CloseFile(myFile);
 end;

 Reading first set of bytes :
 1
 2
 3
 4
 5
 6
 Reading remaining bytes :
 7
 8
 1
 4
 9
 16


Other file processing mechanisms
Typed binary files provide direct access methods in addition to sequential reads and writes. Click on a routine name to learn more:
 
  href='RTL.asp?Name=FilePos'>FilePos  Gives the file position in a binary or text file
  href='RTL.asp?Name=Seek'>Seek     Moves to a new position in the file
  href='RTL.asp?Name=SeekEof'>SeekEof  Skip to the end of the current line or file
  href='RTL.asp?Name=SeekEoln'>SeekEoln Skip to the end of the current line or file


Getting information about files and directories
We have only covered data access to files. There are a number of routines that allow you to do all sorts of things with files and directories that contain them:
 
  href='RTL.asp?Name=ChDir'>ChDir           Change the working drive plus path for a specified drive
  href='RTL.asp?Name=CreateDir'>CreateDir       Create a directory
  href='RTL.asp?Name=DeleteFile'>DeleteFile      Delete a file specified by its file name
  href='RTL.asp?Name=Erase'>Erase           Erase a file
  href='RTL.asp?Name=FileExists'>FileExists      Returns true if the given file exists
  href='RTL.asp?Name=FileSearch'>FileSearch      Search for a file in one or more directories
  href='RTL.asp?Name=FileSetDate'>FileSetDate     Set the last modified date and time of a file
  href='RTL.asp?Name=Flush'>Flush           Flushes buffered text file data to the file
  href='RTL.asp?Name=GetCurrentDir'>GetCurrentDir   Get the current directory (drive plus directory)
  href='RTL.asp?Name=MkDir'>MkDir           Make a directory
  href='RTL.asp?Name=RemoveDir'>RemoveDir       Remove a directory
  href='RTL.asp?Name=Rename'>Rename          Rename a file
  href='RTL.asp?Name=RenameFile'>RenameFile      Rename a file or directory
  href='RTL.asp?Name=RmDir'>RmDir           Remove a directory
  href='RTL.asp?Name=SelectDirectory'>SelectDirectory Display a dialog to allow user selection of a directory
  href='RTL.asp?Name=SetCurrentDir'>SetCurrentDir   Change the current directory
  href='RTL.asp?Name=Truncate'>Truncate        Truncates a file size


Using TStringList to read and write text files
The TStringList class is a very useful utility class that works on a lits of strings, each indexable like an array. The list can be sorted, and supports name/value pair strings, allowing selection by name or value.
 
These lists can be furnished from text files in one fell swoop. Here we show a TStringList object being created, and loaded from a file:
 
 var
   fileData : TStringList;               // Our TStringList variable
 begin
   fileData := TStringList.Create;       // Create the TSTringList object
   fileData.LoadFromFile('Testing.txt'); // Load from Testing.txt file
   ...

We can display the whole file in a Memo box:
 
   memoBox.Text := fileData.Text;

and we can display or process the file with direct access to any line at any time. In the example code below, we open a text file, reverse all lines in the file, and then save it back. Not terribly useful, but it shows the power of TStringList.
 
 var
   fileData : TStringList;
   saveLine : String;
   lines, i : Integer;
 begin
   fileData := TStringList.Create;       // Create the TSTringList object
   fileData.LoadFromFile('Test.txt');    // Load from Testing.txt file
 
  // Reverse the sequence of lines in the file
   lines := fileData.Count;
 
   for i := lines-1 downto (lines div 2) do
   begin
     saveLine := fileData[lines-i-1];
     fileData[lines-i-1] := fileData[i];
     fileData[i] := saveLine;
   end;
 
  // Now display the file
   for i := 0 to lines-1 do
     ShowMessage(fileData[i]);
 
   fileData.SaveToFile('Test.txt');      // Save the reverse sequence file
 end;

Take a look at TStringList to read more.
 
 
 

Delphi Basics © Neil Moffatt All rights reserved.  |  Home Page