SendMessage函数会一点程序的人都知道,它的意思就是向窗口发送消息.向本地进程创建的窗口发送消息.SendMessage的操作会很简单:它只是将指定的窗口过程作为子程序来调用.当窗口过程处理完消息后,会向SendMessage返回一个值。SendMessage再向调用线程返回应该值。
然而,如果一个线程将消息发送给其它线程创建的窗口,SendMessage会复杂很多。Windows会要求创建窗口线程处理消息。因此调用SendMessage不一定会成功。因为我们发送消息有关的数据不在其它线程中,不过你的消息会追加到被调用线程窗口的消息队列中。可能有时你用SendMessage向其它程序成功的发送过消息的经验。对我也有过。因为成功了,所以就没深入一点研究过。
现在用SendMessage修改我我正在编辑文本的记事本的窗口标题
Dim strTitle as String
strTitle="How are you"
Call SendMessage(FindWindow("Notepad","text.txt - Notepad"),WM_SETTEXT,0,strTitle)
咦成功了,stTitle不是在自己的进程空间吗?记事本程序怎么能访问我的地址空间呢?答案是当调用SendMessage时,该函数中的代码要检查是事发送一个WM_SETTEXT消息。如果是,它将地址空间中以0结尾的字符串放到一个内存映射文件中。该内存映射文件将被其它进程所共享。然后它将消息发送到其它进程的线程。当接受线程准备好处理WM_SETTEXT消息时,它确定包含新窗口标题副本的,被共享的内存映射文件在其自己的的地址的空间位置。不过大多数消息不需要这样的处理,只有在进程间发送数据才这样。
那是不是进程间发送消息都会这样建立一个共享的内存映射文件呢,答案是否定的。尤其是在自己定义的消息时。这个时候怎么办呢?一种方法是用WM_COPYDATA消息,一种方法可以用ReadProcessMemory和WriteProcessMemory函数从其它进程读取数据和写入数据
WM_COPYDATA消息是一个特殊的消息同进它与一个特殊的结构体一起使用。来看看这个结构体的形式
Public Type COPYDATASTRUCT
dwData As Long
cbData As Long
lpData As Long
End Type
其中cbData 成员指定了你想要发送给其它进程的字节数,lpData指定要发送的数据的第一个字节地址,dwData成员留给自己用。当SendMessage 看到正在发送的的WM_COPYDATA消息时,它创建一个大小为cbData字节的内存映射文件,并且将数据从你的地址空间复制到该内存映射文件中,然后将消息发送到目标窗口,当接受窗口处理该消息时,lParam参数指向一个存在于接收进程地空间的COPYDATASTRUCT结构。应该结构的lpData成员指向接收进程的地址空间中的共享内存。
现在我们讨论第二种方法,这种方法主要是在其它进程中开避一段内存区域,怎么开避就要用到VirtualAllocEx这个函数了.现在我们用一个实例说明一下,在运行中输入msconfig 打开我们的配置,选择启动项选项卡看看有些什么启动项,现在我就要让全部启动项不不钩选上。
pHandle = OpenProcess(PROCESS_VM_OPERATION Or PROCESS_VM_READ Or PROCESS_VM_WRITE, False, ProcessID)
'先在远程进程中分配内存
pMyItemMemory = VirtualAllocEx(pHandle, 0, Len(myItem), MEM_COMMIT, PAGE_READWRITE)
'做好读取和改变ListView中Item项目属性的工作
myItem.Mask = LVIF_STATE
myItem.iItem = pRow
myItem.StateMask = -1
'把属性写入远程进程
Result = WriteProcessMemory(pHandle, pMyItemMemory, myItem, Len(myItem), szInByte)
'向远程进程发送消息
retValue = SendMessage(hWindow, LVM_GETITEM, 0, ByVal pMyItemMemory)
'把读取的远程进程写入本地缓冲区
Result = ReadProcessMemory(pHandle, pMyItemMemory, myItem, Len(myItem), szOutByte)
'修改读取后的信息
myItem.State = (myItem.State And (Not LVIS_STATEIMAGEMASK)) Or &H1000
'把读取后的信息写入远程进程
Result = WriteProcessMemory(pHandle, pMyItemMemory, myItem, Len(myItem), szInByte)
'向远程进程发送消息
Result = SendMessage(hWindow, LVM_SETITEM, 0, ByVal pMyItemMemory)
'释放分配的远程内存
Result = VirtualFreeEx(pHandle, pMyItemMemory, 0, MEM_RELEASE)
Result = CloseHandle(pHandle)
上面是一段实现这个功能的主要代码。如要代码可以和我联系