web-dev-qa-db-ja.com

C#コンソールはパイプで入力を受け取ります

パラメータを使用してコンソールアプリケーションをプログラムする方法を知っています(例:myProgram.exe param1 param2)。

私の質問は、どのようにプログラムを|で動作させることができるかです、例:echo "Word" | myProgram.exe?

40

ユーザー入力を読み取る場合と同様に、Console.Read()およびConsole.ReadLine()を使用する必要があります。パイプはユーザー入力を透過的に置き換えます。両方を簡単に使用することはできません(確かに可能ですが...)。

編集:

シンプルなcatスタイルのプログラム:

class Program
{
    static void Main(string[] args)
    {
        string s;
        while ((s = Console.ReadLine()) != null)
        {
            Console.WriteLine(s);
        }

    }
}

実行すると、期待どおりに、出力:

 C:\ ...\ConsoleApplication1\bin\Debug> echo "Foo bar baz" | ConsoleApplication1.exe 
 "Foo bar baz" 
 
 C:\ ...\ConsoleApplication1\bin\Debug> 
44

以下は、入力のためにアプリケーションを中断せず、データがorがパイプされていない場合に機能します。少しハック;また、エラーキャッチにより、パイプで多数の呼び出しが行われるとパフォーマンスが低下する可能性がありますが、簡単です。

public static void Main(String[] args)
{

    String pipedText = "";
    bool isKeyAvailable;

    try
    {
        isKeyAvailable = System.Console.KeyAvailable;
    }
    catch (InvalidOperationException expected)
    {
        pipedText = System.Console.In.ReadToEnd();
    }

    //do something with pipedText or the args
}
16
CodeMiller

.NET 4.5では

if (Console.IsInputRedirected)
{
    using(stream s = Console.OpenStandardInput())
    {
        ...
12
gordy

これはそれを行う方法です:

static void Main(string[] args)
{
    Console.SetIn(new StreamReader(Console.OpenStandardInput(8192))); // This will allow input >256 chars
    while (Console.In.Peek() != -1)
    {
        string input = Console.In.ReadLine();
        Console.WriteLine("Data read was " + input);
    }
}

これにより、2つの使用方法が可能になります。読み取り標準入力

C:\test>myProgram.exe
hello
Data read was hello

またはパイプ入力から読み取る:

C:\test>echo hello | myProgram.exe
Data read was hello
8
matt burns

以下は、他のソリューションとpeek()を組み合わせた別の代替ソリューションです。

Peek()がないと、「type t.txt | prog.exe」を実行すると、末尾にctrl-cがないとアプリが返らないことがありました。t.txtは複数行のファイルです。しかし、「prog.exe」または「echo hi | prog.exe」だけで問題なく動作しました。

このコードは、パイプされた入力のみを処理するためのものです。

static int Main(string[] args)
{
    // if nothing is being piped in, then exit
    if (!IsPipedInput())
        return 0;

    while (Console.In.Peek() != -1)
    {
        string input = Console.In.ReadLine();
        Console.WriteLine(input);
    }

    return 0;
}

private static bool IsPipedInput()
{
    try
    {
        bool isKey = Console.KeyAvailable;
        return false;
    }
    catch
    {
        return true;
    }
}
3

これも機能します

c:\ MyApp.exe <input.txt

Stdinから取得した入力を操作するには、StringBuilderを使用する必要がありました。

public static void Main()
{
    List<string> salesLines = new List<string>();
    Console.InputEncoding = Encoding.UTF8;
    using (StreamReader reader = new StreamReader(Console.OpenStandardInput(), Console.InputEncoding))
    {
        string stdin;
        do
        {
            StringBuilder stdinBuilder = new StringBuilder();
            stdin = reader.ReadLine();
            stdinBuilder.Append(stdin);
            var lineIn = stdin;
            if (stdinBuilder.ToString().Trim() != "")
            {
                salesLines.Add(stdinBuilder.ToString().Trim());
            }

        } while (stdin != null);

    }
}
2
Si Zi

Console.Inは、標準入力ストリームをラップしたTextReaderへの参照です。大量のデータをプログラムにパイプする場合、その方法で作業する方が簡単な場合があります。

2
Joel Mueller

提供されている例に問題があります。

  while ((s = Console.ReadLine()) != null)

パイプされたデータなしでプログラムが起動された場合、入力待ちでスタックします。そのため、ユーザーはプログラムを終了するために手動で任意のキーを押す必要があります。

1
Alex N