运行并行任务c#

时间:2015-11-25 15:44:24

标签: c# multithreading windows-runtime windows-store-apps async-await

在Windows商店应用程序项目中,我有一个函数,我在foreach循环中多次调用以填充一些对象

方法标题如下所示

private async Task createInvite(JsonValue item, string meetingid, List<MeetingInvitee> listInvitees)

我试图在这样的并行任务中运行它

List<Task> ts = new List<Task>();
foreach (JsonValue item in invitees)
{
    ts.Add(createInvite(item, meetingid, listInvitees));
}

await Task.WhenAll(ts);

但它似乎并没有创建同时运行的任务。这大约需要10秒钟。

如果我使用此

//List<Task> ts = new List<Task>();
foreach (JsonValue item in invitees)
{
    //ts.Add(createInvite(item, meetingid, listInvitees));
      await createInvite(item, meetingid, listInvitees);
}

//await Task.WhenAll(ts);

它也需要大约10秒钟。

我的第一个选项是不是同时运行多个任务?

修改

private async Task createInvite(JsonValue item, string meetingid, List<MeetingInvitee> listInvitees)
{
    try
    {

        Stopwatch st = Stopwatch.StartNew();
        MeetingInvitee org = new MeetingInvitee();
        MeetingInvitee altorg = new MeetingInvitee();
        JsonObject invitee2;
        JsonObject invitee;
        InviteeDB InviteeForDB = new InviteeDB();
        GappService gappservice = new GappService();

        MeetingInvitee inv = new MeetingInvitee();


        JsonObject.TryParse(item.Stringify(), out invitee2);

        invitee = invitee2["user"].GetObject();



        if (invitee2.ContainsKey("_id"))
        {
            inv.id = invitee2["_id"].GetString();
            InviteeForDB.Id = invitee2["_id"].GetString();
        }
        else
        {
            InviteeForDB.Id = invitee2["user"].GetString();
        }

        if (invitee2.ContainsKey("status"))
        {
            if (invitee2["status"].ValueType == JsonValueType.Null)
            {
                inv.status = string.Empty;
                InviteeForDB.Status = string.Empty;
            }
            else
            {
                inv.status = invitee2["status"].GetString();
                InviteeForDB.Status = invitee2["status"].GetString();
            }
        }
        Stopwatch st2 = Stopwatch.StartNew();
        if (invitee2.ContainsKey("user"))
        {

            if (invitee2["user"].ValueType != JsonValueType.Null)
            {
                User iUser = new User();
                //iUser.Id = InviteeForDB.UserID;

                JsonSerializerSettings sett = new JsonSerializerSettings();
                sett.NullValueHandling = NullValueHandling.Ignore;
                iUser = JsonConvert.DeserializeObject<User>(invitee.Stringify(), sett);

                inv.user = iUser;
            }
            else
                return;
        }
        else
            return;

        InviteeForDB.MeetingID = meetingid;
        ds.inviteeRepository.Add(InviteeForDB);
        listInvitees.Add(inv);
    }
    catch (Exception e)
    {
        Debug.WriteLine(e.Message);
    }
}

2 个答案:

答案 0 :(得分:3)

您的代码实际上并不是异步的(它没有await个语句),因此所有工作都在createInvite()循环内的初始for调用中进行。

您可以使用Parallel.ForEach()实际并行运行,但这会破坏您的代码,因为它实际上不是线程安全的。

答案 1 :(得分:3)

您的方法没有真正的异步。简单地添加async修饰符并不会神奇地使此异步,或者在 parallel 中执行,这是我认为您正在尝试做的事情。它所做的就是向编译器发出需要生成状态机的信号。这段代码将完全同步执行。

您可以使用Task.RunParallel.ForEach在线程池线程上显式调用您的方法,但是查看您的代码看起来它看起来不像是线程安全的那。在继续之前,尝试将CreateInvite隔离为实际上是线程安全的,然后查看上面提到的其中一个选项。