[英]C# check if a COM (Serial) port is already open
是否有一种简单的方法以编程方式检查串行 COM 端口是否已打开/正在使用?
通常我会使用:
try
{
// open port
}
catch (Exception ex)
{
// handle the exception
}
但是,我想以编程方式进行检查,以便尝试使用另一个 COM 端口或类似端口。
前段时间我需要类似的东西来搜索设备。
我获得了一个可用 COM 端口的列表,然后简单地遍历它们,如果它没有抛出异常,我尝试与设备进行通信。 有点粗糙但工作。
var portNames = SerialPort.GetPortNames();
foreach(var port in portNames) {
//Try for every portName and break on the first working
}
我是这样做的:
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
internal static extern SafeFileHandle CreateFile(string lpFileName, int dwDesiredAccess, int dwShareMode, IntPtr securityAttrs, int dwCreationDisposition, int dwFlagsAndAttributes, IntPtr hTemplateFile);
然后后来
int dwFlagsAndAttributes = 0x40000000;
var portName = "COM5";
var isValid = SerialPort.GetPortNames().Any(x => string.Compare(x, portName, true) == 0);
if (!isValid)
throw new System.IO.IOException(string.Format("{0} port was not found", portName));
//Borrowed from Microsoft's Serial Port Open Method :)
SafeFileHandle hFile = CreateFile(@"\\.\" + portName, -1073741824, 0, IntPtr.Zero, 3, dwFlagsAndAttributes, IntPtr.Zero);
if (hFile.IsInvalid)
throw new System.IO.IOException(string.Format("{0} port is already open", portName));
hFile.Close();
using (var serialPort = new SerialPort(portName, 115200, Parity.None, 8, StopBits.One))
{
serialPort.Open();
}
对于不能使用SerialPort.GetPortNames();
因为他们不是针对.net framework
(就像在我的情况下我使用 .Net Core 而不是 .Net Framework)这是我最终做的:
在命令提示符下,如果你输入 mode 你会得到这样的东西:
mode 是一个位于C:\\Windows\\System32\\mode.com
的可执行文件。 只需使用这样的正则表达式解析该可执行文件的结果:
// Code that answers the question
var proc = new Process
{
StartInfo = new ProcessStartInfo
{
FileName = @"C:\Windows\System32\mode.com",
UseShellExecute = false,
RedirectStandardOutput = true,
CreateNoWindow = true
}
};
proc.Start();
proc.WaitForExit(4000); // wait up to 4 seconds. It usually takes less than a second
// get ports being used
var output = proc.StandardOutput.ReadToEnd();
现在,如果您想解析输出,我就是这样做的:
List<string> comPortsBeingUsed = new List<string>();
Regex.Replace(output, @"(?xi) status [\s\w]+? (COM\d) \b ", regexCapture =>
{
comPortsBeingUsed.Add(regexCapture.Groups[1].Value);
return null;
});
foreach(var item in comPortsBeingUsed)
{
Console.WriteLine($"COM port {item} is in use");
}
我想打开下一个可用端口并这样做。 请注意,它不是用于 WPF,而是用于 Windows 窗体。 我用可用的 com 端口填充了一个组合框。 然后我尝试打开第一个。 如果失败,我会从组合框中选择下一个可用项目。 如果选定的索引最终没有改变,则没有可用的备用 com 端口,我们会显示一条消息。
private void GetPortNames()
{
comboBoxComPort.Items.Clear();
foreach (string s in SerialPort.GetPortNames())
{
comboBoxComPort.Items.Add(s);
}
comboBoxComPort.SelectedIndex = 0;
}
private void OpenSerialPort()
{
try
{
serialPort1.PortName = comboBoxComPort.SelectedItem.ToString();
serialPort1.Open();
}
catch (Exception ex)
{
int SelectedIndex = comboBoxComPort.SelectedIndex;
if (comboBoxComPort.SelectedIndex >= comboBoxComPort.Items.Count - 1)
{
comboBoxComPort.SelectedIndex = 0;
}
else
{
comboBoxComPort.SelectedIndex++;
}
if (comboBoxComPort.SelectedIndex == SelectedIndex)
{
buttonOpenClose.Text = "Open Port";
MessageBox.Show("Error accessing port." + Environment.NewLine + ex.Message, "Port Error!!!", MessageBoxButtons.OK);
}
else
{
OpenSerialPort();
}
}
if (serialPort1.IsOpen)
{
StartAsyncSerialReading();
}
}
SerialPort 类有一个Open方法,它会抛出一些异常。 上面的参考资料包含详细的例子。
另请参阅IsOpen属性。
一个简单的测试:
using System;
using System.IO.Ports;
using System.Collections.Generic;
using System.Text;
namespace SerPort1
{
class Program
{
static private SerialPort MyPort;
static void Main(string[] args)
{
MyPort = new SerialPort("COM1");
OpenMyPort();
Console.WriteLine("BaudRate {0}", MyPort.BaudRate);
OpenMyPort();
MyPort.Close();
Console.ReadLine();
}
private static void OpenMyPort()
{
try
{
MyPort.Open();
}
catch (Exception ex)
{
Console.WriteLine("Error opening my port: {0}", ex.Message);
}
}
}
}
这对我有用。
private string portName { get; set; } = string.Empty;
/// <summary>
/// Returns SerialPort Port State (Open / Closed)
/// </summary>
/// <returns></returns>
internal bool HasOpenPort()
{
bool portState = false;
if (portName != string.Empty)
{
using (SerialPort serialPort = new SerialPort(portName))
{
foreach (var itm in SerialPort.GetPortNames())
{
if (itm.Contains(serialPort.PortName))
{
if (serialPort.IsOpen) { portState = true; }
else { portState = false; }
}
}
}
}
else { System.Windows.Forms.MessageBox.Show("Error: No Port Specified."); }
return portState;
}
如果你需要捕获SerialPort(开放)例外: https ://docs.microsoft.com/en-us/dotnet/api/system.io.ports.serialport.open?view = networkwork-4.7.2
分享对我有用的东西(一个简单的辅助方法):
private string portName { get; set; } = string.Empty;
/// <summary>
/// Returns SerialPort Port State (Open / Closed)
/// </summary>
/// <returns></returns>
internal bool HasOpenPort()
{
bool portState = false;
if (portName != string.Empty)
{
using (SerialPort serialPort = new SerialPort(portName))
{
foreach (var itm in SerialPort.GetPortNames())
{
if (itm.Contains(serialPort.PortName))
{
if (serialPort.IsOpen) { portState = true; }
else { portState = false; }
}
}
}
}
else { System.Windows.Forms.MessageBox.Show("Error: No Port Specified."); }
return portState;
}
注意事项:
- 对于更高级的技术,我建议使用ManagementObjectSearcher Class 。
更多信息在这里。
- 对于 Arduino 设备,我会让端口保持打开状态。
- 如果您需要捕获异常,建议使用 Try Catch 块。
- 还检查:“超时异常”
- 有关如何在此处获取 SerialPort (Open) Exceptions 的更多信息。
public void MobileMessages(string ComNo, string MobileMessage, string MobileNo)
{
if (SerialPort.IsOpen )
SerialPort.Close();
try
{
SerialPort.PortName = ComNo;
SerialPort.BaudRate = 9600;
SerialPort.Parity = Parity.None;
SerialPort.StopBits = StopBits.One;
SerialPort.DataBits = 8;
SerialPort.Handshake = Handshake.RequestToSend;
SerialPort.DtrEnable = true;
SerialPort.RtsEnable = true;
SerialPort.NewLine = Constants.vbCrLf;
string message;
message = MobileMessage;
SerialPort.Open();
if (SerialPort.IsOpen )
{
SerialPort.Write("AT" + Constants.vbCrLf);
SerialPort.Write("AT+CMGF=1" + Constants.vbCrLf);
SerialPort.Write("AT+CMGS=" + Strings.Chr(34) + MobileNo + Strings.Chr(34) + Constants.vbCrLf);
SerialPort.Write(message + Strings.Chr(26));
}
else
("Port not available");
SerialPort.Close();
System.Threading.Thread.Sleep(5000);
}
catch (Exception ex)
{
message.show("The port " + ComNo + " does not exist, change port no ");
}
}
几个星期以来,我一直在与这个问题作斗争。 感谢此处和网站https://www.dreamincode.net/forums/topic/91090-c%23-serial-port-unauthorizedaccessexception/的建议。
我终于想出了一个似乎可行的解决方案。
我正在开发的应用程序允许用户连接到 USB 设备并显示来自它的数据。
我正在与之抗争的问题。 除了我正在编写的应用程序之外,我还使用另一个串行终端应用程序来进行测试。 有时我忘记断开其他应用程序正在使用的 COMport。 如果我这样做,并尝试连接我正在编写的应用程序,我会收到“UnAuthorizedAccessException”错误。 与此异常一起出现的还有一些副作用,例如双行数据被吐出以及应用程序在关闭时锁定。
我的解决方案
感谢此处和引用的其他站点的建议,这是我的解决方案。
private void checkAndFillPortNameList()
{
SerialPort _testingSerialPort;
AvailablePortNamesFound.Clear();
List<string> availablePortNames = new List<string>();//mySerial.GetAvailablePortNames();
foreach (string portName in SerialPortDataAccess.GetAvailablePortNames())
{
try
{
_testingSerialPort = new SerialPort(portName);
_testingSerialPort.Open();
if (_testingSerialPort.IsOpen)
{
availablePortNames.Add(portName);
_testingSerialPort.Close();
}
}
catch (Exception ex)
{
}
}
availablePortNames.Sort();
AvailablePortNamesFound = new ObservableCollection<string>(availablePortNames);
}
此例程连接到 combobox,其中包含可供选择的可用 Comports。 如果 Comport 已经被另一个应用程序使用,则该端口名称将不会出现在组合框中。
您可以尝试以下代码来检查端口是否已经打开。 我假设您不知道要检查哪个端口。
foreach (var portName in Serial.GetPortNames()
{
SerialPort port = new SerialPort(portName);
if (port.IsOpen){
/** do something **/
}
else {
/** do something **/
}
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.