4

这是我在 StackOverflow 论坛上的第一篇文章,所以请宽容。我对同步调用但不能异步调用的函数有疑问。下面你会发现同步调用的函数:

private void issueInvoices(List<int> lista)
    {
foreach (int knh_id in lista)
          {
                    Invoice fs = new Invoice();
                    fs.FKS_AKCYZA = false;
                    fs.FKS_CZY_KLON = false;
                    fs.FKS_DATE = Convert.ToDateTime(MTBDataZapisuDoFK.Text);
                    fs.NUMBER = knh_id);
         }
    }

如您所见,我将列表传递给名为 issueInvoices 发票编号列表的函数,并在循环中创建了一些发票。此函数正常工作,但如果我尝试异步调用它(以显示进度条),我的函数无法分配给 fs.FKS_DATE 对象 dateTime。看起来静态函数“Convert.ToDateTime”无法正常工作。但是请看下面的代码,其中函数 issueInvoices 被异步调用......</p>

public delegate void BinaryDelegate(List<int> knh_id);
BinaryDelegate b = new BinaryDelegate(issueInvoices);
IAsyncResult theAsRes = b.BeginInvoke(lista, new AsyncCallback(AddComplete), "Thx U!");
FrmProgressBar fpb=new FrmProgressBar(“Please wait…”);
 fpb.Show();

 /* below i check how many operation i have to do, if all operations are done, then I close fpb window, program is updating progres bar and in thread make operation issueInvoices*/
                        while (ilosc_zrobionych != liczbaKontrahentow)
                       {
                            fpb.PBStan.Value = (int)((100 * ilosc_zrobionych) / liczbaKontrahentow);
                       }
                        fpb.Close();

我放了一些断点,它看起来像程序停止在行,它可以转换为日期时间,但是当我同步执行此操作时,它可以正常工作而没有任何错误。fs.FKS_DATE = Convert.ToDateTime(MTBDataZapisuDoFK.Text); 什么可能导致这个问题以及如何解决它?非常感谢您的回复。

下面是异步调用的整个类:

using System;
using System.Collections.Generic;
using System.Collections;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using Npgsql;
using Castle.ActiveRecord;
using WFR.Model;
using System.Threading;

namespace Faktury_i_Rachunki_2.Forms
{

    public partial class FrmEmisjaFakturPotwierdzonych : FrmBaseForm
    {

        private ArrayList listaSposobowZaplaty;
        public List<int> lista;
        private int liczbaWygenerowach;
        private int liczbaKontrahentow;
        private int ilosc_zrobionych;
        private FrmProgressBar fpb;

        public delegate void BinaryDelegate(List<int> knh_id);


        public FrmEmisjaFakturPotwierdzonych()
        {
            InitializeComponent();
    fpb = new FrmProgressBar("Please wait....");
        }

        private void BtOK_Click(object sender, EventArgs e)
        {                
                BinaryDelegate b = new BinaryDelegate(WyemitujFakture);

                    lista.Add(12);
                    lista.Add(13);
                    lista.Add(17);
                    lista.Add(1);

                liczbaKontrahentow = lista.Count;
                if (TBRejestr.Text.Trim() != "")
                {

                    if (liczbaKontrahentow > 0)
                    {
                        liczbaWygenerowach = 0;
                        ilosc_zrobionych = 0;
                        WyemitujFakture(lista);
              IAsyncResult theAsRes = b.BeginInvoke(lista, new AsyncCallback(AddComplete), "THX");

                        fpb.Show();
                        while (ilosc_zrobionych != liczbaKontrahentow)
                        {
                            fpb.PBStan.Value = (int)((100 * ilosc_zrobionych) / liczbaKontrahentow);
                        }
                        fpb.Close();
                    }

                    try
                    {
                        MessageBox.Show("Wygenerowano " + liczbaWygenerowach.ToString() + " faktur");
                    }
                    catch
                    {
}

        }
}
        private void WyemitujFakture(List<int> lista)
        {
            foreach (int knh_id in lista)
            {
                try
                {
                                    if (luk.Count > 0)
                    {
                        FakturySprzedazy fs = new FakturySprzedazy();
                        fs.FKS_AKCYZA = false;
                        fs.FKS_CZY_KLON = false;
                        fs.FKS_DATA_DOW_KS = Convert.ToDateTime(MTBDataZapisuDoFK.Text);
                        fs.FKS_DATA_FAKTURY = Convert.ToDateTime(MTBDataFaktury.Text);
                        fs.FKS_DATA_SPRZEDAZY = Convert.ToDateTime(MTBDataSprzedazy.Text);
                        liczbaWygenerowach++;
                    }

                }
                catch (Exception ex)
                {
                    MessageBox.Show("Nie można wyemitować faktury dla kontrahenta o id = " + knh_id.ToString() + " " + ex.Message);
                }
                ilosc_zrobionych++;
            }
        }
4

2 回答 2

3

您正在从后台线程访问 UI 控件:

MTBDataZapisuDoFK.Text

这是不允许的。

在调用方法之前获取此值,将其存储在变量中并将值作为参数发送到issueInvoices.

于 2012-09-15T09:44:05.013 回答
1

问题在于获取MTBDataZapisuDoFK.Text(我假设它是一个文本框)的值。获取或设置文本框的文本意味着向其窗口发送消息。但是你让 UI 线程在 while 循环中保持忙碌,因此它不能处理任何消息。

在 while 循环中调用 toApplication.DoEvents()以允许处理消息:

fpb.Show();
while (ilosc_zrobionych != liczbaKontrahentow)
{
    Application.DoEvents();
    fpb.PBStan.Value = (int)((100 * ilosc_zrobionych) / liczbaKontrahentow);
}
fpb.Close();

我假设异步调用该方法的唯一原因是能够在处理 - 方法期间更新 UI WyemitujFakture。使用Application.DoEvents()你不需要异步调用:

fpb = new FrmProgressBar("Please wait....");
fpb.Show();
Application.DoEvents();
WyemitujFakture(lista);
fpb.Close();

您应该在调用Application.DoEvents()后调用fpb.Show()以使表单正确显示。此外,您应该在方法本身而不是构造函数中实例化表单,因为在调用后您不能再次使用相同的实例fpb.Close()(它将被释放)。

然后可以在WyemitujFakture-method中更新进度条:

private void WyemitujFakture(List<int> lista)
{
    foreach (int knh_id in lista)
    {
        try
        {
            if (luk.Count > 0)
            {
                FakturySprzedazy fs = new FakturySprzedazy();
                fs.FKS_AKCYZA = false;
                fs.FKS_CZY_KLON = false;
                fs.FKS_DATA_DOW_KS = Convert.ToDateTime(MTBDataZapisuDoFK.Text);
                fs.FKS_DATA_FAKTURY = Convert.ToDateTime(MTBDataFaktury.Text);
                fs.FKS_DATA_SPRZEDAZY = Convert.ToDateTime(MTBDataSprzedazy.Text);
                liczbaWygenerowach++;
            }

        }
        catch (Exception ex)
        {
            MessageBox.Show("Nie mozna wyemitowac faktury dla kontrahenta o id = " + knh_id.ToString() + " " + ex.Message);
        }
        ilosc_zrobionych++;

        fpb.PBStan.Value = (int)((100 * ilosc_zrobionych) / liczbaKontrahentow);
        Application.DoEvents();
    }
}
于 2012-09-15T15:41:03.793 回答