一般而言,文本文件都只會包含一種格式(例如,以逗號分隔或是固定字段寬度),然而文本文件卻很可能含有多種格式,當遇到此種狀況時該如何處理呢?
如果您的文本文件含有多種格式,應該使用TextFieldParser對象的PeekChars方法去取得數據行從頭算起特定數目的字符,以便藉此來判斷該數據行的格式,然後告知TextFieldParser對象該數據行的格式為何並讀取該數據行。PeekChars方法只會返回指定數目的字符而且不會前進至下一行,通過這樣逐一判別每一筆數據行的格式並逐行讀取數據行的方式,即可夠解析含有多種格式的文字文件並順利讀取。
舉例來說,假設我們在應用程序項目的Text文件夾含有一個名稱為“多格式文本文件.txt”的文本文件,此文本文件的特殊之處在於它含有下列三種格式:
·固定字段寬度分別為5、10與 -1的數據行。
·固定字段寬度分別為6、10、17與 -1的數據行。
·采用逗號分隔的數據行。
上述這三種格式的數據行存在一個特點,就是數據行的開頭字符分別是CK、PB與SP,因此我們只要通過TextFieldParser對象的PeekChars方法取得數據行的前兩個字符,然後根據其值來設定TextFieldType屬性,以及設定Delimiters屬性(或調用SetDelimiters方法)或FieldWidths屬性(或調用SetFieldWidths方法),並使用ReadFields方法來讀取數據行。反復使用此方式來處理每一筆數據行,就能夠順利解析並讀取整個文本文件。
以下的程序代碼示范如何將“多格式文本文件.txt”中的三種格式數據解析出來並分別顯示於各自的DataGridView控件中。相關程序代碼列示如下:
private void CH1_DemoForm035_Load(object sender, EventArgs e) { txtResult.Text = File.ReadAllText(@"Text多格式文本文件.txt"); DataGridView1.ColumnHeadersVisible = true; // 設定欄標題樣式。 DataGridViewCellStyle columnHeaderStyle = new DataGridViewCellStyle(); columnHeaderStyle.BackColor = Color.Beige; columnHeaderStyle.Font = new Font("Verdana", 9, FontStyle.Bold); DataGridView1.ColumnHeadersDefaultCellStyle = columnHeaderStyle; // 設定 DataGridView 控件的數據行數目。 DataGridView1.ColumnCount = 3; // 設定各數據行的標題名稱。 DataGridView1.Columns[0].Name = "類別編號"; DataGridView1.Columns[1].Name = "類別名稱"; DataGridView1.Columns[2].Name = "說明"; DataGridView1.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.AllCells; DataGridView2.ColumnHeadersVisible = true; DataGridView2.ColumnHeadersDefaultCellStyle = columnHeaderStyle; // 設定 DataGridView 控件的數據行數目。 DataGridView2.ColumnCount = 4; // 設定各數據行的標題名稱。 DataGridView2.Columns[0].Name = "產品編號"; DataGridView2.Columns[1].Name = "產品名稱"; DataGridView2.Columns[2].Name = "單位數量"; DataGridView2.Columns[3].Name = "單價"; DataGridView2.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.AllCells; DataGridView3.ColumnHeadersVisible = true; DataGridView3.ColumnHeadersDefaultCellStyle = columnHeaderStyle; // 設定 DataGridView 控件的數據行數目。 DataGridView3.ColumnCount = 3; // 設定各數據行的標題名稱。 DataGridView3.Columns[0].Name = "貨運公司編號"; DataGridView3.Columns[1].Name = "貨運公司名稱"; DataGridView3.Columns[2].Name = "電話"; DataGridView3.AutoSizeColumnsMode = DataGridViewAutoSizeColumnsMode.AllCells; } private void btnParseTextFiles_Click(object sender, EventArgs e) { using (TextFieldParser myReader =new TextFieldParser(@"Text多格式文本文件.txt")) { // 定義三種格式之各欄的寬度與分隔字符。 int[] FirstFormat = { 5, 10, -1 }; int[] SecondFormat = { 6, 10, 17, -1 }; string[] ThirdFormat = { "," }; this.DataGridView1.Rows.Clear(); this.DataGridView2.Rows.Clear(); this.DataGridView3.Rows.Clear(); string[] CurrentRow; while (!myReader.EndOfData) { try { string RowType = myReader.PeekChars(2); switch (RowType) { case "CK": myReader.TextFieldType = FieldType.FixedWidth; myReader.FieldWidths = FirstFormat; // 或是myReader.SetFieldWidths(FirstFormat); CurrentRow = myReader.ReadFields(); this.DataGridView1.Rows.Add(CurrentRow); break; case "PB": myReader.TextFieldType = FieldType.FixedWidth; myReader.FieldWidths = SecondFormat; // 或是myReader.SetFieldWidths(SecondFormat); CurrentRow = myReader.ReadFields(); this.DataGridView2.Rows.Add(CurrentRow); break; case "SP": myReader.TextFieldType = FieldType.Delimited; myReader.Delimiters = ThirdFormat; // 或是myReader.SetDelimiters(ThirdFormat); CurrentRow = myReader.ReadFields(); this.DataGridView3.Rows.Add(CurrentRow); break; } } catch (MalformedLineException ex) { MessageBox.Show("行" + ex.Message + " 是無效的。略過。"); } } // 排序各個 DataGridView 控件的內容。 DataGridView1.Sort(DataGridView1.Columns[0], System.ComponentModel.ListSortDirection.Ascending); DataGridView2.Sort(DataGridView2.Columns[0], System.ComponentModel.ListSortDirection.Ascending); DataGridView3.Sort(DataGridView3.Columns[0], System.ComponentModel.ListSortDirection.Ascending); } }
請注意:
在使用Visual Basic的My對象(或是說My快捷方式)之前,必須先在項目中添加對Microsoft.VisualBasic.dll的引用,然後如下所示導入適當的命名空間:
using Microsoft.VisualBasic.Devices;
如此一來,就可以在Visual C# 中使用與My相似的語法來撰寫程序。