仿91助手的PC与android手机通讯

原文

  知道91助手和豌豆莢吧? 说到这两个东西,最让人好奇的应该是就是和手机的交互了。我之前有研究过电脑和安卓的交互,基本功能已经走通了,在这里我想分享一下。 初初看这个问题觉得很简单,然后如果你有点计算机基础的话深入想一下却发现有很多实现上的空白。

---------------------------------------------------------上面是废话,进入正题。

一、检测设备插入

首先要解决的问题就是自动在设备插入电脑时作出响应。
这个问题得解决方法就是:WM_DEVICECHANGE 事件。
这个事件是一个全局事件,不需要预先向系统注册之类的,只要你的程序有窗口就能随时响应设备变更事件WM_DEVICECHANGE,当然全局事件是进程的Top-Level窗口才能收的到,如果你想在子窗口或者模态窗口中直接收到的话可以考虑使用RegisterDeviceNotification注册一下。
事件的WPARAM参数包含了设备更新的类型,设备变更(Device Change)有很多种情况的嘛,有插入、移除、驱动安装成功神马神马的。类型种类在MSDN中有说,如下:
  • DBT_CONFIGCHANGECANCELED
  • A request to change the current configuration (dock or undock) has been canceled.
  • (设备设置取消,我还没怎么研究过这个设置的问题,不过这个跟我现在讲的主题都没关系的。)
  • DBT_CONFIGCHANGED
  • The current configuration has changed, due to a dock or undock.
  • (设备设置变更)
  • DBT_CUSTOMEVENT
  • A custom event has occurred.
  • (这个只是告诉你,设备驱动发出了一个消息)
  • DBT_DEVICEARRIVAL
  • A device or piece of media has been inserted and is now available.
  • (设备或者多媒体插入)
  • DBT_DEVICEQUERYREMOVE
  • Permission is requested to remove a device or piece of media. Any application can deny this request and cancel the removal.
  • (用戶请求弹出设备,返回TRUE允许弹出,返回BROADCAST_QUERY_DENY拒绝弹出,这就是为什么有些时候会发现U盘死活弹不出,非得强拔,这是因为有些进程一直拒绝弹出。)
  • DBT_DEVICEQUERYREMOVEFAILED
  • A request to remove a device or piece of media has been canceled.
  • (请求弹出失败)
  • DBT_DEVICEREMOVECOMPLETE
  • A device or piece of media has been removed.
  • (请求成功)
  • DBT_DEVICEREMOVEPENDING
  • A device or piece of media is about to be removed. Cannot be denied.
  • (强制弹出U盘,这个在360的强制弹出USB时会收到的了)
  • DBT_DEVICETYPESPECIFIC
  • A device-specific event has occurred.
  • (这个是某些个性设备自定义的消息的方法了,自定义部分在LPARAM指针指向的区域中)
  • DBT_DEVNODES_CHANGED
  • A device has been added to or removed from the system.
  • (DevNodes就是设备管理器里面显示的那棵树的节点,这个跟DBT_DEVICEARRIVAL有一点点区别,因为add有可能是因为你新装了某些驱动产生的消息。另外提个醒SAMSUNG手机插入时就是很扑街的收不到DBT_DEVICEARRIVAL类型,只能收到这个。。。)
  • DBT_QUERYCHANGECONFIG
  • Permission is requested to change the current configuration (dock or undock).
  • (请求修改设备设置)
  • DBT_USERDEFINED
  • The meaning of this message is user-defined.
  • (这个类型主要是给用户一个自定义的方法,上面DBT_DEVICETYPESPECIFIC是设备自定义的,这个主要是进程通过BroadcastSystemMessage 来广播的,我还没怎么用过这个玩意。)

好吧,检测设备插入的问题解决了,后面是判断设备是否是手机了。

二、检测是否是手机

上一篇日志说了如何响应设备插入,但是设备有很多中,多媒体设备,鼠标键盘什么的都是,那如何判断是不是USB设备或者是手机插入呢? 这里就介绍一下我自己的研究结果,当然我没有去研究过苹果设备,但是按道理是类似的。

这里是我自己本人的思路,不一定是最好的方法,如果发现更好的方法我会再拿出来,当然如果你发现更好的方法的话可以留言告诉我。
我的思路是枚举USB设备,并且检查设备的兼容ID和硬件ID
枚举用到的方法包括 SetupDiGetClassDevs 和 SetupDiEnumDeviceInfo, 这两个方法可以在MSDN2008里面查到。
这里简单说明一下用法:

//获取设备信息句柄

HDEVINFO hDevInfo = SetupDiGetClassDevs(NULL,L"USB" ,NULL,DIGCF_ALLCLASSES|DIGCF_PRESENT);

//获取设备信息数据
SP_DEVINFO_DATA deviceInfoData;
deviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
for(int i = 0;SetupDiEnumDeviceInfo(hDevInfo,i,&deviceInfoData);i++) //对USB设备集进行枚举
{
....
}

上面是枚举USB设备的思路,下面讲一下如何分辨USB设备是否就是手机
这里我的思路是判断 兼容ID 和 硬件ID , 这两个ID可以在设备管理器中看得到如下(写这篇日志时我没有android手机,所以我截的图不是手机信息,只是告诉你这么一样东西,你可以自己插入手机试验一下):
 
 
有一部分手机直接判断 兼容ID 是否是"usb\\class_ff&subclass_42"即可,
但是有些手机的兼容ID不是这个串,那就比较麻烦了,需要匹配 硬件ID ,但是 硬件ID
不是固定的,好像是跟手机的硬件有关,但是同一款手机型号是一样的,有些品牌如vivo是一个系列都基本一样的硬件ID,甚至现在市面上的很多山寨手机直
接就是用HTC的一个硬件ID(哈哈,那时候我去手机店采集硬件ID的时候就感概现在的山寨不硬件识别的ID都直接copy了,这个可能是商业问题,我也
不懂.)
获取兼容ID的方法,使用上面枚举获得的HDEVINFO句柄和SP_DEVINFO_DATA数据调用SetupDiGetDeviceRegistryProperty方法:

WORD dataType= 0;

DWORD buffSize = 0;

SetupDiGetDeviceRegistryProperty(hDevInfo,&deviceInfoData,SPDRP_COMPATIBLEIDS,&dataType,NULL,buffSize,&buffSize);

int err = GetLastError();

if(err != ERROR_INSUFFICIENT_BUFFER)

return;

LPTSTR szCompatibleID = (LPTSTR)LocalAlloc(LPTR,buffSize+1);

SetupDiGetDeviceRegistryProperty(hDevInfo,&deviceInfoData,SPDRP_COMPATIBLEIDS,&dataType,(PBYTE)szCompatibleID ,buffSize,&buffSize);

//szCompatibleID 即是兼容ID

如果 兼容ID == "usb\\class_ff&subclass_42" 就直接可以知道这个是手机设备了(注意要兼容ID的大小写不确定的)

如果 兼容ID != "usb\\class_ff&subclass_42" 那么就要匹配硬件ID了,获取硬件ID的方法和获取兼容ID的方法类似:

WORD dataType= 0;

DWORD buffSize = 0;

SetupDiGetDeviceRegistryProperty(hDevInfo,&deviceInfoData,SPDRP_HARDWAREID,&dataType,NULL,buffSize,&buffSize);

int err = GetLastError();

if(err != ERROR_INSUFFICIENT_BUFFER)

return;

LPTSTR szHardwareID = (LPTSTR)LocalAlloc(LPTR,buffSize+1);

SetupDiGetDeviceRegistryProperty(hDevInfo,&deviceInfoData,SPDRP_HARDWAREID,&dataType,(PBYTE)szHardwareID ,buffSize,&buffSize);

//szHardwareID 即是硬件ID

硬件ID 的样子大概是: VID_1234&PID_4321  (1234,4321根据设备有差异,其中VID代表Vendor ID(厂家ID) , PID代表Product ID(产品ID)),VID基本一个厂商. 这个需要收集,但是网上好像有一些VID_PID大全可以满足一般使用.
(其中VID只能判断厂商,有些手机生产商也有别的设备产品,好像索尼有手机也有相机,所以不能轻判哦.

OK,手机判断完成,后面是驱动安装的介绍,手机插入后不一定有驱动,需要有驱动才能进行PC操作手机的功能.

三、检测是否已经安装驱动

上一篇日志说到判断是否是手机设备,但是要与手机进行通讯就必须有驱动程序,否则只能当做“便携储存设备”使用,只能往里面放文件,也许你已经满足了,但 是你想一下91助手只是给你提供存放文件那么简单吗?如果是的话91助手还有鸟用啊?因为我们直接打开“我的电脑”就能打开这个类似U盘的东西了.  再想一想,如果你的程序可以跟手机说"我给个apk你,你安装一下",然后你的手机就装上去了,那不就方便了吗? 这才是卖点~

好,上面都是废话----------------------------------------------------------------------------------------------------------------------
上一篇日志中说到的枚举过程只是判断是否有手机的话似乎太浪费了. 所以这里要提前交代一下再枚举过程发现了手机设备要记录下该设备的
"实例ID" ,这个和"兼容ID"、"硬件ID"不一样,前者是电脑给设备分配的ID便于电脑对I/O设备的管理,而后两者是设备本身的属性信息。
获取实例ID的方法:
	LPTSTR szInstanceID = NULL

WORD iBuffSize = 0;

SetupDiGetDeviceInstanceId(hDevInfoSet,&deviceInfoData,szInstanceID ,iBuffSize,&iBuffSize ); //获取实例ID的buff需要的大小, hDevInfoSet和deviceInfoData

int err = GetLastError();

if(err != ERROR_INSUFFICIENT_BUFFER)

return;

szInstanceID = (LPTSTR)LocalAlloc(LPTR,buffSize*sizeof(WCHAR)); SetupDiGetDeviceInstanceId(hDevInfoSet,&deviceInfoData,szInstanceID ,iBuffSize ,&iBuffSize ); //获取实例ID

要判断设备是否有安装驱动使用到两个方法:CM_Locate_DevNode 和 CM_Get_DevNode_Status ,头文件#include <cfgmgr32.h>,主要实现如下:

DEVINST deviceInstance;

if (CM_Locate_DevNode(&deviceInstance,szInstanceID ,CM_LOCATE_DEVNODE_NORMAL) == CR_SUCCESS) //获取设备ID对应的设备实例句柄

{

DWORD tatus;

DWORD problemNumber;

if (CM_Get_DevNode_Status(&status,&problemNumber,deviceInstance,0) == CR_SUCCESS)

//获取设备状态和设备状态细节

{

if (!(status&DN_HAS_PROBLEM)) //判断设备是否存在问题,代表驱动已安装

{

//设备无异常,就是说驱动正常

}

else

{

if (problemNumber == CM_PROB_DRIVER_FAILED_PRIOR_UNLOAD

|| problemNumber == CM_PROB_DRIVER_FAILED_LOAD)

{

//设备驱动加载不成功

}

else

{

//有不明原因,可以归结为没安装驱动

}

}

}

}

好吧,驱动是否安装的判断就这样子。一点都不麻烦。

四、自动安装手机驱动

  上一节讲到检查驱动安装情况,那么如果遇到没安装手机驱动的话是没办法和手机进行通讯的(除非你是要直接把文件拷贝到手机目录下,好像txt,视频,音乐的话是不用考虑驱动都可以的,当然有驱动这几种文件的拷贝也会是更方便的。)

好吧,开始说一下安装驱动的实现吧:
首先驱动也是分厂商和机型的(当然好像是有万能驱动这个东西的,但是我测试过万能驱动不是完全适合所有手机的),那么说到厂商和机型,应该就会想到VID
和PID了,前面说过VID代表厂商PID代表型号。
那么就知道用什么来匹配驱动了,当然说到匹配的话就说明驱动有很多,虽然有些厂商的所有机型或者某一系列的机型是使用同一个驱动就行了,但是也有很多例外
的(这就说明,自己弄一个仿91助手的东西还是做来自己玩玩的,要做成商业软件的话你还得去收集驱动呢。)
在这里说一下题外话,怎么收集驱动呢? 官网?手机自带光碟?
 这些方法都很蛋疼的,你自己去实践一下就知道了,除非有专门人员帮你收集,否则真的很蛋疼。而我以前的收集驱动的方法就是用豌豆荚插手机,然后豌豆荚会
在C盘的临时文件夹中存放该VID/PID对应的驱动准备安装,然后拷贝下来(-
-),等驱动安装完成豌豆荚会把驱动文件删除,所以你要在豌豆荚询问“是否安装驱动”的时候别确定也别取消,然后去拷贝驱动吧,这里你可能需要用到一个辅
助软件(Everything)帮你定位这个临时文件夹,这个是一个本地文件搜索软件,搜索速度别拿windows的来比,因为windows那个根本没
法比,那个快是瞬间~~~,十分high。
----------想到就心酸的操作。
在收集驱动的过程中,我发现了驱动有两种,一种就是exe的例如SAMSUNG的就是这样,另外一种就是dll的,如下图:
1、exe类型:
 
2、dll类型,有两层目录,首层是.inf硬件信息文件和.cat安全文件,次层是.dll动态链接库文件:
 
 
-------------------------------------------------------------------------------------------
驱动类型和匹配的说完,现在该说如何安装了吧。
第一种类型,exe文件直接运行就行,可以用WinAPI中的CreateProcess()来启动,这个用户交互会多一点,因为会很多“确
认”“下一步”的东西,但是这个我们无法控制,用91助手的时候你也会遇到有些驱动就是要点下一步、下一步,有些就是确定要安装后就后台静默安装了。
第二种类型,用户基本不需要交互,直接后台默认就可以完成安装,我们主要也是要处理这一种。

主要是用到UpdateDriverForPlugAndPlayDevices(HWND hwndParent,LPCWSTR HardwareId,LPCWSTR FullInfPath,DWORD
InstallFlags,PBOOL bRebootRequired);这
个API,不敢相信还有那么长名称的API。这个API第一个参数可以为空句柄,第二个参数是硬件ID(这个在第二节讲过,和兼容ID一起的那个),第三
个是.inf文件的全路径(就是上面第二张图里面那个),第四个参数填0就行(有需要的可以再详细研究这个参数值),第四个要传一个BOOL型的指针进去
等API返回一个是否要重启的值告诉你这个驱动安装后要真正运作起来是否要重启。

五、使用adb获取手机信息

到这里,我知道的就差不多了。后面就是跟android手机的命令传递了。这些操作主要使用到android工具包---adb(android debug bridge)。这个东西是google提供的,网上有大量的教程,使用起来很简单。

介绍一下获取android手机信息的基本流程:
 WinExec("adb -d devices",SW_HIDE); Sleep();

SECURITY_ATTRIBUTES sa;  

sa.nLength = sizeof(SECURITY_ATTRIBUTES);  

char buffer[] = {}; //用1K的空间来存储输出的内容,只要不是显示文件内容,一般情况下是够用了。 

DWORD recvLen;  

DWORD occupyLen = ;

TCHAR command[] = _T("adb -d shell getprop ro.product.brand");  //获取厂商名称

                                //_T("adb -d shell getprop ro.product.model")    //设备型号
//_T("adb -d shell getprop ro.build.version.release") //android版本
//_T("adb -d shell dumpsys iphonesubinfo"} //IMEI码
//_T("adb -d shell cat /sys/class/net/wlan0/address") //MAC地址 HANDLE hRead,hWrite; if (!CreatePipe(&hRead,&hWrite,&sa,)) return ; PROCESS_INFORMATION pi; STARTUPINFO si; si.cb = sizeof(STARTUPINFO); GetStartupInfo(&si); si.wShowWindow = SW_HIDE; si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES; si.hStdError = hWrite; //把子进程的标准错误输出重定向到管道输入 si.hStdOutput = hWrite; //把子进程的标准输出重定向到管道输入 si.hStdInput = hRead; //把子进程的标准输入重定向到管道输出 TCHAR command[] = _T("adb -d shell getprop ro.product.brand"); //获取厂商名称 if (! CreateProcess(NULL, command,NULL,NULL,TRUE,NULL,NULL,NULL,&si,&pi)) // 启动进程以调用ADB { CloseHandle(hWrite); CloseHandle(hRead); return FALSE; } CloseHandle(hWrite); if(WaitForSingleObject(pi.hProcess,) == WAIT_TIMEOUT) //800ms的处理等待时间. { TerminateProcess(pi.hProcess,WAIT_TIMEOUT); CloseHandle(pi.hProcess); return FALSE; } //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// if(ReadFile(hRead,buffer,,&recvLen,NULL)) //IMEI码的要做特殊处理。 { CStringA strIMEI = buffer; if(strIMEI.Find("error:") == -) { strIMEI = strIMEI.Mid(strIMEI.FindOneOf("=")+); strcpy_s(buffer,,strIMEI.GetBuffer()); strIMEI.ReleaseBuffer(); occupyLen=strIMEI.GetLength();} } } /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ReadFile(hRead,buffer,,&occupyLen,NULL); //其他信息直接返回读取到的东西就行。 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// buffer[occupyLen-] = '\0'; CloseHandle(hRead); CloseHandle(pi.hThread); CloseHandle(pi.hProcess); TerminateProcess(pi.hProcess,);

回到目录

[转]仿91助手的PC与android手机通讯的更多相关文章

  1. 简单实现Android手机“全局可调试”(ro.debuggable = 1)的方法【锤子坚果3】

    在Android真机上调试程序有一个前提,就是这个apk包必须有 debuggable=true 的属性才行.而除了自己开发的apk能够控制打包属性之外,其他的程序发行之后显然不会设这个值为 true ...

  2. ORB-SLAM2 运行 —— ROS + Android 手机摄像头

    转载请注明出处,谢谢 原创作者:Mingrui 原创链接:https://www.cnblogs.com/MingruiYu/p/12404730.html 本文要点: ROS 配置安装 解决 sud ...

  3. 借助91助手,将ibook中的pdf文件拷贝至其它的pdf阅读器中(ios设备无需越狱)

    有时候在使用ios自带的ibook阅读pdf文件的时候,会发现ibook有些功能并不是那么方便.最近我就遇到了一例,我想在ibook中放一本比较大的pdf书,页数有几百吧,pdf文件本身每一章节都是有 ...

  4. Droid@screen:在PC屏幕上显示Android手机屏幕

    这里介绍一款工具——Droid@screen,用来获取手机屏幕,显示在PC屏幕上.它集截图.录像等多种功能于一体. 安装 1.    下载地址:http://droid-at-screen.org/d ...

  5. android手机连接PC无法正常安装驱动

    工作当中我们经常会遇到Android手机连接PC的时候无法正确安装驱动,或者安装失败.当然找到正确的驱动文件时首选的解决方案,如果正确的驱动文件依旧无法安装成功我们可以打开我的电脑-->属性-- ...

  6. 网络编程之PC版与Android手机版带断点续传的多线程下载

    一.多线程下载         多线程下载就是抢占服务器资源         原理:服务器CPU 分配给每条线程的时间片相同,服务器带宽平均分配给每条线程,所以客户端开启的线程越多,就能抢占到更多的服 ...

  7. system CPU占用率过高与91助手的关系

    今天正在认真工作,忽然发现电脑越来越慢. 按 CTRL+ALT+DEL打开任务管理器看了下CPU使用率.其中system占用率居然达到了64%.不对劲儿,按照平时习惯,system根本占用了不要这么多 ...

  8. HTML5网页录音和上传到服务器,支持PC、Android,支持IOS微信

    准备做一个网页版聊天界面,表情啊.图片啊.上传文件啊都应该要有,视频就算了,语音还是要的. 本文记录的是在网页上用GitHub上的Recorder进行在线录音和上传到服务器,前几天升了一下级,以后有时 ...

  9. [Android] Android 手机下 仿 微信 客户端 界面 -- 微聊

    Android 手机下 仿 微信 客户端 界面 -- 微聊 (包括聊天列表 + 聊天对话页 + 朋友圈列表页 + 我的/发现 列表页) 项目演示: 功能说明: 1)底部标签切换 (TabHost + ...

随机推荐

  1. 前端常用功能记录(二)—datatables表格(转)

    前端常用功能记录(二)—datatables表格 并不是所有的后台开发都有美工和前端工程师来配合做页面,为了显示数据并有一定的美感,jQuery的DataTables插件对于像我这样的前端菜鸟来说真是 ...

  2. C语言 · 十进制数转八进制数

    算法训练 十进制数转八进制数   时间限制:1.0s   内存限制:512.0MB      编写函数把一个十进制数输出其对应的八进制数. 样例输入 9274 样例输出 22072   #includ ...

  3. <花荣《至尊狐狸》中国股市精英最优套利战术>读书笔记

    书在这里 第一定律:博弈分析是一切实战操作的基础,资金的机会利润由主力投机状态决定,资金的风险承受度由投资标的价值底线锁定 第二定律:套利战术应是背景性的.主动性的.互动性的,是最安全最保守的.最明确 ...

  4. <跟股市谚语学炒股> 读书笔记

    书在这里 一般情况下,当成交清单上显示的买盘金额大.笔数少,卖盘金额小.笔数多时,系主力在建仓.散户在卖出:相反,若买盘金额小.笔数多,卖盘金额大.笔数少时,系主力在出货.散户在买入 一般来说,当大盘 ...

  5. ActiveMQ两种模式PTP和PUB/SUB<转>

    1.PTP模型 PTP(Point-to-Point)模型是基于队列(Queue)的,对于PTP消息模型而言,它的消息目的是一个消息队列(Queue),消息生产者每次发送消息总是把消息送入消息队列中, ...

  6. Vagrant (1) —— 基本安装与配置(上)

    Vagrant (1) -- 基本安装与配置(上) 摘要 基本安装与配置 版本 Vagrant版本: 1.8.1 内容 启动运行 $ vagrant init hashicorp/precise64 ...

  7. 记录一下寄几个儿的greendao数据库升级,可以说是非常菜鸡了嗯

    之前使用的greendao数据库存储服务器所有的历史推送消息,但是后来消息需要加几个新的字段 举个栗子,比如要新增红色框住的字段到数据库中: 本仙女作为一只思想成熟的菜鸡,当然是加了字段就赶紧重新往里 ...

  8. js文档碎片

    //文档碎片:类似一个临时的文档,要所有要加的dom元素先放在这里,达到不要每次操作dom元素提高页面效率 var d1 = new Date(); //创建十个段落,常规的方式 ; i < ; ...

  9. Sword redis补充

    Redis 键(key) Redis 键命令用于管理 redis 的键. redis任何数据类型都有key --删除key的命令 redis> del key Redis 事务 Redis 事务 ...

  10. 【转】Android下使用配置文件(Preferences)

    http://www.aslibra.com/blog/post/android_SharedPreferences.php android下可以方便的使用key-value的配置文件: // S.P ...