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

相关文章

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

Java利用JSONPath操作JSON数据的技术指南

《Java利用JSONPath操作JSON数据的技术指南》JSONPath是一种强大的工具,用于查询和操作JSON数据,类似于SQL的语法,它为处理复杂的JSON数据结构提供了简单且高效... 目录1、简述2、什么是 jsONPath?3、Java 示例3.1 基本查询3.2 过滤查询3.3 递归搜索3.4

java之Objects.nonNull用法代码解读

《java之Objects.nonNull用法代码解读》:本文主要介绍java之Objects.nonNull用法代码,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录Java之Objects.nonwww.chinasem.cnNull用法代码Objects.nonN

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

python+opencv处理颜色之将目标颜色转换实例代码

《python+opencv处理颜色之将目标颜色转换实例代码》OpenCV是一个的跨平台计算机视觉库,可以运行在Linux、Windows和MacOS操作系统上,:本文主要介绍python+ope... 目录下面是代码+ 效果 + 解释转HSV: 关于颜色总是要转HSV的掩膜再标注总结 目标:将红色的部分滤

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用