C# 文件处理
一个 文件 是一个存储在磁盘中带有指定名称和目录路径的数据集合。当打开文件进行读写时,它变成一个 流。
从根本上说,流是通过通信路径传递的字节序列。有两个主要的流:输入流 和 输出流。输入流用于从文件读取数据(读操作),输出流用于向文件写入数据(写操作)。
一、C# I/O 类
System.IO 命名空间有各种不同的类,用于执行各种文件操作,如创建和删除文件、读取或写入文件,关闭文件等。
下表列出了一些 System.IO 命名空间中常用的非抽象类:
I/O 类 | 描述 |
---|---|
BinaryReader | 从二进制流读取原始数据。 |
BinaryWriter | 以二进制格式写入原始数据。 |
BufferedStream | 字节流的临时存储。 |
Directory | 有助于操作目录结构。 |
DirectoryInfo | 用于对目录执行操作。 |
DriveInfo | 提供驱动器的信息。 |
File | 有助于处理文件。 |
FileInfo | 用于对文件执行操作。 |
FileStream | 用于文件中任何位置的读写。 |
MemoryStream | 用于随机访问存储在内存中的数据流。 |
Path | 对路径信息执行操作。 |
StreamReader | 用于从字节流中读取字符。 |
StreamWriter | 用于向一个流中写入字符。 |
StringReader | 用于读取字符串缓冲区。 |
StringWriter | 用于写入字符串缓冲区。 |
二、FileStream 类
System.IO 命名空间中的 FileStream 类有助于文件的读写与关闭。该类派生自抽象类 Stream。
您需要创建一个 FileStream 对象来创建一个新的文件,或打开一个已有的文件。创建 FileStream 对象的语法如下:
1 2 3 |
FileStream <object_name> = <span class="hljs-keyword">new</span> FileStream( <file_name>, <FileMode Enumerator>, <FileAccess Enumerator>, <FileShare Enumerator>); |
例如,创建一个 FileStream 对象 F 来读取名为 sample.txt 的文件:
1 2 |
FileStream F = <span class="hljs-keyword">new</span> FileStream(<span class="hljs-string">"sample.txt"</span>, FileMode.Open, FileAccess.Read, FileShare.Read); |
参数 | 描述 |
---|---|
FileMode | FileMode 枚举定义了各种打开文件的方法。FileMode 枚举的成员有: • Append:打开一个已有的文件,并将光标放置在文件的末尾。如果文件不存在,则创建文件。 • Create:创建一个新的文件。如果文件已存在,则删除旧文件,然后创建新文件。 • CreateNew:指定操作系统应创建一个新的文件。如果文件已存在,则抛出异常。 • Open:打开一个已有的文件。如果文件不存在,则抛出异常。 • OpenOrCreate:指定操作系统应打开一个已有的文件。如果文件不存在,则用指定的名称创建一个新的文件打开。 • Truncate:打开一个已有的文件,文件一旦打开,就将被截断为零字节大小。然后我们可以向文件写入全新的数据,但是保留文件的初始创建日期。如果文件不存在,则抛出异常。 |
FileAccess | FileAccess 枚举的成员有:Read、ReadWrite 和 Write。 |
FileShare | FileShare 枚举的成员有: • Inheritable:允许文件句柄可由子进程继承。Win32 不直接支持此功能。 • None:谢绝共享当前文件。文件关闭前,打开该文件的任何请求(由此进程或另一进程发出的请求)都将失败。 • Read:允许随后打开文件读取。如果未指定此标志,则文件关闭前,任何打开该文件以进行读取的请求(由此进程或另一进程发出的请求)都将失败。但是,即使指定了此标志,仍可能需要附加权限才能够访问该文件。 • ReadWrite:允许随后打开文件读取或写入。如果未指定此标志,则文件关闭前,任何打开该文件以进行读取或写入的请求(由此进程或另一进程发出)都将失败。但是,即使指定了此标志,仍可能需要附加权限才能够访问该文件。 • Write:允许随后打开文件写入。如果未指定此标志,则文件关闭前,任何打开该文件以进行写入的请求(由此进程或另一进过程发出的请求)都将失败。但是,即使指定了此标志,仍可能需要附加权限才能够访问该文件。 • Delete:允许随后删除文件。 |
实例
下面的程序演示了 FileStream 类的用法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
<span class="hljs-keyword">using</span> System; <span class="hljs-keyword">using</span> System.IO; <span class="hljs-keyword">namespace</span> <span class="hljs-title">FileIOApplication</span> { <span class="hljs-keyword">class</span> <span class="hljs-title">Program</span> { <span class="hljs-function"><span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">Main</span>(<span class="hljs-params"><span class="hljs-keyword">string</span>[] args</span>) </span>{ FileStream F = <span class="hljs-keyword">new</span> FileStream(<span class="hljs-string">"test.dat"</span>, FileMode.OpenOrCreate, FileAccess.ReadWrite); <span class="hljs-keyword">for</span> (<span class="hljs-keyword">int</span> i = <span class="hljs-number">1</span>; i <= <span class="hljs-number">20</span>; i++) { F.WriteByte((<span class="hljs-keyword">byte</span>)i); } F.Position = <span class="hljs-number">0</span>; <span class="hljs-keyword">for</span> (<span class="hljs-keyword">int</span> i = <span class="hljs-number">0</span>; i <= <span class="hljs-number">20</span>; i++) { Console.Write(F.ReadByte() + <span class="hljs-string">" "</span>); } F.Close(); Console.ReadKey(); } } } |
当上面的代码被编译和执行时,它会产生下列结果:
1 2 |
<span class="hljs-number">1</span> <span class="hljs-number">2</span> <span class="hljs-number">3</span> <span class="hljs-number">4</span> <span class="hljs-number">5</span> <span class="hljs-number">6</span> <span class="hljs-number">7</span> <span class="hljs-number">8</span> <span class="hljs-number">9</span> <span class="hljs-number">10</span> <span class="hljs-number">11</span> <span class="hljs-number">12</span> <span class="hljs-number">13</span> <span class="hljs-number">14</span> <span class="hljs-number">15</span> <span class="hljs-number">16</span> <span class="hljs-number">17</span> <span class="hljs-number">18</span> <span class="hljs-number">19</span> <span class="hljs-number">20</span> <span class="hljs-number">-1</span> |
三、C# 高级文件操作
上面的实例演示了 C# 中简单的文件操作。但是,要充分利用 C# System.IO 类的强大功能,您需要知道这些类常用的属性和方法。 在下面的章节中,我们将讨论这些类和它们执行的操作。
四、文本文件的读写
StreamReader 和 StreamWriter 类用于文本文件的数据读写。这些类从抽象基类 Stream 继承,Stream 支持文件流的字节读写。
4.1 StreamReader 类
StreamReader 类继承自抽象基类 TextReader,表示阅读器读取一系列字符。
下表列出了 StreamReader 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public override void Close() 关闭 StreamReader 对象和基础流,并释放任何与读者相关的系统资源。 |
2 | public override int Peek() 返回下一个可用的字符,但不使用它。 |
3 | public override int Read() 从输入流中读取下一个字符,并把字符位置往前移一个字符。 |
如需查看完整的方法列表,请访问微软的 C# 文档。 实例
下面的实例演示了读取名为 Jamaica.txt 的文件。文件如下:
1 2 3 4 5 6 |
Down the way <span class="hljs-keyword">where</span> the nights are gay And the sun shines daily on the mountain top I took a trip on a sailing ship And <span class="hljs-keyword">when</span> I reached Jamaica I made a stop |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
<span class="hljs-keyword">using</span> System; <span class="hljs-keyword">using</span> System.IO; <span class="hljs-keyword">namespace</span> <span class="hljs-title">FileApplication</span> { <span class="hljs-keyword">class</span> <span class="hljs-title">Program</span> { <span class="hljs-function"><span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">Main</span>(<span class="hljs-params"><span class="hljs-keyword">string</span>[] args</span>) </span>{ <span class="hljs-keyword">try</span> { <span class="hljs-comment">// 创建一个 StreamReader 的实例来读取文件 </span> <span class="hljs-comment">// using 语句也能关闭 StreamReader</span> <span class="hljs-keyword">using</span> (StreamReader sr = <span class="hljs-keyword">new</span> StreamReader(<span class="hljs-string">"c:/jamaica.txt"</span>)) { <span class="hljs-keyword">string</span> line; <span class="hljs-comment">// 从文件读取并显示行,直到文件的末尾 </span> <span class="hljs-keyword">while</span> ((line = sr.ReadLine()) != <span class="hljs-keyword">null</span>) { Console.WriteLine(line); } } } <span class="hljs-keyword">catch</span> (Exception e) { <span class="hljs-comment">// 向用户显示出错消息</span> Console.WriteLine(<span class="hljs-string">"The file could not be read:"</span>); Console.WriteLine(e.Message); } Console.ReadKey(); } } } |
当您编译和执行上面的程序时,它会显示文件的内容。
4.2 StreamWriter 类
StreamWriter 类继承自抽象类 TextWriter,表示编写器写入一系列字符。
下表列出了 StreamWriter 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public override void Close() 关闭当前的 StreamWriter 对象和基础流。 |
2 | public override void Flush() 清理当前编写器的所有缓冲区,使得所有缓冲数据写入基础流。 |
3 | public virtual void Write(bool value) 把一个布尔值的文本表示形式写入到文本字符串或流。(继承自 TextWriter。) |
4 | public override void Write( char value ) 把一个字符写入到流。 |
5 | public virtual void Write( decimal value ) 把一个十进制值的文本表示形式写入到文本字符串或流。 |
6 | public virtual void Write( double value ) 把一个 8 字节浮点值的文本表示形式写入到文本字符串或流。 |
7 | public virtual void Write( int value ) 把一个 4 字节有符号整数的文本表示形式写入到文本字符串或流。 |
8 | public override void Write( string value ) 把一个字符串写入到流。 |
9 | public virtual void WriteLine() 把行结束符写入到文本字符串或流。 |
如需查看完整的方法列表,请访问微软的 C# 文档。
实例
下面的实例演示了使用 StreamWriter 类向文件写入文本数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
<span class="hljs-keyword">using</span> System; <span class="hljs-keyword">using</span> System.IO; <span class="hljs-keyword">namespace</span> <span class="hljs-title">FileApplication</span> { <span class="hljs-keyword">class</span> <span class="hljs-title">Program</span> { <span class="hljs-function"><span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">Main</span>(<span class="hljs-params"><span class="hljs-keyword">string</span>[] args</span>) </span>{ <span class="hljs-keyword">string</span>[] names = <span class="hljs-keyword">new</span> <span class="hljs-keyword">string</span>[] {<span class="hljs-string">"Zara Ali"</span>, <span class="hljs-string">"Nuha Ali"</span>}; <span class="hljs-keyword">using</span> (StreamWriter sw = <span class="hljs-keyword">new</span> StreamWriter(<span class="hljs-string">"names.txt"</span>)) { <span class="hljs-keyword">foreach</span> (<span class="hljs-keyword">string</span> s <span class="hljs-keyword">in</span> names) { sw.WriteLine(s); } } <span class="hljs-comment">// 从文件中读取并显示每行</span> <span class="hljs-keyword">string</span> line = <span class="hljs-string">""</span>; <span class="hljs-keyword">using</span> (StreamReader sr = <span class="hljs-keyword">new</span> StreamReader(<span class="hljs-string">"names.txt"</span>)) { <span class="hljs-keyword">while</span> ((line = sr.ReadLine()) != <span class="hljs-keyword">null</span>) { Console.WriteLine(line); } } Console.ReadKey(); } } } |
当上面的代码被编译和执行时,它会产生下列结果:
1 2 3 |
Zara Ali Nuha Ali |
五、二进制文件的读写
BinaryReader 和 BinaryWriter 类用于二进制文件的读写。
5.1 BinaryReader 类
BinaryReader 类用于从文件读取二进制数据。一个 BinaryReader 对象通过向它的构造函数传递 FileStream 对象而被创建。
下表列出了 BinaryReader 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public override void Close() 关闭 BinaryReader 对象和基础流。 |
2 | public virtual int Read() 从基础流中读取字符,并把流的当前位置往前移。 |
3 | public virtual bool ReadBoolean() 从当前流中读取一个布尔值,并把流的当前位置往前移一个字节。 |
4 | public virtual byte ReadByte() 从当前流中读取下一个字节,并把流的当前位置往前移一个字节。 |
5 | public virtual byte[] ReadBytes( int count ) 从当前流中读取指定数目的字节到一个字节数组中,并把流的当前位置往前移指定数目的字节。 |
6 | public virtual char ReadChar() 从当前流中读取下一个字节,并把流的当前位置按照所使用的编码和从流中读取的指定的字符往前移。 |
7 | public virtual char[] ReadChars( int count ) 从当前流中读取指定数目的字节,在一个字符数组中返回数组,并把流的当前位置按照所使用的编码和从流中读取的指定的字符往前移。 |
8 | public virtual double ReadDouble() 从当前流中读取一个 8 字节浮点值,并把流的当前位置往前移八个字节。 |
9 | public virtual int ReadInt32() 从当前流中读取一个 4 字节有符号整数,并把流的当前位置往前移四个字节。 |
10 | public virtual string ReadString() 从当前流中读取一个字符串。字符串以长度作为前缀,同时编码为一个七位的整数。 |
如需查看完整的方法列表,请访问微软的 C# 文档。
5.2 BinaryWriter 类
BinaryWriter 类用于向文件写入二进制数据。一个 BinaryWriter 对象通过向它的构造函数传递 FileStream 对象而被创建。
下表列出了 BinaryWriter 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public override void Close() 关闭 BinaryWriter 对象和基础流。 |
2 | public virtual void Flush() 清理当前编写器的所有缓冲区,使得所有缓冲数据写入基础设备。 |
3 | public virtual long Seek( int offset, SeekOrigin origin ) 设置当前流内的位置。 |
4 | public virtual void Write( bool value ) 把一个单字节的布尔值写入到当前流中,0 表示 false,1 表示 true。 |
5 | public virtual void Write( byte value ) 把一个无符号字节写入到当前流中,并把流的位置往前移一个字节。 |
6 | public virtual void Write( byte[] buffer ) 把一个字节数组写入到基础流中。 |
7 | public virtual void Write( char ch ) 把一个 Unicode 字符写入到当前流中,并把流的当前位置按照所使用的编码和要写入到流中的指定的字符往前移。 |
8 | public virtual void Write( char[] chars ) 把一个字符数组写入到当前流中,并把流的当前位置按照所使用的编码和要写入到流中的指定的字符往前移。 |
9 | public virtual void Write( double value ) 把一个 8 字节浮点值写入到当前流中,并把流位置往前移八个字节。 |
10 | public virtual void Write( int value ) 把一个 4 字节有符号整数写入到当前流中,并把流位置往前移四个字节。 |
11 | public virtual void Write( string value ) 把一个以长度为前缀的字符串写入到 BinaryWriter 的当前编码的流中,并把流的当前位置按照所使用的编码和要写入到流中的指定的字符往前移。 |
如需查看完整的方法列表,请访问微软的 C# 文档。
实例
下面的实例演示了读取和写入二进制数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 |
<span class="hljs-keyword">using</span> System; <span class="hljs-keyword">using</span> System.IO; <span class="hljs-keyword">namespace</span> <span class="hljs-title">BinaryFileApplication</span> { <span class="hljs-keyword">class</span> <span class="hljs-title">Program</span> { <span class="hljs-function"><span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">Main</span>(<span class="hljs-params"><span class="hljs-keyword">string</span>[] args</span>) </span>{ BinaryWriter bw; BinaryReader br; <span class="hljs-keyword">int</span> i = <span class="hljs-number">25</span>; <span class="hljs-keyword">double</span> d = <span class="hljs-number">3.14157</span>; <span class="hljs-keyword">bool</span> b = <span class="hljs-keyword">true</span>; <span class="hljs-keyword">string</span> s = <span class="hljs-string">"I am happy"</span>; <span class="hljs-comment">// 创建文件</span> <span class="hljs-keyword">try</span> { bw = <span class="hljs-keyword">new</span> BinaryWriter(<span class="hljs-keyword">new</span> FileStream(<span class="hljs-string">"mydata"</span>, FileMode.Create)); } <span class="hljs-keyword">catch</span> (IOException e) { Console.WriteLine(e.Message + <span class="hljs-string">"\n Cannot create file."</span>); <span class="hljs-keyword">return</span>; } <span class="hljs-comment">// 写入文件</span> <span class="hljs-keyword">try</span> { bw.Write(i); bw.Write(d); bw.Write(b); bw.Write(s); } <span class="hljs-keyword">catch</span> (IOException e) { Console.WriteLine(e.Message + <span class="hljs-string">"\n Cannot write to file."</span>); <span class="hljs-keyword">return</span>; } bw.Close(); <span class="hljs-comment">// 读取文件</span> <span class="hljs-keyword">try</span> { br = <span class="hljs-keyword">new</span> BinaryReader(<span class="hljs-keyword">new</span> FileStream(<span class="hljs-string">"mydata"</span>, FileMode.Open)); } <span class="hljs-keyword">catch</span> (IOException e) { Console.WriteLine(e.Message + <span class="hljs-string">"\n Cannot open file."</span>); <span class="hljs-keyword">return</span>; } <span class="hljs-keyword">try</span> { i = br.ReadInt32(); Console.WriteLine(<span class="hljs-string">"Integer data: {0}"</span>, i); d = br.ReadDouble(); Console.WriteLine(<span class="hljs-string">"Double data: {0}"</span>, d); b = br.ReadBoolean(); Console.WriteLine(<span class="hljs-string">"Boolean data: {0}"</span>, b); s = br.ReadString(); Console.WriteLine(<span class="hljs-string">"String data: {0}"</span>, s); } <span class="hljs-keyword">catch</span> (IOException e) { Console.WriteLine(e.Message + <span class="hljs-string">"\n Cannot read from file."</span>); <span class="hljs-keyword">return</span>; } br.Close(); Console.ReadKey(); } } } |
当上面的代码被编译和执行时,它会产生下列结果:
1 2 3 4 5 |
Integer data: <span class="hljs-number">25</span> Double data: <span class="hljs-number">3.14157</span> Boolean data: True String data: I am happy |
六、Windows 文件系统的操作
C# 允许您使用各种目录和文件相关的类来操作目录和文件,比如 DirectoryInfo 类和 FileInfo 类。
6.1 DirectoryInfo 类
DirectoryInfo 类派生自 FileSystemInfo 类。它提供了各种用于创建、移动、浏览目录和子目录的方法。该类不能被继承。
下表列出了 DirectoryInfo 类中一些常用的属性:
序号 | 属性 & 描述 |
---|---|
1 | Attributes 获取当前文件或目录的属性。 |
2 | CreationTime 获取当前文件或目录的创建时间。 |
3 | Exists 获取一个表示目录是否存在的布尔值。 |
4 | Extension 获取表示文件存在的字符串。 |
5 | FullName 获取目录或文件的完整路径。 |
6 | LastAccessTime 获取当前文件或目录最后被访问的时间。 |
7 | Name 获取该 DirectoryInfo 实例的名称。 |
下表列出了 DirectoryInfo 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public void Create() 创建一个目录。 |
2 | public DirectoryInfo CreateSubdirectory( string path ) 在指定的路径上创建子目录。指定的路径可以是相对于 DirectoryInfo 类的实例的路径。 |
3 | public override void Delete() 如果为空的,则删除该 DirectoryInfo。 |
4 | public DirectoryInfo[] GetDirectories() 返回当前目录的子目录。 |
5 | public FileInfo[] GetFiles() 从当前目录返回文件列表。 |
如需查看完整的属性和方法列表,请访问微软的 C# 文档。
6.2 FileInfo 类
FileInfo 类派生自 FileSystemInfo 类。它提供了用于创建、复制、删除、移动、打开文件的属性和方法,且有助于 FileStream 对象的创建。该类不能被继承。
下表列出了 FileInfo 类中一些常用的属性:
序号 | 属性 & 描述 |
---|---|
1 | Attributes 获取当前文件的属性。 |
2 | CreationTime 获取当前文件的创建时间。 |
3 | Directory 获取文件所属目录的一个实例。 |
4 | Exists 获取一个表示文件是否存在的布尔值。 |
5 | Extension 获取表示文件存在的字符串。 |
6 | FullName 获取文件的完整路径。 |
7 | LastAccessTime 获取当前文件最后被访问的时间。 |
8 | LastWriteTime 获取文件最后被写入的时间。 |
9 | Length 获取当前文件的大小,以字节为单位。 |
10 | Name 获取文件的名称。 |
下表列出了 FileInfo 类中一些常用的方法:
序号 | 方法 & 描述 |
---|---|
1 | public StreamWriter AppendText() 创建一个 StreamWriter,追加文本到由 FileInfo 的实例表示的文件中。 |
2 | public FileStream Create() 创建一个文件。 |
3 | public override void Delete() 永久删除一个文件。 |
4 | public void MoveTo( string destFileName ) 移动一个指定的文件到一个新的位置,提供选项来指定新的文件名。 |
5 | public FileStream Open( FileMode mode ) 以指定的模式打开一个文件。 |
6 | public FileStream Open( FileMode mode, FileAccess access ) 以指定的模式,使用 read、write 或 read/write 访问,来打开一个文件。 |
7 | public FileStream Open( FileMode mode, FileAccess access, FileShare share ) 以指定的模式,使用 read、write 或 read/write 访问,以及指定的分享选项,来打开一个文件。 |
8 | public FileStream OpenRead() 创建一个只读的 FileStream。 |
9 | public FileStream OpenWrite() 创建一个只写的 FileStream。 |
如需查看完整的属性和方法列表,请访问微软的 C# 文档。
实例
下面的实例演示了上面提到的类的用法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
<span class="hljs-keyword">using</span> System; <span class="hljs-keyword">using</span> System.IO; <span class="hljs-keyword">namespace</span> <span class="hljs-title">WindowsFileApplication</span> { <span class="hljs-keyword">class</span> <span class="hljs-title">Program</span> { <span class="hljs-function"><span class="hljs-keyword">static</span> <span class="hljs-keyword">void</span> <span class="hljs-title">Main</span>(<span class="hljs-params"><span class="hljs-keyword">string</span>[] args</span>) </span>{ <span class="hljs-comment">// 创建一个 DirectoryInfo 对象</span> DirectoryInfo mydir = <span class="hljs-keyword">new</span> DirectoryInfo(<span class="hljs-string">@"c:\Windows"</span>); <span class="hljs-comment">// 获取目录中的文件以及它们的名称和大小</span> FileInfo [] f = mydir.GetFiles(); <span class="hljs-keyword">foreach</span> (FileInfo file <span class="hljs-keyword">in</span> f) { Console.WriteLine(<span class="hljs-string">"File Name: {0} Size: {1}"</span>, file.Name, file.Length); } Console.ReadKey(); } } } |
当您编译和执行上面的程序时,它会显示文件的名称及它们在 Windows 目录中的大小。