我有一个程序,每隔X秒就会通过串行COM报告文件夹结构的大小。 当用户也通过串行COM发送命令时,它还能够报告文件夹结构中所有文件的列表。
举个例子: 用户每隔10秒获取一次大小信息,在某些时候他们想要知道文件列表。因此,他们通过COM端口发送“1”,然后程序开始报告文件。
我遇到的问题是两个函数无法同时写入COM端口,因此它开始抛出异常。
我想做的是让两个功能等到另一个完成。这是我的代码。
这是写入COM端口的简单功能:
private void ComWrite(string msg)
{
ComPort.Write(msg);
}
我从这两个问题中称之为:
这个报告文件名:
private void GetFileNames()
{
fileNames = Directory.GetFiles(textBox1.Text, "*.wav", SearchOption.AllDirectories);
for (int i = 0; i < fileNames.Length; i++)
{
ComWrite((fileNames[i] + "\r\n"));
}
}
这是一个计时器并报告文件夹大小:
public void OnTimedEvent(object source, ElapsedEventArgs elapsed)
{
folderSize = DirSize(new DirectoryInfo(textBox1.Text)) / 1000000;
string labelText = folderSize.ToString() + "Mb";
label3.Text = labelText;
if (checkBox1.Checked)
{
try
{
ComWrite(labelText + "\r\n");
label9.Text = labelText;
}
catch (Exception)
{
MessageBox.Show("Please Open COMPORT before sending command");
}
}
}
如何实现异步函数或其他一些阻止它们相互脱落的方法?
编辑:请求的代码。
这是在表单加载时设置COM端口。
private void Form1_Load(object sender, EventArgs e)
{
label5.Text = "Idle";
ComPort.BaudRate = Convert.ToInt32("9600");
ComPort.DataBits = Convert.ToInt16("8");
ComPort.StopBits = (StopBits)Enum.Parse(typeof(StopBits), "One");
ComPort.Handshake = (Handshake)Enum.Parse(typeof(Handshake), "None");
ComPort.Parity = (Parity)Enum.Parse(typeof(Parity), "None");
}
然后点击此处打开您通过组合框选择的COM端口:
private void btnPortState_Click(object sender, EventArgs e)
{
try
{
if (btnPortState.Text == "COMPort Closed / Click to Open")
{
btnPortState.BackColor = Color.Green;
btnPortState.Text = "COMPort Open / Click To Close";
ComPort.PortName = Convert.ToString(cboPorts.Text);
ComPort.Open();
}
else if (btnPortState.Text == "COMPort Open / Click To Close")
{
btnPortState.Text = "COMPort Closed / Click to Open";
btnPortState.BackColor = Color.Firebrick;
ComPort.Close();
}
}
catch (Exception)
{
MessageBox.Show("You must select a COMPORT before opening it.");
}
}
答案 0 :(得分:2)
我认为您想要的是对您的com端口进行的任何交易的互斥。
如果问题是2个不同的线程可以同时开始写入comport,这可以通过使用信号量来解决。
在你的情况下,我猜你在发送文件名时传输目录大小并不好。所以你不能简单地在ComWrite
内进行snychonize。
相反,我会提出以下建议:
在课堂上:
//semaphore so only 1 Thread at a time can pass through
private static Semaphore semaphore = new Semaphore(1, 1); //(1,1) -> see comments
在OnTimedEvent
:
if (checkBox1.Checked)
{
try
{
semaphore.WaitOne(); //Increases semaphore counter
ComWrite(labelText + "\r\n");
semaphore.Release();
label9.Text = labelText;
}
(...)
和
private void GetFileNames()
{
semaphore.WaitOne();
(...) //writing the file names to ComPort
semaphore.Release();
}
Semaphores有点像柜台一样工作。如果计数器达到其最大计数,则Wait
调用不再有线程可以通过并等待轮到他们。释放信号量基本上告诉计数器减少,现在允许等待线程进入。