3

我是 C# 新手。我有两个button和两个label。我想要的是,当我点击button1开始计数并显示在label1. 当我按下button2它时恢复计数button1开始计数button2并显示在标签2中。这是我的代码

bool testt = true;
int i = 0;
int j = 0;

private void button1_Click(object sender, EventArgs e)
{
    while (testt)
    {
        label1.Text = i.ToString();
        i++;
        System.Threading.Thread.Sleep(50);
        if (i > 5000)
        {
            i = 0;
        }
    }
}

private void button2_Click(object sender, EventArgs e)
{
    testt = false;
    while (!testt)
    {
        label2.Text = j.ToString();
        j++;
        System.Threading.Thread.Sleep(50);
        if (j > 5000)
        {
            i = 0;
        }
    }
}

这里的问题是当我单击一个按钮时,它不允许我单击另一个按钮。我使用了一个全局变量来检测按下了哪个按钮。但它不起作用。我怎样才能使这项工作?

4

2 回答 2

2

您正在 UI 线程上执行所有这些操作,因此您正在阻止它。您需要让另一个线程进行实际计数并将标签更改委托给主线程,Control.Invoke以便主线程有时间运行消息泵。您可以使用内置的BackgroundWorker或简单地使用 或 创建一个ThreadPool线程TaskFactory

我写了一个简短的例子来说明如何使用 LINQ 和 ThreadPool 来完成,这不是设计方面的最佳方法,但它会给你一个开发适当解决方案的起点

    semaphore = true;
    int i = 0;
    int j = 0;

    private void button1_Click(object sender, EventArgs e)
    {
        semaphore = true;
        ExecuteAsync(()=>
            {
                while (semaphore)
                {
                    //Dispatch a call to the UI thread to change the label
                    Invoke((MethodInvoker)(() => ChangeLabel(label1, i.ToString())));
                    i++;
                    Thread.Sleep(50);
                    if (i > 5000)
                    {
                        i = 0;
                    }
                }
            });
    }

    //Executes a function on a ThreadPool thread
    private void ExecuteAsync(Action action)
    {
        ThreadPool.QueueUserWorkItem(
            obj => action());
    }

    private void ChangeLabel(Label label, string labelText)
    {
        label.Text = labelText;
    }

    private void button2_Click(object sender, EventArgs e)
    {
        semaphore = false;
        ExecuteAsync(() =>
                {
                    while (!semaphore)
                    {
                        //Dispatch a call to the UI thread to change the label
                        Invoke((MethodInvoker)(() => ChangeLabel(label2, j.ToString())));
                        j++;
                        Thread.Sleep(50);
                        if (j > 5000)
                        {
                            i = 0;
                        }
                    }
                });
    }
于 2013-03-16T22:01:41.990 回答
0

您在这里遇到的问题是线程问题。

当你有 winforms 时,主线程的工作是显示和使界面交互并执行你给它的任何任务。

在您的代码中,您开始倒计时。但是在那里,使界面工作的同一线程现在正忙于降低您的价值,因此在准备好之前它不会更新界面或响应按钮。

你想要的是一个帮助线程为你做倒计时(或者老实说,启动一个 TIMER 代替)。否则,在您完成该循环之前,界面将是非交互式的。

计时器是最简单的方法。

http://msdn.microsoft.com/en-us/library/system.windows.forms.timer.aspx

于 2013-03-16T21:57:42.833 回答