Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 1632|回复: 0

C# 中 SerialPort.GetPortNames 获取串口号错误的问题及解决方法

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-23 13:19:46 | 显示全部楼层 |阅读模式

    开发环境:VS2008+.NET3.5

    以前一直没发现这个问题,感觉SerialPort.GetPortNames方法很好用,只需要这么一行就能直接获取到系统的串口列表。

    但当我们系统中存在虚拟串口时(部分蓝牙设备或者手机接上电脑后会虚拟出一些串口),此方法获取到的串口号可能就不是我们想要的结果了,如图:

    对比图片红色标记区域会发现,.net提供的方法获取的串口号后面带有一个特殊字符,如果仅是特殊字符那还好说,过滤一下就好了,但你GOOGLE会发现,有不少朋友碰到过这样的问题,他们串口号后面多出的字符可能是字母,数字,或者跟我一样是特殊字符,后两种情况都还好说,但如果多出的是数字呢(例如“COM3”变成了“COM36”)?

     

    起初我也只是过滤一下非数字,但在生产环境下什么情况都有可能发生,所以又改成了调用系统API读取注册表并处理读取结果的方案,以下为我的代码:

            [DllImport("advapi32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true)]
            private static extern int RegCloseKey(int hKey);
    
            [DllImport("advapi32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true, EntryPoint = "RegOpenKeyA")]
            private static extern int RegOpenKey(uint hKey, string lpSubKey, ref int phkResult);
    
            [DllImport("advapi32.dll", CharSet = CharSet.Ansi, SetLastError = true, ExactSpelling = true, EntryPoint = "RegEnumValueA")]
            private static extern int RegEnumValue(int hKey, int dwIndex, [MarshalAs(UnmanagedType.VBByRefStr)] ref string lpValueName, ref int lpcbValueName, int lpReserved, int lpType, [MarshalAs(UnmanagedType.VBByRefStr)] ref string lpData, ref int lpcbData);

     

            /// <summary>
            /// 获取当前计算机的串口名称数组
            /// </summary>
            /// <returns></returns>
            public static string[] GetSerialPortNames()
            {
                #region 方式一:调用系统API(DLL文件:advapi32.dll)读取注册表,并处理读取结果的“字符串结束符”
                string[] ports = null;
                List<string> portsList = new List<string>();
                uint HKEY_LOCAL_MACHINE = 0x80000002;
                int hKey = -1;
                int ret = RegOpenKey(HKEY_LOCAL_MACHINE, @"Hardware\DEVICEMAP\SERIALCOMM", ref hKey);
                try
                {
                    if (ret == 0)
                    {
                        int index = 0;
                        int BufferSize = 255;
                        int ERROR_NO_MORE_ITEMS = 259;
                        string valueName = "".PadRight(BufferSize, ' ');
                        int valueNameLength = BufferSize;
                        int valueLength = BufferSize;
                        string value = "".PadRight(BufferSize, ' ');
                        while (RegEnumValue(hKey, index, ref valueName, ref valueNameLength, 0, 0, ref value, ref valueLength) != ERROR_NO_MORE_ITEMS)
                        {
                            if (valueLength > 0)
                            {
                                if (value[valueLength - 1] == 0)
                                    valueLength -= 1;
                                portsList.Add(value.Substring(0, valueLength));
                            }
                            index += 1;
                            valueName = "".PadRight(BufferSize, ' ');
                            valueNameLength = BufferSize;
                            valueLength = BufferSize;
                        }
                    }
                }
                catch (Exception)
                {
                }
                finally 
                {
                    if (ret == 0)
                        RegCloseKey(hKey);
                }
                if (portsList.Count == 0)
                    ports = new string[0];
                else
                    ports = portsList.ToArray();
                return ports;
                #endregion
    
                #region 方式二:C#方式读取注册表后过滤非数字(此方式无法过滤数字,可能造成部分虚拟串口获取错误,例如 COM3 识别为 COM32)
                //HashSet<char> numbers = new HashSet<char>(new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' });
                //string[] ports = null;
    
                //RegistryKey localMachine = null;
                //RegistryKey subKey = null;
                //new RegistryPermission(RegistryPermissionAccess.Read, @"HKEY_LOCAL_MACHINE\HARDWARE\DEVICEMAP\SERIALCOMM").Assert();
                //try
                //{
                //    localMachine = Registry.LocalMachine;
                //    subKey = localMachine.OpenSubKey(@"HARDWARE\DEVICEMAP\SERIALCOMM", false);
                //    if (subKey != null)
                //    {
                //        string[] valueNames = subKey.GetValueNames();
                //        ports = new string[valueNames.Length];
                //        for (int j = 0; j < valueNames.Length; j++)
                //        {
                //            object obj = subKey.GetValue(valueNames[j]);
                //            if (obj == null)
                //                continue;
                //            string str = (string)obj;
                //            if (string.IsNullOrEmpty(str) || str.Length < 4)
                //                continue;
    
                //            string port = "COM";
                //            char[] cs = str.Substring(3).Replace("\n", "").ToCharArray();
                //            foreach (char c in cs)
                //            {
                //                if (numbers.Contains(c))
                //                    port += c.ToString();
                //                else
                //                    break;
                //            }
                //            ports[j] = port;
                //        }
                //    }
                //}
                //finally
                //{
                //    if (localMachine != null)
                //        localMachine.Close();
                //    if (subKey != null)
                //        subKey.Close();
                //    System.Security.CodeAccessPermission.RevertAssert();
                //}
    
                //if (ports == null)
                //    ports = new string[0];
                //return ports;
                #endregion
            }
       

     

    另外,针对以上问题,微软其实在08年的时候给过回复,说在下一版本.net(也就是.net Framework 4)中解决此问题,确实,在.net Framework 4 中已经不会存在此问题,所以如果有条件的话大家还是用.net Framework 4 开发比较好些,没办法必须得用3.5的话那就试试上面所说的方法吧。经测试4.0确实没有上问题了!

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-12-23 07:13 , Processed in 0.067638 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表