Group
File Handling in C#
Objective
1. Provide the file name to the program when prompted.
2. The program will read the file and display its contents in hexadecimal format.
3. Each row will display 16 bytes in hexadecimal followed by their corresponding ASCII characters.
4. Bytes with ASCII code less than 32 will be displayed as a dot.
5. After displaying 24 rows (16 bytes per row), the program will pause and wait for the user to press a key to display the next 24 rows.
6. To exit the program, press 'q' when prompted after displaying the final screen of content.
Create a "dump" utility: a hex viewer that displays the contents of a file, with 16 bytes in each row and 24 rows in each screen. The program should pause after displaying each screen before displaying the next 24 rows.
Example C# Exercise
Show C# Code
using System;
using System.IO;
class HexViewerUtility
{
// Main method that begins execution of the program
static void Main(string[] args)
{
// Prompt the user to enter the file name
Console.Write("Enter the file name: ");
string fileName = Console.ReadLine();
// Check if the file exists
if (!File.Exists(fileName))
{
Console.WriteLine("File does not exist.");
return;
}
// Open the file for reading
byte[] fileBytes = File.ReadAllBytes(fileName);
int totalBytes = fileBytes.Length;
int bytesPerRow = 16; // Number of bytes to display per row
int rowsPerScreen = 24; // Number of rows to display per screen
int screenCount = (int)Math.Ceiling((double)totalBytes / (bytesPerRow * rowsPerScreen));
// Loop through each screen
for (int screen = 0; screen < screenCount; screen++)
{
Console.Clear(); // Clear the console for the next screen
// Loop through each row in the screen
for (int row = 0; row < rowsPerScreen; row++)
{
int offset = (screen * rowsPerScreen + row) * bytesPerRow;
if (offset >= totalBytes) break; // Exit if we reach the end of the file
// Display the hexadecimal bytes in this row
for (int byteIndex = 0; byteIndex < bytesPerRow; byteIndex++)
{
if (offset + byteIndex < totalBytes)
{
Console.Write($"{fileBytes[offset + byteIndex]:X2} "); // Print byte in hexadecimal format
}
else
{
Console.Write(" "); // Padding if we don't have 16 bytes for this row
}
}
Console.Write(" "); // Space between hex and ASCII parts
// Display the corresponding ASCII characters in this row
for (int byteIndex = 0; byteIndex < bytesPerRow; byteIndex++)
{
if (offset + byteIndex < totalBytes)
{
char currentChar = (char)fileBytes[offset + byteIndex];
Console.Write(currentChar >= 32 && currentChar <= 126 ? currentChar : '.'); // Replace non-printable characters with a dot
}
else
{
Console.Write(" "); // Padding for missing bytes in this row
}
}
Console.WriteLine(); // Move to the next line after this row
}
// Pause after displaying each screen
if (screen < screenCount - 1)
{
Console.WriteLine("\nPress any key to view the next screen...");
Console.ReadKey();
}
}
Console.WriteLine("End of file display.");
}
}
Output
//Case 1 - Displaying File Content:
//Assume the file contains the following byte sequence:
48 65 6C 6C 6F 20 57 6F 72 6C 64 0A 0D
//The program will display:
48 65 6C 6C 6F 20 57 6F 72 6C 64 0A 0D . . . . . . . . . . . . .
//After showing 24 rows of data, the program will pause, and the user can press any key to continue viewing the next set of data.
//Case 2 - File Does Not Exist:
//For the command:
Enter the file name: nonExistentFile.bin
//Output:
File does not exist.