Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/excel/24.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C# Async和Await命令编译时不会出错,但不会让我的Windows窗体响应_C#_Excel_Winforms_Asynchronous - Fatal编程技术网

C# Async和Await命令编译时不会出错,但不会让我的Windows窗体响应

C# Async和Await命令编译时不会出错,但不会让我的Windows窗体响应,c#,excel,winforms,asynchronous,C#,Excel,Winforms,Asynchronous,我有一张excel表格,上面有各种工具箱项目。我有一个下拉菜单,允许用户选择一个值并点击运行按钮my button1,假设数据在Excel工作表中,并且当前是手动完成的,则所有操作都正常。但是,如果我想添加第二个按钮2来自动刷新SQL Server中的数据,它会运行。。。但不是异步的。这是我的按钮2: private async void button2_Click(object sender, EventArgs e) { var Excel = Globals.Thi

我有一张excel表格,上面有各种工具箱项目。我有一个下拉菜单,允许用户选择一个值并点击运行按钮my button1,假设数据在Excel工作表中,并且当前是手动完成的,则所有操作都正常。但是,如果我想添加第二个按钮2来自动刷新SQL Server中的数据,它会运行。。。但不是异步的。这是我的按钮2:

private async void button2_Click(object sender, EventArgs e)
    {
        var Excel = Globals.ThisWorkbook.Application;
        var activebook = Excel.ActiveWorkbook;
        var ws = Excel.ActiveSheet;
        SQLServer server = new SQLServer();
        int t = await Task.Run(() => server.getInfo(activebook, ws, Excel));     
    }
代码运行良好,但我不想让用户在excel窗口响应之前等待填写6000行excel工作表。我做错什么了吗

编辑 我很抱歉给你带来困惑,我对这个网站还是相当陌生的,并不是故意造成这么多困惑的。我正在尝试运行此方法:

 public void getInfo(Workbook book, Worksheet activesheet,     Microsoft.Office.Interop.Excel.Application app)
    {          
        SqlConnection cnn;

        string connectionstring = "#########";
        string sql = null; 

        ////***Opening SQL Database

        connectionstring = "Data Source=########;Initial Catalog=BrightreeData;Persist Security Info=True;User ID=######;Password=######";
        cnn = new SqlConnection(connectionstring);
        cnn.Open(); 
        activesheet.Range["K" + 1].Value = "Connected";


        ////** Write your Sql Query here
        sql = "SELECT [StopReason], [SOKey], [NickName] FROM [BrightreeData].[dbo].[SalesOrder] Where [StopReason] = 'Ineligible Policy' ORDER BY [SOKey]";           


        ///*** Preparing to retrieve value from the database
        SQL.DataTable dtable = new SQL.DataTable();

        SqlDataAdapter dscmd = new SqlDataAdapter(sql, cnn);
        SQL.DataSet ds = new SQL.DataSet();
        dscmd.Fill(dtable);
        int count = dtable.Rows.Count;
        for (int i = 0; i < dtable.Rows.Count; i++)
        {               

            try
            {
     check: activesheet.Range["C" + 1].Value = (i + 1) + " out of " + count + " rows";
            soService.DataFetchServiceResponseUsingSalesOrder salesorder = soAction.SalesOrderFetchByBrightreeID(dtable.Rows[i]["SOKey"].ToString());
            activesheet.Range["J" + (8 + i)].Value = dtable.Rows[i]["NickName"];
            activesheet.Range["A" + (8 + i)].Value = salesorder.Items[0].SalesOrderInsuranceInfo.Payors[1].payorPolicyInfo.Name.ToString();
            activesheet.Range["D" + (8 + i)].Value = salesorder.Items[0].SalesOrderInsuranceInfo.Payors[1].payorPolicyInfo.PolicyNumber;
            //If the current row has the same policy as the last row, it is a repeat and nothing worth grabbing that data.
            //
            //    Eventually add something at the end of the program that deletes rows that are blank?...for now I will manually delete empty rows.
            if(activesheet.Range["D" + (8 + i)].Value == activesheet.Range["D" + (7 + i)].Value & activesheet.Range["D" + (8 + i)].Text != "")
            {
                activesheet.Range["C" + (8 + i)].Value = "Repeat Patient";
                count--;
                i++;
                goto check;
            }
            activesheet.Range["J" + (8 + i)].Value = "St Lukes";
            activesheet.Range["K" + (8 + i)].Value = System.DateTime.Today.Date;
            string ptID = Convert.ToString(salesorder.Items[0].SalesOrderClinicalInfo.Patient.BrightreeID);
            ptService.DataFetchServiceResponseUsingPatient ptSearch = ptAction.PatientFetchByBrightreeID(ptID);
            activesheet.Range["B" + (8 + i)].Value = ptSearch.Items[0].PatientGeneralInfo.Name.First;
            activesheet.Range["C" + (8 + i)].Value = ptSearch.Items[0].PatientGeneralInfo.Name.Last;
            activesheet.Range["E" + (8 + i)].Value = ptSearch.Items[0].PatientGeneralInfo.BirthDate;
            activesheet.Range["L" + (8 + i)].Value = ptSearch.Items[0].PatientGeneralInfo.PtID;
            }catch
            {
                activesheet.Range["B" + (8 + i)].Value = "Error getting patient information";
            }

        }   

    }
在我的button2\u Click方法的最后一行抛出错误

 An exception of type 'Microsoft.CSharp.RuntimeBinder.RuntimeBinderException'
occurred in System.Core.dll but was not handled in user code

Additional information: Cannot implicitly convert type 'type' to 'object'
试试这个:

await Task.Run((Action)(() => server.getInfo(activebook, ws, Excel)));
出于某种原因,可能是因为您传递的参数是COM对象,所以系统将参数和方法调用视为动态的,并且会错误地认为您传递的委托将返回一个值

如果将鼠标移到visual studio中运行的Task.Run上,则应看到系统正在尝试调用此重载:

Task<TResult> Run<TResult>(Func<TResult> function)
其中TResult是动态的。

请尝试以下操作:

await Task.Run((Action)(() => server.getInfo(activebook, ws, Excel)));
出于某种原因,可能是因为您传递的参数是COM对象,所以系统将参数和方法调用视为动态的,并且会错误地认为您传递的委托将返回一个值

如果将鼠标移到visual studio中运行的Task.Run上,则应看到系统正在尝试调用此重载:

Task<TResult> Run<TResult>(Func<TResult> function)

TResult是动态的。

你在做两次同样的事情,不是吗?一次同步,然后异步等待它在后台线程上再次运行?我建议删除对getInfo的第一个调用。请注意,您调用server.getInfo两次-在任务中调用一次。运行一次,不运行一次。在任何情况下,将数据读取与填充工作表分离都是一个好主意:请将代码作为文本而不仅仅是图像发布。你让你的问题变得更难解决。。。但是是的,你好像打了两次getInfo。。。为什么?顺便提一下,Excel API是线程安全的吗?也就是说,是否允许您从创建Excel工作表的线程以外的线程向Excel工作表添加数据?@adv12您可以不受影响,但每次调用都会被封送回应用程序线程。最终的结果是,在另一个线程上执行大量互操作的速度要慢一个数量级。同样的事情你要做两次,不是吗?一次同步,然后异步等待它在后台线程上再次运行?我建议删除对getInfo的第一个调用。请注意,您调用server.getInfo两次-在任务中调用一次。运行一次,不运行一次。在任何情况下,将数据读取与填充工作表分离都是一个好主意:请将代码作为文本而不仅仅是图像发布。你让你的问题变得更难解决。。。但是是的,你好像打了两次getInfo。。。为什么?顺便提一下,Excel API是线程安全的吗?也就是说,是否允许您从创建Excel工作表的线程以外的线程向Excel工作表添加数据?@adv12您可以不受影响,但每次调用都会被封送回应用程序线程。最终的结果是,在另一个线程上执行大量互操作要慢一个数量级。我尝试了您的更改,但编译器给了我错误。我尝试了System.Action和Microsoft.Office.Excel.Action调用,但两者似乎都不喜欢。对不起,我忘记了括号。更新了答案,成功了!非常感谢你!我欠你一大笔钱。我搞乱了动作课,但不确定它是如何运作的。多谢各位@Rinktacular,很高兴能帮上忙。我尝试了你的更改,但编译器给了我错误。我尝试了System.Action和Microsoft.Office.Excel.Action调用,但两者似乎都不喜欢。对不起,我忘记了括号。更新了答案,成功了!非常感谢你!我欠你一大笔钱。我搞乱了动作课,但不确定它是如何运作的。多谢各位@Rinktacular,很乐意帮忙。