关于SendMessage

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)

  上面是一段实现这个功能的主要代码。如要代码可以和我联系

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值