প্রসেস.স্টার্টের কোনও অ্যাসিঙ্ক সমতুল্য কি আছে?


141

শিরোনামের পরামর্শ অনুসারে, Process.Startআমি অপেক্ষা করতে পারি এমন সমতুল্য (আপনাকে অন্য অ্যাপ্লিকেশন বা ব্যাচ ফাইল চালানোর অনুমতি দেয়)?

আমি একটি ছোট কনসোল অ্যাপ্লিকেশন নিয়ে খেলছি এবং এটি async ব্যবহার করার জন্য অপেক্ষা করার উপযুক্ত জায়গা বলে মনে হচ্ছে এবং অপেক্ষা করার জন্য আমি এই দৃশ্যের জন্য কোনও ডকুমেন্টেশন পাই না।

আমি যা ভাবছি তা এই লাইনের সাথে কিছু হয়:

void async RunCommand()
{
    var result = await Process.RunAsync("command to run");
}

2
আপনি কেন ফিরে আসা প্রক্রিয়া অবজেক্টটিতে অপেক্ষা করুন?
সিম্পলবার

2
এবং যাইহোক, এমন আরও শোনা যাচ্ছে যে আপনি "অ্যাসিঙ্ক" সমাধানের পরিবর্তে একটি "সিঙ্কড" সমাধান খুঁজছেন, সুতরাং শিরোনাম বিভ্রান্ত করছে।
সিম্পলবার

2
নিবন্ধন করুন নিশ্চয় Process.Start হয় ASYNC এবং ও.পি. একটি সমকালীন সংস্করণ চান বলে মনে হচ্ছে।
ওডে

10
ওপি নতুন #
অ্যাসিঙ্ক

4
ঠিক আছে, আমি আমার পোস্টটি আরও কিছুটা পরিষ্কার হওয়ার জন্য আপডেট করেছি। আমি কেন এটি চাই তার ব্যাখ্যাটি সহজ। এমন একটি দৃশ্যের চিত্র দিন যেখানে আপনাকে বাহ্যিক কমান্ড চালাতে হবে (7zip এর মতো কিছু) এবং তারপরে অ্যাপ্লিকেশনটির প্রবাহ চালিয়ে যেতে হবে। এটি হ'ল async / প্রত্যাশাটি সুবিধার্থে বোঝানো হয়েছিল এবং এখনও কোনও প্রক্রিয়া চালানোর এবং এটির প্রস্থানের অপেক্ষার কোনও উপায় নেই বলে মনে হচ্ছে।
লিঙ্কেরো

উত্তর:


196

Process.Start()কেবল প্রক্রিয়া শুরু করে, এটি শেষ না হওয়া পর্যন্ত অপেক্ষা করে না, সুতরাং এটি তৈরি করার পক্ষে এটি তেমন অর্থবোধ করে না async। আপনি যদি এখনও এটি করতে চান তবে আপনি এমন কিছু করতে পারেন await Task.Run(() => Process.Start(fileName))

কিন্তু, যদি আপনি অ্যাসিঙ্ক্রোনাস শেষ প্রক্রিয়ার জন্য অপেক্ষা করতে চান, আপনি ব্যবহার করতে পারেন ঘটনা একসঙ্গে সঙ্গে :ExitedTaskCompletionSource

static Task<int> RunProcessAsync(string fileName)
{
    var tcs = new TaskCompletionSource<int>();

    var process = new Process
    {
        StartInfo = { FileName = fileName },
        EnableRaisingEvents = true
    };

    process.Exited += (sender, args) =>
    {
        tcs.SetResult(process.ExitCode);
        process.Dispose();
    };

    process.Start();

    return tcs.Task;
}

36
অবশেষে আমি এটির জন্য গিথুবকে কিছু আঁকিয়ে ধরতে পেরেছি - এটির কোনও বাতিলকরণ / সময়সীমা সমর্থন নেই, তবে এটি আপনার জন্য স্ট্যান্ডার্ড আউটপুট এবং স্ট্যান্ডার্ড ত্রুটিটি সংগ্রহ করবে least github.com/jamesmanning/RunProcessAsTask
জেমস ম্যানিং

3
এই কার্যকারিতাটি মেডেলিয়নশেল
নুগেট

8
সত্যই গুরুত্বপূর্ণ: আপনি যে ক্রমটিতে বিভিন্ন বৈশিষ্ট্য সেট করেছেন processএবং process.StartInfoক্রমটি যখন এটি চালানো হয় তখন কী পরিবর্তন হয় .Start()। আপনি যদি এখানে উদাহরণস্বরূপ বৈশিষ্ট্য .EnableRaisingEvents = trueসেট করার আগে কল করেন তবে StartInfoজিনিসগুলি প্রত্যাশার মতো কাজ করে। আপনি যদি এটি পরে সেট করেন, উদাহরণস্বরূপ এটিকে একত্রে রাখার জন্য .Exited, যদিও আপনি আগে এটি কল করেছিলেন .Start(), এটি সঠিকভাবে কাজ করতে ব্যর্থ হয় - .Exitedপ্রক্রিয়াটি প্রকৃতপক্ষে প্রস্থান হওয়ার অপেক্ষা না করে অবিলম্বে অগ্নিকাণ্ডগুলি। কেন জানি না, শুধু সাবধানতার কথা।
ক্রিস মোসচিনি

2
@svick উইন্ডো ফর্মটিতে, process.SynchronizingObjectইভেন্টগুলি হ্যান্ডেল করার পদ্ধতিগুলি (যেমন প্রস্থানিত, আউটপুট ডেটা রিসিভড, ত্রুটি-ডেটা রিসিভড) পৃথক থ্রেডে কল করা হয় সেগুলি এড়াতে ফর্মগুলির উপাদানগুলিতে সেট করা উচিত।
কেভিনবুউই

4
এটি গুটিয়ে রাখা আসলেই অর্থবোধ করে Process.Startনা Task.Run। উদাহরণস্বরূপ, ইউএনসির একটি পথ সিঙ্ক্রোনালি সমাধান করা হবে। এই স্নিপেটটি সম্পূর্ণ হতে 30 সেকেন্ড পর্যন্ত সময় নিতে পারে:Process.Start(@"\\live.sysinternals.com\whatever")
জাবি

55

এসভিকের উত্তরের উপর ভিত্তি করে আমার নেওয়া এখানে । এটি আউটপুট পুনঃনির্দেশ, প্রস্থান কোড রক্ষণাবেক্ষণ এবং সামান্য ভাল ত্রুটি পরিচালনার যোগ করে ( Processবস্তুটি এটি শুরু না করা সত্ত্বেও নিষ্পত্তি করা):

public static async Task<int> RunProcessAsync(string fileName, string args)
{
    using (var process = new Process
    {
        StartInfo =
        {
            FileName = fileName, Arguments = args,
            UseShellExecute = false, CreateNoWindow = true,
            RedirectStandardOutput = true, RedirectStandardError = true
        },
        EnableRaisingEvents = true
    })
    {
        return await RunProcessAsync(process).ConfigureAwait(false);
    }
}    
private static Task<int> RunProcessAsync(Process process)
{
    var tcs = new TaskCompletionSource<int>();

    process.Exited += (s, ea) => tcs.SetResult(process.ExitCode);
    process.OutputDataReceived += (s, ea) => Console.WriteLine(ea.Data);
    process.ErrorDataReceived += (s, ea) => Console.WriteLine("ERR: " + ea.Data);

    bool started = process.Start();
    if (!started)
    {
        //you may allow for the process to be re-used (started = false) 
        //but I'm not sure about the guarantees of the Exited event in such a case
        throw new InvalidOperationException("Could not start process: " + process);
    }

    process.BeginOutputReadLine();
    process.BeginErrorReadLine();

    return tcs.Task;
}

1
সবেমাত্র এই আকর্ষণীয় সমাধানটি খুঁজে পেয়েছি। আমি সি তে নতুন হিসাবে আমি কীভাবে এটি ব্যবহার করব তা নিশ্চিত নই async Task<int> RunProcessAsync(string fileName, string args)। আমি এই উদাহরণটি মানিয়ে নিয়েছি এবং একে একে তিনটি বস্তু পাস করেছি। আমি কীভাবে ঘটনা উত্থাপনের জন্য অপেক্ষা করতে পারি? যেমন। আমার আবেদন থামার আগে .. অনেক অনেক ধন্যবাদ
14

3
@ মার্সারশাইন আপনার অর্থটি ঠিক কী তা আমি বুঝতে পারছি না, সম্ভবত আপনার কোনও কোড দিয়ে একটি নতুন প্রশ্ন শুরু করা উচিত যাতে আপনি কী চেষ্টা করেছেন তা আমরা দেখতে পাই এবং সেখান থেকে চালিয়ে যেতে পারি।
ওহাদ স্নাইডার

4
চমত্কার উত্তর। ভিত্তি স্থাপনের জন্য আপনাকে ধন্যবাদ এবং এই অতি কার্যকর প্রসারের জন্য ওহাদকে ধন্যবাদ জানাই।
গর্ডন বিন

1
@ সুপারজেএমএন কোডটি পড়ছেন ( রেফারেন্সসোর্স.মাইক্রোসফট / # সিস্টেমেট / সার্ভিসেস / মনিটরিং / ৮ ) আমি Disposeইভেন্ট হ্যান্ডলারটিকে নালাগুলি বিশ্বাস করি না , সুতরাং তাত্ত্বিকভাবে আপনি যদি ফোন করেছেন Disposeতবে রেফারেন্সটিকে চারপাশে রাখেন তবে আমি বিশ্বাস করি এটি ফাঁস হয়ে যাবে। যাইহোক, যখন Processঅবজেক্টটির আরও কোনও উল্লেখ নেই এবং এটি সংগ্রহ করা (আবর্জনা) সংগ্রহ করা হয়, ইভেন্ট হ্যান্ডলার তালিকার দিকে ইঙ্গিত করে এমন কেউ নেই। সুতরাং এটি সংগ্রহ করা হয়, এবং এখন সেই প্রতিনিধিদের কোনও উল্লেখ নেই যা তালিকায় আগে ছিল, সুতরাং অবশেষে তারা আবর্জনা সংগ্রহ করে।
ওহাদ স্নাইডার

1
@ সুপারজেএমএন: মজার বিষয় হল এটি এর চেয়ে জটিল / শক্তিশালী। একটির জন্য, Disposeকিছু সংস্থান পরিষ্কার করে তবে কোনও ফাঁস হওয়া রেফারেন্সটিকে processচারপাশে রাখার থেকে বিরত রাখে না । প্রকৃতপক্ষে, আপনি খেয়াল processকরবেন যা হ্যান্ডলারগুলিকে বোঝায় তবে Exitedহ্যান্ডলারের একটি উল্লেখও রয়েছে process। কিছু সিস্টেমে, সেই বিজ্ঞপ্তিযুক্ত রেফারেন্সটি আবর্জনা সংগ্রহকে আটকাতে পারে, তবে .NET এ ব্যবহৃত অ্যালগরিদমটি এখনও এতক্ষণ পরিষ্কার করে দেবে যতক্ষণ না কোনও "দ্বীপে" বাইরের রেফারেন্স সহ সমস্ত কিছু বেঁচে থাকে।
TheRubberDuck

4

এখানে অন্য পদ্ধতি। অনুরূপ ধারণা svick এবং ওহদ এর উত্তর কিন্তু থাকা একটি এক্সটেনশন পদ্ধতি ব্যবহার করে Processপ্রকার।

সম্প্রসারণ পদ্ধতি:

public static Task RunAsync(this Process process)
{
    var tcs = new TaskCompletionSource<object>();
    process.EnableRaisingEvents = true;
    process.Exited += (s, e) => tcs.TrySetResult(null);
    // not sure on best way to handle false being returned
    if (!process.Start()) tcs.SetException(new Exception("Failed to start process."));
    return tcs.Task;
}

একটি সমন্বিত পদ্ধতিতে ব্যবহারের ক্ষেত্রে উদাহরণ:

public async Task ExecuteAsync(string executablePath)
{
    using (var process = new Process())
    {
        // configure process
        process.StartInfo.FileName = executablePath;
        process.StartInfo.UseShellExecute = false;
        process.StartInfo.CreateNoWindow = true;
        // run process asynchronously
        await process.RunAsync();
        // do stuff with results
        Console.WriteLine($"Process finished running at {process.ExitTime} with exit code {process.ExitCode}");
    };// dispose process
}

4

আমি একটি প্রক্রিয়া শুরু করার জন্য একটি ক্লাস তৈরি করেছি এবং এটি বিভিন্ন প্রয়োজনীয়তার কারণে গত বছরগুলিতে বেড়ে চলেছে। ব্যবহারের সময় আমি প্রসেস ক্লাসের সাথে এক্সপোস্টকোড নিষ্পত্তি এবং এমনকি পড়ার সাথে বেশ কয়েকটি সমস্যা খুঁজে পেয়েছি। সুতরাং এই সব আমার ক্লাস দ্বারা স্থির।

শ্রেণীর বিভিন্ন সম্ভাবনা রয়েছে, উদাহরণস্বরূপ পাঠ্য আউটপুট, অ্যাডমিন বা বিভিন্ন ব্যবহারকারী হিসাবে শুরু করুন, ব্যতিক্রমগুলি ধরুন এবং এই সমস্ত অ্যাসিনক্রোনাস ইনক্লিকেশনও শুরু করুন। বাতিলকরণ। দুর্দান্ত যে এক্সিকিউট করার সময় পঠন আউটপুটও সম্ভব।

public class ProcessSettings
{
    public string FileName { get; set; }
    public string Arguments { get; set; } = "";
    public string WorkingDirectory { get; set; } = "";
    public string InputText { get; set; } = null;
    public int Timeout_milliseconds { get; set; } = -1;
    public bool ReadOutput { get; set; }
    public bool ShowWindow { get; set; }
    public bool KeepWindowOpen { get; set; }
    public bool StartAsAdministrator { get; set; }
    public string StartAsUsername { get; set; }
    public string StartAsUsername_Password { get; set; }
    public string StartAsUsername_Domain { get; set; }
    public bool DontReadExitCode { get; set; }
    public bool ThrowExceptions { get; set; }
    public CancellationToken CancellationToken { get; set; }
}

public class ProcessOutputReader   // Optional, to get the output while executing instead only as result at the end
{
    public event TextEventHandler OutputChanged;
    public event TextEventHandler OutputErrorChanged;
    public void UpdateOutput(string text)
    {
        OutputChanged?.Invoke(this, new TextEventArgs(text));
    }
    public void UpdateOutputError(string text)
    {
        OutputErrorChanged?.Invoke(this, new TextEventArgs(text));
    }
    public delegate void TextEventHandler(object sender, TextEventArgs e);
    public class TextEventArgs : EventArgs
    {
        public string Text { get; }
        public TextEventArgs(string text) { Text = text; }
    }
}

public class ProcessResult
{
    public string Output { get; set; }
    public string OutputError { get; set; }
    public int ExitCode { get; set; }
    public bool WasCancelled { get; set; }
    public bool WasSuccessful { get; set; }
}

public class ProcessStarter
{
    public ProcessResult Execute(ProcessSettings settings, ProcessOutputReader outputReader = null)
    {
        return Task.Run(() => ExecuteAsync(settings, outputReader)).GetAwaiter().GetResult();
    }

    public async Task<ProcessResult> ExecuteAsync(ProcessSettings settings, ProcessOutputReader outputReader = null)
    {
        if (settings.FileName == null) throw new ArgumentNullException(nameof(ProcessSettings.FileName));
        if (settings.Arguments == null) throw new ArgumentNullException(nameof(ProcessSettings.Arguments));

        var cmdSwitches = "/Q " + (settings.KeepWindowOpen ? "/K" : "/C");

        var arguments = $"{cmdSwitches} {settings.FileName} {settings.Arguments}";
        var startInfo = new ProcessStartInfo("cmd", arguments)
        {
            UseShellExecute = false,
            RedirectStandardOutput = settings.ReadOutput,
            RedirectStandardError = settings.ReadOutput,
            RedirectStandardInput = settings.InputText != null,
            CreateNoWindow = !(settings.ShowWindow || settings.KeepWindowOpen),
        };
        if (!string.IsNullOrWhiteSpace(settings.StartAsUsername))
        {
            if (string.IsNullOrWhiteSpace(settings.StartAsUsername_Password))
                throw new ArgumentNullException(nameof(ProcessSettings.StartAsUsername_Password));
            if (string.IsNullOrWhiteSpace(settings.StartAsUsername_Domain))
                throw new ArgumentNullException(nameof(ProcessSettings.StartAsUsername_Domain));
            if (string.IsNullOrWhiteSpace(settings.WorkingDirectory))
                settings.WorkingDirectory = Path.GetPathRoot(Path.GetTempPath());

            startInfo.UserName = settings.StartAsUsername;
            startInfo.PasswordInClearText = settings.StartAsUsername_Password;
            startInfo.Domain = settings.StartAsUsername_Domain;
        }
        var output = new StringBuilder();
        var error = new StringBuilder();
        if (!settings.ReadOutput)
        {
            output.AppendLine($"Enable {nameof(ProcessSettings.ReadOutput)} to get Output");
        }
        if (settings.StartAsAdministrator)
        {
            startInfo.Verb = "runas";
            startInfo.UseShellExecute = true;  // Verb="runas" only possible with ShellExecute=true.
            startInfo.RedirectStandardOutput = startInfo.RedirectStandardError = startInfo.RedirectStandardInput = false;
            output.AppendLine("Output couldn't be read when started as Administrator");
        }
        if (!string.IsNullOrWhiteSpace(settings.WorkingDirectory))
        {
            startInfo.WorkingDirectory = settings.WorkingDirectory;
        }
        var result = new ProcessResult();
        var taskCompletionSourceProcess = new TaskCompletionSource<bool>();

        var process = new Process { StartInfo = startInfo, EnableRaisingEvents = true };
        try
        {
            process.OutputDataReceived += (sender, e) =>
            {
                if (e?.Data != null)
                {
                    output.AppendLine(e.Data);
                    outputReader?.UpdateOutput(e.Data);
                }
            };
            process.ErrorDataReceived += (sender, e) =>
            {
                if (e?.Data != null)
                {
                    error.AppendLine(e.Data);
                    outputReader?.UpdateOutputError(e.Data);
                }
            };
            process.Exited += (sender, e) =>
            {
                try { (sender as Process)?.WaitForExit(); } catch (InvalidOperationException) { }
                taskCompletionSourceProcess.TrySetResult(false);
            };

            var success = false;
            try
            {
                process.Start();
                success = true;
            }
            catch (System.ComponentModel.Win32Exception ex)
            {
                if (ex.NativeErrorCode == 1223)
                {
                    error.AppendLine("AdminRights request Cancelled by User!! " + ex);
                    if (settings.ThrowExceptions) taskCompletionSourceProcess.SetException(ex); else taskCompletionSourceProcess.TrySetResult(false);
                }
                else
                {
                    error.AppendLine("Win32Exception thrown: " + ex);
                    if (settings.ThrowExceptions) taskCompletionSourceProcess.SetException(ex); else taskCompletionSourceProcess.TrySetResult(false);
                }
            }
            catch (Exception ex)
            {
                error.AppendLine("Exception thrown: " + ex);
                if (settings.ThrowExceptions) taskCompletionSourceProcess.SetException(ex); else taskCompletionSourceProcess.TrySetResult(false);
            }
            if (success && startInfo.RedirectStandardOutput)
                process.BeginOutputReadLine();
            if (success && startInfo.RedirectStandardError)
                process.BeginErrorReadLine();
            if (success && startInfo.RedirectStandardInput)
            {
                var writeInputTask = Task.Factory.StartNew(() => WriteInputTask());
            }

            async void WriteInputTask()
            {
                var processRunning = true;
                await Task.Delay(50).ConfigureAwait(false);
                try { processRunning = !process.HasExited; } catch { }
                while (processRunning)
                {
                    if (settings.InputText != null)
                    {
                        try
                        {
                            await process.StandardInput.WriteLineAsync(settings.InputText).ConfigureAwait(false);
                            await process.StandardInput.FlushAsync().ConfigureAwait(false);
                            settings.InputText = null;
                        }
                        catch { }
                    }
                    await Task.Delay(5).ConfigureAwait(false);
                    try { processRunning = !process.HasExited; } catch { processRunning = false; }
                }
            }

            if (success && settings.CancellationToken != default(CancellationToken))
                settings.CancellationToken.Register(() => taskCompletionSourceProcess.TrySetResult(true));
            if (success && settings.Timeout_milliseconds > 0)
                new CancellationTokenSource(settings.Timeout_milliseconds).Token.Register(() => taskCompletionSourceProcess.TrySetResult(true));

            var taskProcess = taskCompletionSourceProcess.Task;
            await taskProcess.ConfigureAwait(false);
            if (taskProcess.Result == true) // process was cancelled by token or timeout
            {
                if (!process.HasExited)
                {
                    result.WasCancelled = true;
                    error.AppendLine("Process was cancelled!");
                    try
                    {
                        process.CloseMainWindow();
                        await Task.Delay(30).ConfigureAwait(false);
                        if (!process.HasExited)
                        {
                            process.Kill();
                        }
                    }
                    catch { }
                }
            }
            result.ExitCode = -1;
            if (!settings.DontReadExitCode)     // Reason: sometimes, like when timeout /t 30 is started, reading the ExitCode is only possible if the timeout expired, even if process.Kill was called before.
            {
                try { result.ExitCode = process.ExitCode; }
                catch { output.AppendLine("Reading ExitCode failed."); }
            }
            process.Close();
        }
        finally { var disposeTask = Task.Factory.StartNew(() => process.Dispose()); }    // start in new Task because disposing sometimes waits until the process is finished, for example while executing following command: ping -n 30 -w 1000 127.0.0.1 > nul
        if (result.ExitCode == -1073741510 && !result.WasCancelled)
        {
            error.AppendLine($"Process exited by user!");
        }
        result.WasSuccessful = !result.WasCancelled && result.ExitCode == 0;
        result.Output = output.ToString();
        result.OutputError = error.ToString();
        return result;
    }
}

1

আমি মনে করি আপনার যা ব্যবহার করা উচিত তা হ'ল:

using System.Diagnostics;
using System.Threading;
using System.Threading.Tasks;

namespace Extensions
{
    public static class ProcessExtensions
    {
        public static async Task<int> WaitForExitAsync(this Process process, CancellationToken cancellationToken = default)
        {
            process = process ?? throw new ArgumentNullException(nameof(process));
            process.EnableRaisingEvents = true;

            var completionSource = new TaskCompletionSource<int>();

            process.Exited += (sender, args) =>
            {
                completionSource.TrySetResult(process.ExitCode);
            };
            if (process.HasExited)
            {
                return process.ExitCode;
            }

            using var registration = cancellationToken.Register(
                () => completionSource.TrySetCanceled(cancellationToken));

            return await completionSource.Task.ConfigureAwait(false);
        }
    }
}

ব্যবহারের উদাহরণ:

public static async Task<int> StartProcessAsync(ProcessStartInfo info, CancellationToken cancellationToken = default)
{
    path = path ?? throw new ArgumentNullException(nameof(path));
    if (!File.Exists(path))
    {
        throw new ArgumentException(@"File is not exists", nameof(path));
    }

    using var process = Process.Start(info);
    if (process == null)
    {
        throw new InvalidOperationException("Process is null");
    }

    try
    {
        return await process.WaitForExitAsync(cancellationToken).ConfigureAwait(false);
    }
    catch (OperationCanceledException)
    {
        process.Kill();

        throw;
    }
}

কোনটি গ্রহণ করার বিন্দু কী CancellationToken, যদি এটি বাতিল করা Killপ্রক্রিয়া করে না ?
থিওডর জৌলিয়াস

CancellationTokenমধ্যে WaitForExitAsyncপদ্ধতি প্রয়োজন হয় কেবল অপেক্ষা বাতিল বা সময়সীমার সেট পাবে। একটি প্রক্রিয়া হত্যা এখানে করা যেতে পারে StartProcessAsync: process `` চেষ্টা করুন process প্রক্রিয়ার জন্য অপেক্ষা করুন aউইটফরেক্সটেক্সিট } ক্যাচ (অপারেশনকেনসেল এক্সেকশন) {প্রক্রিয়া.কিল (); } `` `
কনস্টান্টিন এস

আমার মতামতটি যখন কোনও পদ্ধতি গ্রহণ করে CancellationToken, টোকেন বাতিল করার ফলে অপারেশন বাতিল হওয়ার ফলস্বরূপ হওয়া উচিত, প্রতীক্ষার বাতিল হওয়া নয়। পদ্ধতির কলকারী সাধারণত এটি আশা করে। যদি কলার কেবল অপেক্ষারত বাতিল করতে চান এবং পটভূমিতে অপারেশনটি এখনও চলমান রাখতে চান তবে বাহ্যিকভাবে করা বেশ সহজ ( এখানে একটি এক্সটেনশন পদ্ধতি AsCancelableযা কেবল এটি করছে)।
থিওডর জোলিয়াস

আমি মনে করি যে এই সিদ্ধান্তটি কলার দ্বারা নেওয়া উচিত (বিশেষত এই ক্ষেত্রে, কারণ এই পদ্ধতিটি ওয়েট দিয়ে শুরু হয়, সাধারণভাবে আমি আপনার সাথে একমত হই), যেমন নতুন ব্যবহারের উদাহরণ হিসাবে।
কনস্টান্টিন এস

0

আমি প্রক্রিয়া নিষ্পত্তি সম্পর্কে সত্যই উদ্বিগ্ন, প্রস্থান অ্যাসিঙ্কের জন্য অপেক্ষা করার কী?? এটি আমার প্রস্তাব (পূর্ববর্তী উপর ভিত্তি করে):

public static class ProcessExtensions
{
    public static Task WaitForExitAsync(this Process process)
    {
        var tcs = new TaskCompletionSource<object>();
        process.EnableRaisingEvents = true;
        process.Exited += (s, e) => tcs.TrySetResult(null);
        return process.HasExited ? Task.CompletedTask : tcs.Task;
    }        
}

তারপরে, এটি এর মতো ব্যবহার করুন:

public static async Task<int> ExecAsync(string command, string args)
{
    ProcessStartInfo psi = new ProcessStartInfo();
    psi.FileName = command;
    psi.Arguments = args;

    using (Process proc = Process.Start(psi))
    {
        await proc.WaitForExitAsync();
        return proc.ExitCode;
    }
}
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.