Clipboard中的延迟提交技术

来源:互联网 发布:彼得原理 知乎 编辑:程序博客网 时间:2024/04/30 12:48

当把数据放入剪贴簿中时,一般来说要制作一份数据的副本,并将包含这份副本的内存块句柄传给剪贴簿。对非常大的数据项来说,这种方法会浪费内存空间。如果使用者不想把数据粘贴到另一个程序里,那么,在被其它内容取代之前,它将一直占据着内存空间。

通过使用一种叫做「延迟提出」的技术可以避免这个问题。实际上,直到另一个程序需要数据,程序才提供这份数据。为此,不将数据句柄传给Windows,而是在SetClipboardData呼叫中使用NULL:

OpenClipboard              (hwnd) ;        EmptyClipboard             () ;        SetClipboardData   (iFormat, NULL) ;        CloseClipboard             () ;        

可以有多个使用不同iFormat值的SetClipboardData呼叫,对其中某些呼叫可使用NULL值。而对其他一些则使用实际的句柄值。

前面的过程比较简单,以下的过程就要稍微复杂一些了。当另一个程序呼叫GetClipboardData时,Windows将检查那种格式的句柄是否为NULL。如果是,Windows将给「剪贴簿所有者」(您的程序)发送一个消息,要求取得数据的实际句柄,这时您的程序必须提供这个句柄。

更具体地说,「剪贴簿所有者」是将数据放入剪贴簿的最后一个窗口。当一个程序呼叫OpenClipboard时,Windows储存呼叫这个函数时所用的窗口句柄,这个句柄标示打开剪贴簿的窗口。一旦收到一个EmptyClipboard呼叫,Windows就使这个窗口作为新的剪贴簿所有者。

使用延迟提出技术的程序在它的窗口消息处理程序中必须处理三个消息:WM_RENDERFORMAT、WM_RENDERALLFORMATS和WM_DESTROYCLIPBOARD。当另一个程序呼叫GetClipboardData时,Windows给窗口消息处理程序发送一个WM_RENDERFORMAT消息,wParam的值是所要求的格式。在处理WM_RENDERFORMAT消息时,不要打开或清空剪贴簿。为wParam所指定的格式建立一个整体内存块,把数据传给它,并用正确的格式和相应句柄呼叫SetClipboardData。很明显地,为了在处理WM_RENDERFORMAT时正确地构造出此数据,需要在程序中保留这些信息。当另一个程序呼叫EmptyClipboard时,Windows给您的程序发送一个WM_DESTROYCLIPBOARD消息,告诉您不再需要构造剪贴簿数据的信息。您的程序不再是剪贴簿的所有者。

如果程序在它自己仍然是剪贴簿所有者的时候就要终止执行,并且剪贴簿上仍然包含着该程序用SetClipboardData设定的NULL数据句柄,它将收到WM_RENDERALLFORMATS消息。这时,应该打开剪贴簿,清空它,把数据加载内存块中,并为每种格式呼叫SetClipboardData,然后关闭剪贴簿。WM_RENDERALLFORMATS消息是窗口消息处理程序最后收到的消息之一。它后面跟有WM_DESTROYCLIPBOARD消息(由于已经提出了所有数据),然后是正常的WM_DESTROY消息。

如果您的程序只能向剪贴簿传输一种格式的数据(例如文字),那么您可以把WM_RENDERALLFORMATS和WM_RENDERFORMAT处理结合在一起。这些程序代码应该类似下面这样:

case        WM_RENDERALLFORMATS :                   OpenClipboard (hwnd) ;                   EmptyClipboard () ;                                                                 // fall through        case        WM_RENDERFORMAT :           // 将文字放入整体内存块                   SetClipboardData (CF_TEXT, hGlobal) ;                   if (message == WM_RENDERALLFORMATS)                          CloseClipboard () ;                   return 0 ;        

如果您的程序使用好几种剪贴簿格式,那么您可能想为wParam所要求的格式处理WM_ RENDERFORMAT。除非程序在存放构造数据所需的信息时遇到困难,否则不需要处理WM_DESTROYCLIPBOARD消息。

原创粉丝点击