「C Sharpの基礎 - ファイル」の版間の差分

提供:MochiuWiki : SUSE, EC, PCB
ナビゲーションに移動 検索に移動
102行目: 102行目:
== テキストファイル ==
== テキストファイル ==
==== テキストファイルの読み込みと書き込み ====
==== テキストファイルの読み込みと書き込み ====
StreamReaderクラスを使用したテキストファイルの読み込み、using構文によるファイルのクローズ、例外処理など、<br>
<code>StreamReader</code>クラスを使用して、<br>
.NET Frameworkにおけるテキストファイルの読み込みの基本を解説する。<br>
テキストファイルの読み込みと書き込み、<code>using</code>構文によるファイルのクローズ、例外処理等を記載する。<br>
<br>
<br>
テキストファイルの内容を読み込むには、以下の3つの処理からなる。<br>
テキストファイルの内容を読み込むには、以下の3つの処理からなる。<br>
# '''ファイルを開く'''
# ファイルを開く
# '''ファイルの内容を読み込む'''
# ファイルの内容を読み込む
# '''ファイルを閉じる'''
# ファイルを閉じる
<br>
<br>
==== テキストファイルを開く ====
==== テキストファイルを開く ====
ファイルを開くには幾つかの記述方法があるが、UTF-8のファイルをファイル名を指定して開くには、<br>
ファイルを開くにはいくつかの方法が存在するが、UTF-8のファイルを開く場合、<br>
StreamReaderクラス(System.IO名前空間)を次のようにして使用するのが最も簡便である。<br>
<code>StreamReader</code>クラス(<code>System.IO</code>名前空間)を使用する。<br>
  <syntaxhighlight lang="c#">
  <syntaxhighlight lang="c#">
  StreamReader sr = new StreamReader(@"readme.txt", Encoding.UTF8);
  StreamReader sr = new StreamReader(@"readme.txt", Encoding.UTF8);
  </source>
  </syntaxhighlight>
<br>
<br>
また、Windowsで最も一般的に使用されているShift-JISのファイルを開くには次のようにする。<br>
また、Shift-JISのファイルを開く場合は、以下のように記述する。<br>
  <source lang="c#">
  <syntaxhighlight lang="c#">
  StreamReader sr = new StreamReader(@"readme.txt", Encoding.GetEncoding("Shift_JIS"));
  StreamReader sr = new StreamReader(@"readme.txt", Encoding.GetEncoding("Shift_JIS"));
  </syntaxhighlight>
  </syntaxhighlight>
<br>
<br>
==== テキストファイルを読み込む ====
==== テキストファイルを読み込む ====
StreamReaderクラスのReadToEndメソッドを使用することで、ファイルの内容を全て1つの文字列に読み込むことができる。<br>
<code>StreamReader</code>クラスの<code>ReadToEnd</code>メソッドを使用して、テキストファイルの内容を一括して読み込むことができる。<br>
ReadToEndメソッドでは、改行文字等も含めて全ての1つの文字列として読み込まれる。<br>
また、<code>ReadToEnd</code>メソッドでは、改行文字等も含めて、全ての1つの文字列として読み込まれる。<br>
<br>
<br>
==== usingステートメントを使用してファイルを閉じる ====
==== usingステートメントを使用してファイルを閉じる ====
StreamReaderクラスはIDisposableインタフェースを実装しているため(スーパークラスのTextReaderクラスが実装している)<br>
<code>StreamReader</code>クラスは、<code>IDisposable</code>インタフェースを実装しているため、(スーパークラスの<code>TextReader</code>クラスが実装している)<br>
Closeメソッドの代わりにusingステートメントを利用することも出来る。<br>
<code>Close</code>メソッドの代わりに、<code>using</code>ステートメントを利用することもできる。<br>
usingステートメントの実行を終えた時、自動的にStreamReaderクラスのDisposeメソッドを呼び出して、<br>
<br>
その中でCloseメソッドが呼び出される。<br>
<u><code>using</code>ステートメントを抜ける時、自動的に<code>StreamReader</code>クラスの<code>Dispose</code>メソッドを呼び出して、</u><br>
<u>その中で<code>Close</code>メソッドが呼び出される。</u><br>
<br>
<br>
==== ファイルが読み込めない時のエラー処理 ====
==== ファイルが読み込めない時のエラー処理 ====
開こうとするファイルが存在しない場合やファイルの読み込み中にディスク等に何らかの障害が発生した場合、<br>
ファイルが存在しない場合やファイルの読み込み中にディスク等に何らかの障害が発生する場合、<br>
例外が発生してプログラムの実行が中断されるので、例外処理を記述する必要がある。<br>
例外が発生してプログラムの実行が中断されるため、例外処理を記述する必要がある。<br>
<br>
<br>
==== サンプルコード(ファイルの内容を全て1つの文字列に読み込む) ====
==== テキストファイルの一括読み込み(変数の使用) ====
以下の例では、<code>ReadToEnd</code>メソッドを非同期処理に対応した<code>ReadToEndAsync</code>メソッドを使用して、<br>
ファイルの全ての内容を一括して読み込んでいる。<br>
  <syntaxhighlight lang="c#">
  <syntaxhighlight lang="c#">
  using System;
  using System;
159行目: 162行目:
  </syntaxhighlight>
  </syntaxhighlight>
<br>
<br>
==== サンプルコード(ファイルの内容を1行ずつ読み込む) ====
==== ファイルの内容を1行ずつ読み込む ====
処理によっては、ファイル全体を1つの文字列として読み込むよりも、1行ずつ読み込んで文字列の配列にした方が扱いやすい場合がある。<br>
ファイルの全ての内容を1つの変数に読み込むよりも、1行ずつ読み込んで文字列の配列に代入する方が扱いやすい場合がある。<br>
ファイルから1行読み込むには、ReadLineメソッドを使用する。<br>
<br>
ReadLineメソッドは、StreamReaderクラスにより開いたファイルから1行だけ読み取り、内部の読み取り位置を1行分進める。<br>
ファイルから1行読み込むには、<code>StreamReader</code>クラスの<code>ReadLine</code>メソッドを使用する。<br>
読み取り位置がファイルの最後に達した場合はnullを返す。<br>
<code>ReadLine</code>メソッドは、ファイルから1行のみ読み込み、内部で読み取り位置を1行分進める。<br>
読み取り位置がファイルの末尾に達した場合、<code>null</code>を返す。<br>
<br>
なお、.NET Framework 2.0以降では、<code>File</code>クラス(<code>System.IO</code>名前空間)において、<br>
ファイルを開く・内容を読み込む・ファイルを閉じるという一連の処理を1つのメソッドで行う<code>ReadAllText</code>メソッドと<code>ReadAllLines</code>メソッドが追加されている。<br>
  <syntaxhighlight lang="c#">
  <syntaxhighlight lang="c#">
  using System;
  using System;
194行目: 201行目:
  </syntaxhighlight>
  </syntaxhighlight>
<br>
<br>
尚、.NET Framework 2.0以降ではファイルを開き、内容を読み込み、閉じるという一連の処理を1つのメソッドで出来るReadAllTextメソッドと<br>
==== テキストファイルの一括読み込み(変数の使用) ====
ReadAllLinesメソッドがFileクラス(System.IO名前空間)に追加されている。<br>
テキストファイルの全ての内容を1つの変数に読み込む場合、<code>File</code>クラスの<code>ReadAllText</code>メソッドを使用する。<br>
<code>ReadAllText</code>メソッドで使用する標準の文字コードは、UTF-8である。<br>
<br>
<br>
==== サンプルコード(File.ReadAllTextメソッドによる文字列への読み込み) ====
以下の例では、テキストファイルの内容を一括して変数に読み込んでいる。<br>
テキストファイル全体を1つの文字列に読み込む場合は、FileクラスのReadAllTextメソッドが利用できる。<br>
ReadAllTextメソッドで使用するデフォルトの文字コードは、UTF-8である。<br>
以下にこれを使ったサンプルコードを示す。<br>
  <syntaxhighlight lang="c#">
  <syntaxhighlight lang="c#">
  using System.Text;
  using System.Text;
215行目: 220行目:
  </syntaxhighlight>
  </syntaxhighlight>
<br>
<br>
==== テキストファイルの一括読み込み ====
==== テキストファイルの一括読み込み(配列の使用) ====
テキストを1行ずつ処理する場合、各行を要素とする文字列配列を使用する。<br>
テキストを1行ずつ処理する場合、各行を要素とする文字列配列を使用する。<br>
これは、<code>System.IO.File</code>クラスの<code>ReadAllLines</code>メソッドを使用して実行する。<br>
これは、<code>System.IO.File</code>クラスの<code>ReadAllLines</code>メソッドを使用して実行する。<br>
223行目: 228行目:
ファイル末尾に改行文字がある場合、<code>ReadAllLines</code>メソッドで得られる行数は1行少なくなる。<br>
ファイル末尾に改行文字がある場合、<code>ReadAllLines</code>メソッドで得られる行数は1行少なくなる。<br>
<br>
<br>
以下の例では、テキストファイルの内容を一括して読み込んでいる。<br>
以下の例では、テキストファイルの内容を一括して配列に読み込んでいる。<br>
  <syntaxhighlight lang="c#">
  <syntaxhighlight lang="c#">
  using System.Text;
  using System.Text;

2021年3月5日 (金) 13:44時点における版

バイナリファイル

バイナリファイルの読み込みと書き込み

ファイルの内容をすべてbyte配列に読み込む、または、byte配列全体をファイルに書き込む場合、
Fileクラス(System.IO名前空間)のReadAllBytesメソッドおよびWriteAllBytesメソッドを使用する。
これらのメソッド(静的)では、ファイルを開く、byte配列への読み込みまたはbyte配列の書き込み、ファイルを閉じるという3つの処理を
ReadAllBytesメソッドで実行可能である。
ReadAllBytesメソッドは、読み込むファイルのパス名をパラメータに指定して呼ぶと、戻り値としてファイルの内容を読み込んだbyte配列を返す。
また、WriteAllBytesメソッドでは、書き込むファイルのパス名(ファイルが存在する場合は上書き)と書き込むbyte配列をパラメータで指定する。
尚、WriteAllBytesメソッドの戻り値はない。

 byte[] data = File.ReadAllBytes(@"hoge.bin"); // 読み込み
 File.WriteAllBytes(@"newhoge.bin", data);     // 書き込み


サンプルコード(バイナリファイルを固定サイズで分割する)

次のサンプルコードは、ReadAllBytesメソッドとWriteAllBytesメソッドを使用したサンプルコードである。
このプログラムでは、バイナリファイルを読み込み、それを固定サイズ(サイズ : 1[MB])の複数ファイルに分割して出力する。
尚、出力する最後のファイルは固定サイズに満たない場合がある。

 using System;
 using System.IO;
 
 class BinarySplit
 {
    static void Main()
    {
       // バイナリファイルの読み込み
       byte[] src = File.ReadAllBytes(@"hoge.bin");
 
       int FILESIZE = 1024 * 1024; // 分割サイズ

       int num = 0;
 
       for (int remain = src.Length; remain > 0; remain -= FILESIZE)
       {
          int length = Math.Min(FILESIZE, remain);  // 作成する分割ファイルの実際のサイズ

          byte[] dest = new byte[length];           // 分割ファイルへ書き出すbyte配列の作成
          Array.Copy(src, num * FILESIZE, dest, 0, length);
 
          // 出力ファイル名(out0001.bin、out0002.bin、……)
          string name = String.Format("out{0:D4}.bin", num + 1);

          // byte配列のファイルへの書き込み
          File.WriteAllBytes(name, dest);

          num++;
       }
    }
 }


ReadAllBytesメソッドは、ファイル全体を一度に読み込むため、巨大なファイルを扱う場合には、
読み込みに掛かる時間や使用するメモリ量について注意が必要である。

サンプルコード(非同期でバイナリファイルを読み書きする)

バイナリファイルへ書き込んだ後、それを読み込むサンプルコードに示す。
この例では、書き込み用に開くときにFileMode.Createを指定しているので上書きになる。(追記する場合 : FileMode.Append)

 async Task BinaryReadWriteAsync(byte[] data)
 {
    const string FilePath = @".\sample.dat";  // 読み書きするファイル

    // バイナリファイルへ書き込み
    // ファイルを上書きモードで開く(ファイルが存在しない場合は作成)
    // 追加モードにするにはFileModeをAppendに変える
    using (var fs = new FileStream(FilePath, FileMode.Create, FileAccess.Write))
    {
       await fs.WriteAsync(data, 0, data.Length);  // バイナリデータを非同期で書き込む
    }
 
    // バイナリファイル読み込み
    byte[] result;  // データを格納する配列

    // ファイルを読み取りモードで開く
    using (var fs = new FileStream(FilePath, FileMode.Open, FileAccess.Read))
    {
       result = new byte[fs.Length];  // データ格納用の配列を確保する
 
       // バイナリデータを非同期で読み込む
       await fs.ReadAsync(result, 0, (int)fs.Length);
    }

    // 読み込んだ内容をコンソールへ出力する
    for (int i=0; i < result.Length; i++)
    {
       Write($"{result[i]:X2} ");
       if (i % 16 == 7)
       {
          Write(" ");
       }
       else if (i % 16 == 15)
       {
          WriteLine();
       }
    }
    WriteLine();
 }



テキストファイル

テキストファイルの読み込みと書き込み

StreamReaderクラスを使用して、
テキストファイルの読み込みと書き込み、using構文によるファイルのクローズ、例外処理等を記載する。

テキストファイルの内容を読み込むには、以下の3つの処理からなる。

  1. ファイルを開く
  2. ファイルの内容を読み込む
  3. ファイルを閉じる


テキストファイルを開く

ファイルを開くにはいくつかの方法が存在するが、UTF-8のファイルを開く場合、
StreamReaderクラス(System.IO名前空間)を使用する。

 StreamReader sr = new StreamReader(@"readme.txt", Encoding.UTF8);


また、Shift-JISのファイルを開く場合は、以下のように記述する。

 StreamReader sr = new StreamReader(@"readme.txt", Encoding.GetEncoding("Shift_JIS"));


テキストファイルを読み込む

StreamReaderクラスのReadToEndメソッドを使用して、テキストファイルの内容を一括して読み込むことができる。
また、ReadToEndメソッドでは、改行文字等も含めて、全ての1つの文字列として読み込まれる。

usingステートメントを使用してファイルを閉じる

StreamReaderクラスは、IDisposableインタフェースを実装しているため、(スーパークラスのTextReaderクラスが実装している)
Closeメソッドの代わりに、usingステートメントを利用することもできる。

usingステートメントを抜ける時、自動的にStreamReaderクラスのDisposeメソッドを呼び出して、
その中でCloseメソッドが呼び出される。

ファイルが読み込めない時のエラー処理

ファイルが存在しない場合やファイルの読み込み中にディスク等に何らかの障害が発生する場合、
例外が発生してプログラムの実行が中断されるため、例外処理を記述する必要がある。

テキストファイルの一括読み込み(変数の使用)

以下の例では、ReadToEndメソッドを非同期処理に対応したReadToEndAsyncメソッドを使用して、
ファイルの全ての内容を一括して読み込んでいる。

 using System;
 using System.IO;
 using System.Text;
 
 string text = string.Empty;
 
 try
 {
    using (StreamReader sr = new StreamReader(@"hoge.txt", Encoding.UTF8))
    {
       text = await sr.ReadToEndAsync();
    }

    Console.Write(text); 
 }
 catch (Exception ex)
 {
    Console.WriteLine(ex.Message);
 }


ファイルの内容を1行ずつ読み込む

ファイルの全ての内容を1つの変数に読み込むよりも、1行ずつ読み込んで文字列の配列に代入する方が扱いやすい場合がある。

ファイルから1行読み込むには、StreamReaderクラスのReadLineメソッドを使用する。
ReadLineメソッドは、ファイルから1行のみ読み込み、内部で読み取り位置を1行分進める。
読み取り位置がファイルの末尾に達した場合、nullを返す。

なお、.NET Framework 2.0以降では、Fileクラス(System.IO名前空間)において、
ファイルを開く・内容を読み込む・ファイルを閉じるという一連の処理を1つのメソッドで行うReadAllTextメソッドとReadAllLinesメソッドが追加されている。

 using System;
 using System.IO;
 using System.Text;
 using System.Collections;
 
 string line = string.Empty;
 List<string> list = new List<string>();
 
 try
 {
    using (StreamReader sr = new StreamReader(@"hoge.txt", Encoding.UTF8))
    {
       while ((line = await sr.ReadLineAsync()) != null)
       {
          list.Add(line);
       }
    }
 }
 catch (Exception ex)
 {
    Console.WriteLine(ex.Message);
 }
 
 foreach (string strLine in list)
 {
    Console.WriteLine(strLine);
 }


テキストファイルの一括読み込み(変数の使用)

テキストファイルの全ての内容を1つの変数に読み込む場合、FileクラスのReadAllTextメソッドを使用する。
ReadAllTextメソッドで使用する標準の文字コードは、UTF-8である。

以下の例では、テキストファイルの内容を一括して変数に読み込んでいる。

 using System.Text;
 using System.IO;
 
 try
 {  // ファイルの読み込み
    string text1 = File.ReadAllText(@"hoge.txt", Encoding.UTF8);
 }
 catch (Exception ex)
 {
    Console.WriteLine(ex.Message);
 }


テキストファイルの一括読み込み(配列の使用)

テキストを1行ずつ処理する場合、各行を要素とする文字列配列を使用する。
これは、System.IO.FileクラスのReadAllLinesメソッドを使用して実行する。
ReadAllLinesメソッドで使用する標準の文字コードは、UTF-8である。

なお、ReadAllLinesメソッドは、ファイル末尾の改行文字を無視する。
ファイル末尾に改行文字がある場合、ReadAllLinesメソッドで得られる行数は1行少なくなる。

以下の例では、テキストファイルの内容を一括して配列に読み込んでいる。

 using System.Text;
 using System.IO;
 
 try
 {  // ファイルの読み込み
    string[] lines1 = File.ReadAllLines(@"hoge.txt", Encoding.UTF8);
 }
 catch (Exception ex)
 {
    Console.WriteLine(ex.Message);
 }



ファイルを確実に削除する

System.IO.FileクラスのDeleteメソッドを使用する場合、読み取り専用ファイルに対して実行すると例外が発生する。
これを解決するには、ファイルを削除する前に、ファイルの読み取り専用属性を解除する。

以下の例では、ファイルの読み取り専用属性を解除して、ファイルを削除している。

 public static void main(string[] args)
 {
    // 読み取り専用ファイルの場合でも削除する
    DeleteFile(@"C:\Hoge\");
 }
 
 /// -----------------------------------------------------------------------------
 /// <summary>
 ///     指定したファイルを削除する</summary>
 /// <param name="stFilePath">
 ///     削除するファイルまでのパス</param>
 /// -----------------------------------------------------------------------------
 public static void DeleteFile(string stFilePath)
 {
    System.IO.FileInfo cFileInfo = new System.IO.FileInfo(stFilePath);
 
    // ファイルが存在しているか判断する
    if(cFileInfo.Exists)
    {
       // 読み取り専用属性がある場合は、読み取り専用属性を解除する
       if((cFileInfo.Attributes & System.IO.FileAttributes.ReadOnly) == System.IO.FileAttributes.ReadOnly)
       {
          cFileInfo.Attributes = System.IO.FileAttributes.Normal;
       }
 
       // ファイルを削除する
       cFileInfo.Delete();
    }
 }