C # разбор текстовых данных построчно - PullRequest
1 голос
/ 14 апреля 2011

У меня есть проблема с моим проектом, позвольте мне объяснить здесь немного, и мне нужно некоторое экспертное руководство здесь, потому что я новичок в программировании.У меня есть данные в блокноте, например:

10192 20351 30473 40499 50449 60234    
10192 20207 30206 40203 50205 60226    
10192 20252 30312 40376 50334 60252

это 26 строк данных, но я показываю только 3 строки только для примера.Вот некоторые правила сортировки по приоритету:

-Я хочу только прочитать текстовый файл, а затем извлечь номер.Пример: 10192 20351. и т. Д.

-У меня есть 6 столбцов ListView, и я хочу отобразить каждую строку числа в своем столбце

Столбец 1 |Колонка 2 |Колонка 3 | Колонка 4 |Колонка 5 |Столбец 6

10192   | 20351   | 30473  | 40499  |50449  | 60234

- конечно, если возможно, первые 2 цифры каждых 5 цифр числа - это уникальный код, все, что я хочу, это всего лишь 3 последние цифры.Например: 192 351 473 499 234. так что для каждого числа будет мод на 10.000.

Полагаю, я вас многих смущаю, извините, вот мой текущий код

закрытый делегат void UpdateUiTextDelegate (String Text);private void serial_DataRectained (отправитель объекта, System.IO.Ports.SerialDataReceivedEventArgs e) {// Сбор символов, полученных в наш «буфер» (строку).строка полученный_данные;receive_data = serial.ReadExisting ();Dispatcher.Invoke (DispatcherPriority.Send, новый UpdateUiTextDelegate (WriteData), receive_data);}

    private void WriteData(String Text)
    {
        if (bufferData != "" || Text[0] == '1')
            bufferData += Text;
        if (bufferData.Length >= 35)
        {
            using (System.IO.StreamWriter file = new System.IO.StreamWriter(@"C:\Users\Rads\Desktop\Training06.txt", true))
            {
                file.WriteLine(bufferData);
            }
            listBox1.Items.Add(bufferData);
            bufferData = "";
        }
    }
    #endregion

    private void Window_Loaded(object sender, RoutedEventArgs e)
    {

    }

    //Browse .txt file
    private void Browse_btn_Click(object sender, RoutedEventArgs e)
    {
        Microsoft.Win32.OpenFileDialog dlg = new Microsoft.Win32.OpenFileDialog();

        dlg.DefaultExt = ".txt";
        dlg.Filter = "Text document (.txt)|*.txt";

        Nullable<bool> result = dlg.ShowDialog();

        if (result == true)
        {
            string filename = dlg.FileName;
            textBox.Text = filename;
        }
    }

    private void Parsing_String(string filename)
    {
        List<Row> list = new List<Row>();

        foreach (String str in File.ReadLines(filename))
        {
            String[] strCols = str.Split(Convert.ToChar(" "));
            list.Add(new Row()
            {
                Column1 = strCols[0].Substring(2),
                Column2 = strCols[1].Substring(2),
                Column3 = strCols[2].Substring(2),
                Column4 = strCols[3].Substring(2),
                Column5 = strCols[4].Substring(2),
                Column6 = strCols[5].Substring(2),


            });
        }

        dg.ItemsSource = list;
    }

    public class Row
    {
        public string Column1 { get; set; }
        public string Column2 { get; set; }
        public string Column3 { get; set; }
        public string Column4 { get; set; }
        public string Column5 { get; set; }
        public string Column6 { get; set; }

    }

Код XAML

<Window x:Class="SamplingData.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="MainWindow" Height="368" Width="401" Loaded="Window_Loaded">

<TabControl Height="332" HorizontalAlignment="Left" Name="tabControl1" VerticalAlignment="Top" Width="380">
    <TabItem Header="Sampling" Name="Sampling">
        <Grid>
            <Label Content="DATA RECEIVED" Height="28" HorizontalAlignment="Left" Margin="6,6,0,0" Name="label1" VerticalAlignment="Top" />
            <Button Content="Connect" Height="23" HorizontalAlignment="Left" Margin="264,6,0,0" Name="ConnectButton" VerticalAlignment="Top" Width="75" Click="Connect_Comms" />
            <ListBox Height="222" HorizontalAlignment="Left" Margin="10,37,0,0" Name="listBox1" VerticalAlignment="Top" Width="329" />
        </Grid>
    </TabItem>
    <TabItem Header="Training" Name="tabItem1">
        <Grid>
            <Button Content="Training" Height="23" HorizontalAlignment="Left" Margin="243,28,0,0" Name="Train_Btn" VerticalAlignment="Top" Width="75" />
            <Button Content="Browse" Height="23" HorizontalAlignment="Left" Margin="243,6,0,0" Name="Browse_btn" VerticalAlignment="Top" Width="75" Click="Browse_btn_Click" />
            <TextBox Height="23" HorizontalAlignment="Left" Margin="6,7,0,0" Name="textBox" VerticalAlignment="Top" Width="231" Background="{x:Null}"></TextBox>
            <RadioButton Content="RadioButton" Height="16" HorizontalAlignment="Left" Margin="507,76,0,0" Name="radioButton1" VerticalAlignment="Top" />
            <DataGrid x:Name="dg" AutoGenerateColumns="False" Margin="0,57,0,0" DataContext="{Binding}">
                <DataGrid.Columns>
                    <DataGridTextColumn Binding="{Binding Column1}" Header="Column 1"></DataGridTextColumn>
                    <DataGridTextColumn Binding="{Binding Column2}" Header="Column 2"></DataGridTextColumn>
                    <DataGridTextColumn Binding="{Binding Column3}" Header="Column 3"></DataGridTextColumn>
                    <DataGridTextColumn Binding="{Binding Column4}" Header="Column 4"></DataGridTextColumn>
                    <DataGridTextColumn Binding="{Binding Column5}" Header="Column 5"></DataGridTextColumn>
                    <DataGridTextColumn Binding="{Binding Column6}" Header="Column 6"></DataGridTextColumn>
                </DataGrid.Columns>
            </DataGrid>
        </Grid>
    </TabItem>
</TabControl>

Большое спасибо, если кто-то может мне помочь ..

Ответы [ 6 ]

2 голосов
/ 14 апреля 2011

Я думаю, вам совсем не нужно регулярное выражение. Вам просто нужно string.Split по строке и пробелу.

string[] lines = data.Split(Enviroment.NewLine);

Для каждой строки вы можете получить поля, разделив строку на пробел.

string[] fields = line.Split(' ');
1 голос
/ 14 апреля 2011

Хорошо, это, вероятно, излишне, но вы упомянули, что хотите поместить данные в представление списка (возможно, в сетку данных?), И в этом случае вы можете захотеть получить данные в какой-либо форме объекта.Это действительно зависит от того, что вы собираетесь делать с данными после их получения.

Работа в предположении, что, получив данные, вы захотите манипулировать ими или что-то делать.Более того, попробуйте что-то вроде этого - вы сможете вставить это прямо в новое консольное приложение и запустить его.

namespace ConsoleApplication6
{
    using System;
    using System.Collections.Generic;
    using System.IO;
    using System.Text.RegularExpressions;

    class Program
    {
        static void Main(string[] args)
        {
            string filename = @"c:\test.txt";

            // Because you're working with a small file, we'll just read all the lines into memory
            List<LineData> processedLines = new List<LineData>();
            foreach (var line in File.ReadAllLines(filename))
            {
                processedLines.Add(new LineData(line));
            }

            // Write out the line data to the console to prove that it has been read
            foreach (var processedLine in processedLines)
            {
                Console.WriteLine(
                    "{0},{1},{2},{3},{4},{5}", 
                    processedLine.Column1, 
                    processedLine.Column2,
                    processedLine.Column3,
                    processedLine.Column4,
                    processedLine.Column5,
                    processedLine.Column6);
            }
        }
    }

    public class LineData
    {
        public LineData(string line)
        {
            // Regex basically means find two digits ("Prefix") followed by 3 digits ("Value")
            Regex regex = new Regex(@"(?<Prefix>\d{2})(?<Value>\d{3})");
            var lineMatches = regex.Matches(line);
            if (lineMatches.Count != 6)
            {
                // You should really be throwing your own exception type...
                throw new Exception("Expected 6 columns!");
            }

            this.Column1 = this.ExtractMatchData(lineMatches[0]);
            this.Column2 = this.ExtractMatchData(lineMatches[1]);
            this.Column3 = this.ExtractMatchData(lineMatches[2]);
            this.Column4 = this.ExtractMatchData(lineMatches[3]);
            this.Column5 = this.ExtractMatchData(lineMatches[4]);
            this.Column6 = this.ExtractMatchData(lineMatches[5]);
        }

        private string ExtractMatchData(Match match)
        {
            return match.Groups["Value"].Value;
        }

        public string Column1 { get; set; }
        public string Column2 { get; set; }
        public string Column3 { get; set; }
        public string Column4 { get; set; }
        public string Column5 { get; set; }
        public string Column6 { get; set; }
    }
}
1 голос
/ 14 апреля 2011

Что-то вроде этого возможно:

var result = from row in theFileAsString.Split('\n')
             select new {
                Columns = row.Split(' ').Select(s => s.Substring(2))
             }

Там у вас будет IEnumerable с элементами, каждый из которых имеет свойство Columns, содержащее строки с данными.

Очень непроверенный, но вы поняли идею.

1 голос
/ 14 апреля 2011

РЕДАКТИРОВАТЬ: ДОБАВЛЕНО переменная строки, как предложено Ash:

Учитывая, что вы не указали, к какому объекту вы собираетесь записывать данные, я предполагаю, что это datarow с именем Row:

private void Parsing_String(string filename)    
{
    DataTable dt = CreateDataTable();
    foreach (String str in File.ReadLines(filename))
    {
      String[] strCols = str.Split(Convert.ToChar(" "));
      DataRow Row = dt.NewRow(); //Where dt is a DataTable
      for (int i =0; i < strCols.length; i++)
      {
           Row[i] = strCols[i].Substring(2); //This will start reading from the third character
      }
      dt.Rows.Add(Row);
     }
      listView1.ItemsSource = dt.Rows;
}

//**EDIT**: Just in case you don't have a datatable and you want to create a small one:

public DataTable CreateDataTable()
    {
        DataTable dt = new DataTable();

        new string[] { "Column 1", "Column 2", "Column 3", "Column 4", "Column 5", "Column 6" }
            .ToList()
            .ForEach(c => { dt.Columns.Add(new DataColumn(c)); });
        return dt;
    }

РЕДАКТИРОВАТЬ: Последняя попытка (игнорировать код выше):

    private void Parsing_String(string filename)
    {
        List<Row> list = new List<Row>();

        foreach (String str in File.ReadLines(filename))
        {
            String[] strCols = str.Split(Convert.ToChar(" "));
            list.Add(new Row() 
            {
                Column1 = strCols[0].Substring(2),
                Column2 = strCols[1].Substring(2),
                Column3 = strCols[2].Substring(2),
                Column4 = strCols[3].Substring(2),
                Column5 = strCols[4].Substring(2),
                Column6 = strCols[5].Substring(2)
            });
        }

        dg.ItemsSource = list;
    }

    public class Row
    {
        public string Column1 { get; set; }
        public string Column2 { get; set; }
        public string Column3 { get; set; }
        public string Column4 { get; set; }
        public string Column5 { get; set; }
        public string Column6 { get; set; }
    }

А потом в вашем xaml:

Если вы хотите указать пользовательские заголовки, вам придется изменить его, чтобы он не генерировал столбцы автоматически, а использовал привязки, т. Е .:

   <DataGrid x:Name="dg" AutoGenerateColumns="False">
        <DataGrid.Columns>
            <DataGridTextColumn Binding="{Binding Column1}" Header="Column 1"></DataGridTextColumn>
            <DataGridTextColumn Binding="{Binding Column2}" Header="Column 2"></DataGridTextColumn>
            <DataGridTextColumn Binding="{Binding Column3}" Header="Column 3"></DataGridTextColumn>
            <DataGridTextColumn Binding="{Binding Column4}" Header="Column 4"></DataGridTextColumn>
            <DataGridTextColumn Binding="{Binding Column5}" Header="Column 5"></DataGridTextColumn>
            <DataGridTextColumn Binding="{Binding Column6}" Header="Column 6"></DataGridTextColumn>
        </DataGrid.Columns>
    </DataGrid>

Редактировать Это вычислит итоги первого столбца

protected int CalculateFirstColumnTotal(List<Row> list)
{
    int total = 0;
    foreach (Row row in list)
      total += int.Parse(row.Column1);
}

EDIT :

На самом деле вы никогда не вызываете метод Parsing_String, добавьте следующую строку в метод обзора:

private void Browse_btn_Click(object sender, RoutedEventArgs e)    
{
    //Existing Code
    Parsing_String(textBox.Text);  //Add this line to the last line of the method.
}
0 голосов
/ 14 апреля 2011

Вам не нужно использовать регулярное выражение для этого.

var data = @"10192 20351 30473 40499 50449 60234
10192 20207 30206 40203 50205 60226
10192 20252 30312 40376 50334 60252";

var result = from line in data.Split(new[] { "\r\n", "\n" }, StringSplitOptions.RemoveEmptyEntries)
             let splitted = line.Split(new[] {' '}, StringSplitOptions.RemoveEmptyEntries)
             select splitted.Select(s => s.Substring(2));

или для файлов

using System.IO; // In the top

var result = from line in File.ReadLines("path")
             let splitted = line.Split(new[] {' '}, StringSplitOptions.RemoveEmptyEntries)
             select splitted.Select(s => s.Substring(2));

result теперь будет содержать последовательность последовательностей строк (гдепервые два символа удалены).Эта версия работает как с Unix, так и с Windows.Также удаляются дополнительные пробелы, которые могут привести к сбою других ответов.

0 голосов
/ 14 апреля 2011
using (var stream = File.Open(this.filename, FileMode.Open, FileAccess.Read)
{
    var reader = new StreamReader(stream);
    var data = reader.ReadLine();
    while (!String.IsNullOrWhitespace(data))
    {
        string[] columns = data.Split(' ');
        Console.WriteLine(string.Format("{0} {1} {2} {3} {4} {5}", columns[0], columns[1],));
        data = reader.ReadLine();
    }
}
...