VC++位移操作>>和<<以及逻辑驱动器插拔产生的掩码dbv.dbcv_unitmask进行分析的相关代码

本文主要是介绍VC++位移操作>>和<<以及逻辑驱动器插拔产生的掩码dbv.dbcv_unitmask进行分析的相关代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

VC++位移操作>>和<<以及逻辑驱动器插拔产生的掩码dbv.dbcv_unitmask进行分析的相关代码

  • 一、VC++位移操作符<<和>>
    • 1、右位移操作符 >>:
    • 2、左位移操作符 <<:
  • 二、逻辑驱动器插拔产生的掩码 dbv.dbcv_unitmask 进行分析的相关代码
    • 1、设备发生改变的消息 WM_DEVICECHANGE
      • A、微软官方 VC++ 代码:
      • B、驱动器号与设备掩码 dbcv_unitmask 的关系
    • 2、设备状态发生改变 WM_DEVICECHANGE 的改进 C# 代码:

一、VC++位移操作符<<和>>

1、右位移操作符 >>:

右移运算符>>,是将一个数的二进制位全部右移若干位,低位移出(舍弃)。高位的空位补符号位,即正数补零,负数补1。

语法格式: 需要移位的数字 >> 移位的次数

例如:
11 >> 2,等于2,是将数字11的二进制右移2位,即00001011------>00000010(2);
2 >> 1:相当于2/2
99 >> 1: 相当于 99/2 向下取整为49
99 >> 2:相当于 99/pow(2,2)向下取整为24
999 >> i:相当于999/pow(2,i)
整数 >> i:相当于将这个整数化为二进制整数,并去掉这个数的末尾的 i 位数字

cout << (2 >> 1) << endl;//输出为 1
cout << (99 >> 1) << endl;//输出为 49
cout << (99 >> 2) << endl;// 24
cout << (999 >> 3) << endl;// 124
cout << (1000 >> 4) << endl;// 62

2、左位移操作符 <<:

左移运算符<<,是将一个数的二进制位全部左移若干位,低位右补0,高位左移后溢出,舍弃。

语法格式: 需要移位的数字 << 移位的次数

例如:
将a的二进制数左移2位,右补0。
若a=15,即二进制数00001111,左移2位得00111100,即十进制数60。

3 << 2,则是将数字3左移2位。即00000011----->00001100(12);
2 << 1:相当于22。
99 << 1: 相当于 99
2 。
99 << 2:相当于 99*pow(2,2)。
999 << i:相当于999 * pow(2,i)。
整数 << i:相当于将这个整数化为二进制整数,并在这个数的末尾加上 i 个0。

cout << (2 << 1) << endl;    //输出为 4
cout << (99 << 1) << endl;    //输出为 198
cout << (99 << 2) << endl;    // 396
cout << (999 << 3) << endl;    // 7992
cout << (1000 << 4) << endl;    // 16000

数字1 << 左移:
1 << i = pow(2,i)
2 << i = pow(2,i) * 2
3 << i = pow(2,i) * 3
x << i = pow(2,i) * x

cout << (1 << 0) << endl;    // 1
cout << (1 << 1) << endl;    // 2
cout << (1 << 2) << endl;    // 4
cout << (1 << 3) << endl;    // 8
cout << (1 << 4) << endl;    // 16

二、逻辑驱动器插拔产生的掩码 dbv.dbcv_unitmask 进行分析的相关代码

1、设备发生改变的消息 WM_DEVICECHANGE

Windows 在设备发生的情况下,如可插拔设备 U盘、移动硬盘、USB串口的插入和弹出,光盘插入和弹出,会有相应的响应,在这里对于编程的识别进行探讨:

Windows 定义了一些控制代码和指令,当添加新设备或介质(如 CD 或 DVD)以及删除现有设备或介质时,Windows 会给所有顶级窗口发送一组默认设备改变的消息 WM_DEVICECHANGE ,在微软官方文档中有具体描述:

检测介质插入或删除

每个 WM_DEVICECHANGE 消息都有描述更改的关联事件,以及提供有关更改的详细信息的结构。 该结构包含与事件无关的标头 DEV_BROADCAST_HDR,后跟与事件相关的成员。 与事件相关的成员描述应用事件的设备。 若要使用此结构,应用程序必须首先确定事件类型和设备类型。 然后,它们可以使用正确的结构执行适当的操作。

当用户将新的 CD 或 DVD 插入驱动器时,应用程序会收到 WM_DEVICECHANGE 消息和 DBT_DEVICEARRIVAL 事件。 应用程序必须检查事件,以确保到达的设备的类型是卷(dbch_devicetype 成员是 DBT_DEVTYP_VOLUME),并且更换会影响介质(dbcv_flags 成员是 DBTF_MEDIA)。

当用户从驱动器中删除 CD 或 DVD 时,应用程序会收到 WM_DEVICECHANGE 消息和 DBT_DEVICEREMOVECOMPLETE 事件。 同样,应用程序必须检查事件,以确保要删除的设备是卷,并且更换会影响介质。

A、微软官方 VC++ 代码:

//C++
#include <windows.h>
#include <dbt.h>
#include <strsafe.h>
#pragma comment(lib, "user32.lib" )void Main_OnDeviceChange( HWND hwnd, WPARAM wParam, LPARAM lParam );
char FirstDriveFromMask( ULONG unitmask );  //prototype/*------------------------------------------------------------------Main_OnDeviceChange( hwnd, wParam, lParam )描述:处理发送到应用程序顶级窗口的 WM_DEVICECHANGE 消息。
--------------------------------------------------------------------*/void Main_OnDeviceChange( HWND hwnd, WPARAM wParam, LPARAM lParam ){PDEV_BROADCAST_HDR lpdb = (PDEV_BROADCAST_HDR)lParam;TCHAR szMsg[80];switch(wParam ){case DBT_DEVICEARRIVAL:// 检查驱动器中是否插入了CD或DVD。if (lpdb -> dbch_devicetype == DBT_DEVTYP_VOLUME)//驱动类型为逻辑卷{PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;if (lpdbv -> dbcv_flags & DBTF_MEDIA){StringCchPrintf( szMsg, sizeof(szMsg)/sizeof(szMsg[0]), TEXT("驱动器 %c :介质已插入。\n"), FirstDriveFromMask(lpdbv ->dbcv_unitmask) );MessageBox( hwnd, szMsg, TEXT("WM_DEVICECHANGE"), MB_OK );}}break;case DBT_DEVICEREMOVECOMPLETE:// 检查是否已从驱动器中弹出CD或DVD。if (lpdb -> dbch_devicetype == DBT_DEVTYP_VOLUME)//驱动类型为逻辑卷{PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;if (lpdbv -> dbcv_flags & DBTF_MEDIA){StringCchPrintf( szMsg, sizeof(szMsg)/sizeof(szMsg[0]), TEXT("驱动器 %c: 介质已弹出。\n" ),FirstDriveFromMask(lpdbv ->dbcv_unitmask) );MessageBox( hwnd, szMsg, TEXT("WM_DEVICECHANGE" ), MB_OK );}}break;default:/*由于其他设备或原因,处理其他 WM_DEVICECHANGE 通知。*/ ;}
}/*------------------------------------------------------------------FirstDriveFromMask( unitmask )
描述
从驱动器号掩码中查找第一个有效的驱动器号。
掩码的格式必须为 bit 0=A、bit 1=B、bit 2=C,依此类推。
当相应的位设置为1时,将定义有效的驱动器号。
返回找到的第一个驱动器号。
--------------------------------------------------------------------*/char FirstDriveFromMask( ULONG unitmask ){char i;for (i = 0; i < 26; ++i){if (unitmask & 0x1)break;unitmask = unitmask >> 1;}return( i + 'A' );
}

B、驱动器号与设备掩码 dbcv_unitmask 的关系

从上面C++代码上可以看到对于位操作的右移指令 >>,从中可以看到驱动器掩码与驱动器号的对应关系:

驱动器号字母值unitmaskunitmaskunitmask
卷号字母10进制数16进制数10进制数二进制数
A6511=2^010^00 =0000000000000001
B6622=2^110^01 =0000000000000010
C6744=2^210^02 =0000000000000100
D6888=2^310^03 =0000000000001000
E691016=2^410^04 =0000000000010000
F702032=2^510^05 =0000000000100000
G714064=2^610^06 =0000000001000000
H7280128=2^710^07 =0000000010000000
I73100256=2^810^08 =0000000100000000
J74200512=2^910^09 =0000001000000000
K754001024=2^1010^10 =000001000000000
L768002048=2^1110^11 =000010000000000
M7710004096=2^1210^12 =000100000000000
N7820008096=2^1310^13 =001000000000000
O79400016384=2^1410^14 =010000000000000
P80800032768=2^1510^15 =100000000000000
Q以此类推

从上表对应关系可以看出,对于驱动器号 A 盘的 unitmask 值,不能再执行右位移>>操作。

在对移动U盘和移动硬盘编写的识别程序中,参照应用微软以上 C++ 代码迁移到 C# 中,可以发现只有一个分区的移动U盘和移动硬盘完全没有问题。但对于移动U盘和移动硬盘有多个分区时,程序就无法完整识别(Windows 11),unitmask 值会随着设备插入出现 2 个值,第一个值对应多分区的第一个盘符值,第二个值就无法用微软提供的程序算法解释了。通过对多分区移动盘的识别发现,如果移动盘符是 H、I、J、K 四个盘,系统给出了 2 个 unitmask 值,第一个值对应的第一个 H 盘,第二个值比对应的最后一个盘符 K 大,取整后就是 K 盘。

当设备被插入/拔出的时候,WINDOWS会向每个窗体发送WM_DEVICECHANGE 消息,当消息的wParam 值等于 DBT_DEVICEARRIVAL 时,表示Media设备被插入并且已经可用;如果wParam值等于DBT_DEVICEREMOVECOMPLETE,表示Media设备已经被移出。

它们的lParam都指向一个 DEV_BROADCAST_HDR结构体,其原形如下:

//C++
typedef struct _DEV_BROADCAST_HDR
{DWORD dbch_size;DWORD dbch_devicetype;DWORD dbch_reserved;
} DEV_BROADCAST_HDR, *PDEV_BROADCAST_HDR;

这个结构体仅仅是一个“头”(HDR),其后还有附加数据,dbch_size表示结构体实例的字节数,当其中的dbch_devicetype字段值等于DBT_DEVTYP_VOLUME时,表示当前设备是逻辑驱动器,且lParam实际上指向的应该是DEV_BROADCAST_VOLUME 结构体实例,DEV_BROADCAST_VOLUME 结构体原形如下:

//C++
typedef struct _DEV_BROADCAST_VOLUME
{DWORD dbcv_size;DWORD dbcv_devicetype;DWORD dbcv_reserved;DWORD dbcv_unitmask;WORD dbcv_flags;
} DEV_BROADCAST_VOLUME, *PDEV_BROADCAST_VOLUME;

其中dbcv_unitmask 字段表示当前改变的驱动器掩码,第一位表示驱动器号A,第二位表示驱动器号B,第三位表示驱动器号C,以此类推…… dbcv_flags 表示驱动器的类别,如果等于1,则是光盘驱动器;如果是2,则是网络驱动器;如果是硬盘、U盘则都等于0

所以,我只需要在程序中捕捉WM_DEVICECHANGE 消息,然后根据具体情况去处理即可。

2、设备状态发生改变 WM_DEVICECHANGE 的改进 C# 代码:

//C# 响应消息声明
public const int WM_DEVICECHANGE = 0x219;//U盘插入后,OS的底层会自动检测到,然后向应用程序发送“硬件设备状态改变“的消息
public const int DBT_DEVICEARRIVAL = 0x8000;  //就是用来表示U盘可用的。一个设备或媒体已被插入一块,现在可用。
public const int DBT_DEVICEQUERYREMOVE = 0x8001;  //审批要求删除一个设备或媒体作品。任何应用程序也不能否认这一要求,并取消删除。
public const int DBT_DEVICEQUERYREMOVEFAILED = 0x8002;  //请求删除一个设备或媒体片已被取消。
public const int DBT_DEVICEREMOVECOMPLETE = 0x8004;  //一个设备或媒体片已被删除。
public const int DBT_DEVICEREMOVEPENDING = 0x8003;  //一个设备或媒体一块即将被删除。不能否认的。
//C#
public static List<char> Volumes = new List<char>();protected override void DefWndProc(ref Message m){if (m.Msg == WM_DEVICECHANGE)//设备发生改变{int wp = m.WParam.ToInt32();//存储设备插/拔/弹if (wp == DBT_DEVICEARRIVAL || wp == DBT_DEVICEQUERYREMOVE || wp == DBT_DEVICEREMOVECOMPLETE || wp == DBT_DEVICEREMOVEPENDING) {DEV_BROADCAST_HDR dbhdr = (DEV_BROADCAST_HDR)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_HDR));if (dbhdr.dbch_devicetype == 2)//驱动类型为DBT_DEVTYP_VOLUME 逻辑卷{DEV_BROADCAST_VOLUME dbv = (DEV_BROADCAST_VOLUME)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_VOLUME));if (dbv.dbcv_flags == 0)//驱动器标识硬盘和U盘{char[] volums = GetVolumes(dbv.dbcv_unitmask);int len = volums.Length;int start = (int)volums[0];int end = (int)volums[len - 1];for (int i = start; i <= end; i++){string disk =((char)i).ToString() + ":";if(wp == DBT_DEVICEARRIVAL) //存储设备插入{//插入设备后的操作}else if (wp == DBT_DEVICEREMOVECOMPLETE) //存储设备拔/弹出{//拔出设备后的操作}} }}   else if (dbhdr.dbch_devicetype == 3)//DBT_DEVTYP_PORT=0x00000003 设备 (串行或并行) {//响应串口并口设备操作} }  } base.DefWndProc(ref m);           }                    /// <summary>根据驱动器掩码返回驱动器号数组</summary>/// <param name="Mask">驱动器掩码</param>/// <returns>返回驱动器号数组</returns>public static char[] GetVolumes(UInt32 Mask){for (int i = 0; i < 32; i++){//uint p = (uint)Math.Pow(2, i);//原参考代码uint p = (uint)Math.Floor(Math.Log(Mask, 2));//算出驱动器掩码 dbcv_unitmask 以2为底的指数,再取整加A的数值就是盘符。//if ((p | Mask) == p)//原参考代码if (i == p){Volumes.Add((char)('A' + i));}}return Volumes.ToArray();}

移动 U盘的驱动类型是 Removable Media,
移动硬盘的驱动类型是 External hard disk media,
本地硬盘的驱动类型是 Fixed hard disk media。

在对以下 C# 代码进行测试,插入后的多分区移动磁盘可以识别盘符,是对系统所有磁盘进行判断,不能对插入、拔出进行响应。

//C#/// <summary>获取U盘和可移动硬盘盘符名称</summary>/// <returns></returns>public static List<string> GetDVN(){List<string> lstdisk = new List<string>();ManagementClass mgtcls = new ManagementClass("Win32_DiskDrive");var disks = mgtcls.GetInstances();foreach (ManagementObject mo in disks){if (mo.Properties["mediatype"].Value == null || mo.Properties["mediatype"].Value.ToString() == "External hard disk media" || mo.Properties["mediatype"].Value.ToString() == "Removable Media"){foreach (ManagementObject diskpartition in mo.GetRelated("win32_diskpartition")){foreach (ManagementObject disk in diskpartition.GetRelated("win32_logicaldisk")){lstdisk.Add(disk.Properties["name"].Value.ToString());}}// continue;}}return lstdisk;}

综合上述代码总结,可以对设备发生变化时进行的相应的程序操作。

这篇关于VC++位移操作>>和<<以及逻辑驱动器插拔产生的掩码dbv.dbcv_unitmask进行分析的相关代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/999739

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

Nginx设置连接超时并进行测试的方法步骤

《Nginx设置连接超时并进行测试的方法步骤》在高并发场景下,如果客户端与服务器的连接长时间未响应,会占用大量的系统资源,影响其他正常请求的处理效率,为了解决这个问题,可以通过设置Nginx的连接... 目录设置连接超时目的操作步骤测试连接超时测试方法:总结:设置连接超时目的设置客户端与服务器之间的连接

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景