1,594
社区成员
发帖
与我相关
我的任务
分享
谢谢大家的关心,问题还是自己解决了。
想太复杂了。
手动消息处理只有在循环可以继续的情况下使用没问题
但我用的线程处理的是硬件数据接收,一直处于等待消息状态,直到有数据上来才会继续,所以在等待的时候作不了其他操作,运行不了那个函数。
公布下解决办法 :
线程中重载Create函数
constructor Create();
函数中间定义ADODataSet的创建或者创建窗口都行
begin
inherited Create(False);
//此处创建动态窗体
end;
Execute中进行调用操作。
以前都是这样操作的,没出过什么问题,这次图简单,将创建的过程放到了Execute中,才出现了怪现象 ,呵呵。以前也碰到过Execute声明的递增变量偶尔会不按递增规则来,后来线程中只好在上面声明全局变量才放心使用。
欢迎大家帮我解释原理。


function TCFADODBStorage.OpenConnection: Boolean;
//线程里打开ADOConnection,在XP下切换输入法程序会死掉
//原因:ADO自动创建一个ADODB.AsyncEventMessenger窗口,然后会有一个对应的IME窗口,但线程里没有消息循环
// XP下输入法切换时会SendMessage给IME窗口并等待返回,IME窗口并不会处理消息,造成死锁
//这个函数把线程里的IME窗口释放掉,切换输入法时就不会有消息过来了
procedure _FreeIMEWindow;
const
IME_WINDOW_CLASS = 'IME';
IME_WINDOW_TEXT = 'Default IME';
var
h : HWND;
pid : DWORD;
dh : HWND;
begin
h := FindWindow(IME_WINDOW_CLASS, IME_WINDOW_TEXT);
while IsWindow(h) do
begin
if GetWindowThreadProcessId(h, pid) = GetCurrentThreadId then
dh := h
else
dh := 0;
h := FindWindowEx(0, h, IME_WINDOW_CLASS, IME_WINDOW_TEXT);
if dh <> 0 then
DestroyWindow(dh);
end;
end;
begin
try
FConnection.ConnectionString := ConnectionString;
FConnection.Connected := True;
if GetCurrentThreadId <> Global.MainThreadID{全局的主线程ID,如果是主线程,不需要Free} then
_FreeIMEWindow;
except on E: Exception do
FLastErrorMessage := e.Message;
end;
Result := FConnection.Connected;
end;