0

我想为我的家庭作业展示一个针对特定问题的信号量应用示例。我在我的 C# 表单中添加了 3 个按钮,我想表明在特定时间只有一个按钮在我的代码中执行银行帐户功能。当我在两秒钟内点击三个按钮时,银行账户功能必须只运行一次。因为我在银行帐户功能中有一个 Thread.Sleep(6000) 等待 6 秒。但是我的三个点击以 6 秒的间隔连续运行。连续按三个按钮时,如何将代码更改为仅运行一次。该代码是:

命名空间 semafor_form

{

public partial class Form1 : Form

   {
   Semaphore semafor=new Semaphore(1,1);

   delegate void SetTextCallback(string text);

   private void SetText(string text)
   {

       if (this.textBox2.InvokeRequired)
       {
           SetTextCallback d = new SetTextCallback(SetText);
           this.Invoke(d, new object[] { text });
       }
       else
       {
           this.textBox2.Text = text;
       }
   }

      public Form1()
    {
        InitializeComponent();

    }

    private void Form1_Load(object sender, EventArgs e)
    {

    }

    private void BankAccount()
    {
       semafor.WaitOne();
       double a = Convert.ToDouble (textBox1.Text) + Convert.ToDouble (textBox2.Text);
       Thread.Sleep(6000);
       semafor.Release(); 
       SetText(a.ToString());  
    }

    private void btnATM_Click(object sender, EventArgs e)
    {
        Thread t = new Thread(new ThreadStart(BankAccount));
        t.Start();
     }

    private void btnCOUNTER_Click(object sender, EventArgs e)
    {

        Thread t = new Thread(new ThreadStart(BankAccount));
        t.Start();
    }

    private void btnINT_Click(object sender, EventArgs e)
    {

        Thread t = new Thread(new ThreadStart(BankAccount));
        t.Start();
    }
 }   

}

4

2 回答 2

0

这听起来确实不适合使用信号量。如果我没看错的话,您的问题定义表明这三个按钮是互斥的:按下其中任何一个按钮都会使所有按钮在六秒钟内处于非活动状态。您可以为此使用信号量,但互斥量会更合适。

无论如何,您遇到的问题是您正在等待信号量,因此当第一个事务完成时,其他线程中的一个将获取信号量并进行处理。您要做的是尝试获取信号量。这是一个例子。

后者的一个例子:

private void BankAccount()
{
   if (semafor.WaitOne(0))  // tries to acquire the semaphore
   {
       double a = Convert.ToDouble (textBox1.Text) + Convert.ToDouble (textBox2.Text);
       Thread.Sleep(6000);
       semafor.Release(); 
       SetText(a.ToString());
   }
}

说, “WaitOne(0)尝试获取信号量。如果它不是立即可用,则返回 false。如果它可用,获取它并返回 true。”

您也可以在按钮处理程序中执行此操作。也就是说,让按钮处理程序获取信号量(使用WaitOne(0)),如果无法获取信号量,则让它退出而不启动线程。如果它确实获得了信号量,则启动线程并在完成后让线程 proc 释放信号量。

于 2013-05-04T12:57:59.260 回答
0

我可能误读了你的问题。您不希望按钮在使用线程时做任何事情吗?(所以你会错过交易?)

尝试这个:

private void BankAccount()
{
   if (semafor.WaitOne(0))
   {
       double a = Convert.ToDouble (textBox1.Text) + Convert.ToDouble (textBox2.Text);
       Thread.Sleep(6000);
       semafor.Release(); 
       SetText(a.ToString()); 
   } 
}

尝试更改Semaphore semafor=new Semaphore(1,1);Semaphore semafor=new Semaphore(0,1);

您正在初始化一个新的信号量而从未释放它。

于 2013-05-04T10:59:55.180 回答