【问题标题】:How to pause a thread while continue others thread in C# UWP如何在 C# UWP 中继续其他线程时暂停线程
【发布时间】:2018-05-02 21:39:35
【问题描述】:

我正在构建一个应用程序,它将连接到 Twitter,接收推文数据并将它们存储到数据库中。之后,应用程序将从数据库中检索数据,分析内容,如果有与另一个预设数据库匹配的内容,则会弹出 toast 通知。

总共将有 5 个任务需要同时运行。

Task 1 => GetKeyword 方法(从数据库中获取关键字列表)

任务 2 => Connect 方法(连接到 Twitter 并通过关键字过滤流数据并存储到数据库中)

任务 3 => RetrieveData 方法(从数据库中检索存储的流数据)

任务 4 => 分析方法(分析推文内容并找到匹配的内容)

Task 5 => Notify 方法(如果匹配,通过弹出 toast 通知通知用户)

所以以前我设法让它们完全运行,但是使用这些代码有时它会永远停留在Connect 任务,并且不会继续执行RetrieveData 任务和Analyze 任务。

如何限制Connect 任务的运行时间并让它在RetrieveData 任务和Analyze 任务上继续运行?我已经限制了存储在方法本身内的流数据的数量,但似乎只停止流本身而不是线程。

这是关于多线程的代码:

List<string> streamdata = new List<string>();
List<string> keyList = new List<string>();
try
{
    var task = Task.Run(() => GetKeyword(0))
               .ContinueWith(prevTask => Connecting(1000, keyList))
               .ContinueWith(prevTask => RetrieveData(1500))
               .ContinueWith(prevTask => MakeRequest(2000, streamdata))
               .ContinueWith(prevTask => Notify(2500, cyberbully, notification));
    task.Wait();

}
catch (Exception ex)
{
    MessageDialog messagebox = new MessageDialog("Task running error:" + ex);
    await messagebox.ShowAsync();
}

这是连接方法代码:

public static void Connecting(int sleepTime, List<string> keyList)
{
    //Set the token that provided by Twitter to gain authorized access into Twitter database
    Auth.SetUserCredentials("YTNuoC9rrJs8g9kZ0hRweKrpp", "wXj6VSl68jeFStRWHDnhG19oP1WZGeBFMNgT3KCkI6MaX46SMT", "892680922322960384-8ka1NuhgiuxjSLUffQVdwmnOIbIduZa", "y92ycGrGCJS9vBJU79gq34rV6FCwNjBPFFOqhEHaTQe1l");

    //Create stream with filter stream type
    var stream = Stream.CreateFilteredStream();
    int numoftweet = 0;
    //Set language filter to English only
    stream.AddTweetLanguageFilter(LanguageFilter.English);
    //Connect to database that stored the keyword
    foreach (var key in keyList)
    {
        stream.AddTrack(key);
    }
    //Let the stream match with all the conditions stated above
    stream.MatchingTweetReceived += async (sender, argument) =>
    {
        //Connect to MongoDB server and database
        var tweet = argument.Tweet;
        try
        {
            var client = new MongoClient();
            var database = client.GetDatabase("StreamData");
            var collection = database.GetCollection<BsonDocument>("StreamData");
            //Exclude any Retweeted Tweets
            if (tweet.IsRetweet) return;
            //Store the data as a BsonDocument into MongoDB database
            var tweetdata = new BsonDocument
            {
                //Store only the data that needed from a Tweet
                {"Timestamp", tweet.TweetLocalCreationDate},
                {"TweetID", tweet.IdStr},
                {"TweetContent",tweet.Text},
                {"DateCreated", tweet.CreatedBy.CreatedAt.Date},
                {"UserID", tweet.CreatedBy.IdStr},
                {"Username", tweet.CreatedBy.Name}
            };
            //Insert data into MongoDB database
            await collection.InsertOneAsync(tweetdata);
            //Every tweets streamed, add 1 into the variable
            numoftweet += 1;
            //If the number of tweets exceed 100, stopped the stream
            if (numoftweet >= 100)
            {
                stream.StopStream();
            }
        }
        //Catch if any exception/errors occured
        catch (Exception ex)
        {
            MessageDialog messagebox = new MessageDialog("MongoDB Connection Error:" + ex);
            await messagebox.ShowAsync();
        }
    };
    //Start the stream
    stream.StartStreamMatchingAllConditions();
}

备注:这是一个 UWP 应用程序,此代码位于按钮后面。

【问题讨论】:

  • 这五个任务不会“同时”运行(它们也不能,因为它们每个都依赖于前一个任务的结果)。由于您的任务本身可能是异步的(网络调用、数据库查找等),您应该更仔细地研究如何使用 async / await -- 可能从一个更简单的示例问题开始。
  • @PeterTorr-MSFT 是的,这些方法是异步的。我明白了,所以我应该使用 async/await 来运行它们而不是使用 Task.Run()?
  • 是的。您可以只在每次通话时await,一个接一个。
  • @PeterTorr-MSFT 谢谢!!

标签: c# multithreading uwp async-await task-parallel-library


【解决方案1】:

如 cmets 中所述,只需直接执行 await 您的方法,无需添加 Task.Run(看起来您的方法没有返回任何内容):

await GetKeyword(0);
await Connecting(1000, keyList);
await RetrieveData(1500);
await MakeRequest(2000, streamdata);
await Notify(2500, cyberbully, notification);

旁注:不要将async void 用于您的方法,它仅用于事件处理程序。如果您的方法没有返回任何内容,请使用async Task:

public static async Task Connecting(int sleepTime, List<string> keyList)

【讨论】:

  • @Mr.AF 谢谢,#SOReadyToHelp :)
【解决方案2】:

如何在 C# UWP 中暂停线程同时继续其他线程

您可以使用 ManualResetEvent

通知一个或多个等待线程发生了事件

reference

【讨论】:

  • 抱歉回复晚了,这是否意味着我使用 MRE 设置这 5 个线程,最初所有线程都在等待,从第一个线程开始,当第一个线程完成时通知第二个线程,是这样吗有效吗?
  • 是的,检查参考是否有用别忘了给我投票 LOL!
  • 但是,我确实想知道为什么第二个线程即 Connect 线程没有停止,即使我确实限制了方法中流式传输的数据数量,当它达到 100 个数据时我停止了流,但线程并没有停止,所以数据只是停止存储到数据库并停止流式传输,但线程本身仍在运行,不会进入下一个线程。
  • @Corene 我检查了你的代码,但我没有发现它需要深入的代码研究,但我建议你检查是否有循环或保持线程运行的东西
  • 您不需要 MRE ——事实上,您应该永远等待 UI 线程中的任何内容,否则您的应用将出现挂起(实际上可能会死锁) )。只需将await 用于异步方法即可。
猜你喜欢
  • 2019-10-03
  • 1970-01-01
  • 2021-09-12
  • 1970-01-01
  • 1970-01-01
  • 2016-12-20
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多