本文共 2451 字,大约阅读时间需要 8 分钟。
技术攻关
从今天开始准备P0中所需技术的Demo,有了Demo就知道原理了,正式开始写的时候就没那么头大了.
捕捉窗口的技术
目的:得到当前鼠标位置是哪个窗口.
以前学过一点点Win32,知道在Windows操作系统中,所有你见到的东西--无论是窗体还是控件,都是"窗口",是窗口,就会与一个WindowClass相关联.这个WindowClass,虽然有Class字样,但要注意,Win32编程用的是C语言,所以不是OO中的那个Class.
下班前,照着书写了一个小程序,把代码贴在下面,大家可以自己运行一下.运行的时候,把鼠标四处移动一下.这个程序会以每秒一次的速率捕捉当前鼠标所指的窗口,并将窗口的标题(如果有)输出到命令行.
//------水之真谛
//------http://blog.csdn.net/FantasiaX
using System;
using System.Text;
using System.Threading;
using System.Runtime.InteropServices;
namespace RnP
{
[StructLayout(LayoutKind.Sequential)]
public struct Win32Point
{
public int x;
public int y;
}
class Program
{
[DllImport("user32.dll")]
private static extern bool GetCursorPos( ref Win32Point win32Point);
[DllImport("user32.dll")]
private static extern uint WindowFromPoint(Win32Point win32Point);
[DllImport("user32.dll")]
private static extern int GetWindowText(uint windowHandle, StringBuilder windowText, int maxChars);
static void Main(string[] args)
{
Win32Point p= new Win32Point();
while (true)
{
GetCursorPos(ref p);
uint u = WindowFromPoint(p);
StringBuilder sb = new StringBuilder(20);
GetWindowText(u, sb, 20);
Console.WriteLine(sb.ToString());
Thread.Sleep(1000);
}
}
}
}
技术分析 昨天说来着,最头疼的一件事情是如何对付那些Win32里的数据类型.这些类型除了简单类型外,有很多是结构体.在从Win32到.NET的转换中,要用到MarshalAs这个Attribute,今天还注意到了另一个Attribute,那就是 [StructLayout(LayoutKind.Sequential)], 不过查过MSDN才知道,这个Attribute加不加无所谓--这是默认值. 今天最大的收获在这句里: [DllImport("user32.dll")] private static extern bool GetCursorPos( ref Win32Point win32Point); 我们可以对照一下其Win32的原型: BOOL GetCursorPos( LPPOINT lpPoint ); 这个函数只接受一个参数,也就是LPPOINT,是一个指针类型.原来我想的是:指针吗,一个地址而已,转换成uint类型就行了.可后来想想,这是个于驴的智商相当想法,因为.NET平台(托管代码情况下,不考虑不安全代码)根本不支持指针,所以就算拿到这个以uint表达的指针也没用,因为你无法通过这个"指针"访问其代表的内存.呵呵,切身地开始怀念C语言了...自由与安全永远是冲突的. 遂,从书找到上面的代码,不过我稍微做了一点改动.通过使用 ref Win32Point win32Point 一个参数, .NET平台巧妙地避开(或者说是"跳过")了指针,直接把一个Win32Point类型的参数"灌"进一个变量中了. 是不是可以这样理解: 当从Win32向.NET转换API函数的时候,如果遇到[out]类型指针参数ptr_T时,就使用 ref T 作为参数.明天我可能试着在RECT类型上试验一下:p 刚才在调试的时候,发现这个问题: Win32Point p= new Win32Point(); 必需显式地赋值,也就是说去掉new Win32Point()是不行的.按理说是可以的.因为按照常识,
int x;
与
Int32 x = new Int32();
x=0;
是等效的.
不知道是不是.NET的Bug.
我又试了试改成这
[DllImport("user32.dll")]
private static extern bool GetCursorPos( out Win32Point win32Point); 也就是把ref 改成了out,就可以了.
当然,下面的代码也要改成
........
Win32Point p;//= new Win32Point(); ........ GetCursorPos(out p); 哈哈,这下简单了:以后就记住-------- Win32的[out]参数,对应.NET的out参数就OK啦!
有所得,收工啦!!!
大家周末快乐撒!
本文转自 水之真谛 51CTO博客,原文链接:http://blog.51cto.com/liutiemeng/18759,如需转载请自行联系原作者