繁体   English   中英

C# 检查 COM(串行)端口是否已打开

[英]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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM