基本信息
源码名称:大漠插件C++调用
源码大小:0.05M
文件格式:.rar
开发语言:C/C++
更新时间:2020-08-21
友情提示:(无需注册或充值,赞助后即可获取资源下载链接)
嘿,亲!知识可是无价之宝呢,但咱这精心整理的资料也耗费了不少心血呀。小小地破费一下,绝对物超所值哦!如有下载和支付问题,请联系我们QQ(微信同号):813200300
本次赞助数额为: 1 元×
微信扫码支付:1 元
×
请留下您的邮箱,我们将在2小时内将文件发到您的邮箱
源码介绍
example(多线程模板).rar
example(多线程模板).rar
#include "stdafx.h" #include "script.h" #include "thread_control.h" #include "log.h" void DoWork(long index); unsigned WINAPI SubThread(PVOID pParam); void CheckException(long index); // 我们做暂停和恢复操作,就主要靠这个延时函数,要求脚本所有用到延时的地方,全部用这个,这样我们可以有很多机会去暂停线程 void ScriptDelay(long index,long time) { // 判断是否有暂停和结束标记 if (g_info[index].is_stop) { // 直接退出当前线程,退出前一定要在本线程内解绑,因为绑定模式0和2在线程外解绑的话会失败,导致解绑失败. if (g_info[index].dm) { g_info[index].dm->UnBindWindow(); } _endthreadex(0); } if (g_info[index].is_pause) { g_info[index].thread_state = State_Pause; ThreadNotifyUI_Post(NOTIFY_UPDATE,index); // 如果你想要在暂停时让用户可以操作,那么可以调用EnableBind,但是不要去调用LockInput,LockInput不是用来解除后台的,具体参考LockInput的说明 if (g_info[index].dm) { g_info[index].dm->EnableBind(5); } // 我们暂停的方法是死循环,然后延时,而不是调用系统的接口 // 这样开销最小,并且效率也还不错 while (1) { if (!g_info[index].is_pause) { g_info[index].thread_state = State_Runing; ThreadNotifyUI_Post(NOTIFY_UPDATE,index); // 开启后台 if (g_info[index].dm) { g_info[index].dm->EnableBind(1); } break; } if (g_info[index].is_stop) { // 直接退出当前线程,退出前一定要在本线程内解绑,因为绑定模式0和2在线程外解绑的话会失败,导致解绑失败. if (g_info[index].dm) { g_info[index].dm->UnBindWindow(); } _endthreadex(0); } Sleep(1); } } // 可能暂停,恢复时会让状态错乱,这里再判断一次 if (g_info[index].thread_state != State_Runing) { g_info[index].thread_state = State_Runing; ThreadNotifyUI_Post(NOTIFY_UPDATE,index); // 开启后台 if (g_info[index].dm) { g_info[index].dm->EnableBind(1); } } Sleep(time); } void SetTaskState(long index,const TCHAR * state) { _tcscpy(g_info[index].task_state,state); ThreadNotifyUI_Post(NOTIFY_UPDATE,index); } // 脚本主线程,脚本的主要逻辑在这里处理 unsigned WINAPI MainThread(PVOID pParam) { long index = (long)(DWORD_PTR)pParam; dmsoft * dm; // 初始化当前线程com组件为MTA模式 CoInitializeEx (NULL,0); g_info[index].thread_state = State_Runing; ThreadNotifyUI_Post(NOTIFY_UPDATE,index); // 创建对象 g_info[index].dm = new dmsoft; dm = g_info[index].dm; // 检测对象是否创建成功,虽然这个一般不会失败,但为了程序健壮性考虑还是加上,如果内存吃紧,还是可能会失败 if (dm == NULL || dm->Ver().GetLength() == 0) { Log(_T("对象创建失败")); ThreadNotifyUI_Post(NOTIFY_STOP,index); return 0; } // 开启全局共享字库 dm->EnableShareDict(1); // 其他设置,比如路径等等 dm->SetPath(_T("c:\test_game")); // 开始绑定,主绑定一定要第一个绑定,并且主绑定所在的线程绝对不能结束,否则会造成绑定失效 long dm_ret = dm->BindWindowEx(g_info[index].hwnd,_T("normal"),_T("normal"),_T("dx"),_T("dx.public.anti.api"),0); if (dm_ret != 1) { Log(_T("主:绑定失败,错误码:%d"),dm->GetLastError()); // 通知主线程进行结束操作(释放资源) ThreadNotifyUI_Post(NOTIFY_STOP,index); return 0; } // 禁止输入 dm->LockInput(4); // 我们可以创建副线程了 // 更新副线程信息 g_info[index MAX_HWND].thread_state = State_Starting; // 通知UI,副线程开始创建了 ThreadNotifyUI_Post(NOTIFY_UPDATE,index MAX_HWND); // 创建副线程 g_info[index MAX_HWND].handle = (HANDLE)_beginthreadex(0, 0, SubThread, (PVOID)(DWORD_PTR)(index MAX_HWND), 0, 0); if (g_info[index MAX_HWND].handle == NULL) { // 这里必须要解绑,因为模式0和2不在当前线程解绑的话,会导致无法解绑 dm->UnBindWindow(); Log(_T("创建副线程失败")); // 通知主线程进行结束操作(释放资源) ThreadNotifyUI_Post(NOTIFY_STOP,index); return 0; } while (1) { SetTaskState(index,_T("开始做任务")); DoWork(index); SetTaskState(index,_T("任务完成")); ScriptDelay(index,1000); } } void DoWork(long index) { dmsoft * dm = g_info[index].dm; dm->KeyPressChar(_T("m")); ScriptDelay(index,500); } void SetExcepState(long index,const TCHAR * state) { _tcscpy(g_info[index].excep_state,state); ThreadNotifyUI_Post(NOTIFY_STOP,index); } // 脚本副线程,用于检测异常等 unsigned WINAPI SubThread(PVOID pParam) { long index = (long)(DWORD_PTR)pParam; dmsoft * dm; // 初始化当前线程com组件为MTA模式 CoInitializeEx (NULL,0); g_info[index].thread_state = State_Runing; ThreadNotifyUI_Post(NOTIFY_UPDATE,index); // 创建对象 g_info[index].dm = new dmsoft; dm = g_info[index].dm; // 检测对象是否创建成功,虽然这个一般不会失败,但为了程序健壮性考虑还是加上,如果内存吃紧,还是可能会失败 if (dm == NULL || dm->Ver().GetLength() == 0) { Log(_T("对象创建失败")); SetExcepState(index,_T("对象创建失败")); return 0; } // 开启全局共享字库 dm->EnableShareDict(1); // 其他设置,比如路径等等 dm->SetPath(_T("c:\test_game")); long dm_ret = dm->BindWindowEx(g_info[index].hwnd,_T("normal"),_T("normal"),_T("dx"),_T("dx.public.anti.api"),0); if (dm_ret != 1) { Log(_T("副:绑定失败,错误码:%d"),dm->GetLastError()); // 通知主线程进行结束操作(释放资源) SetExcepState(index,_T("副:绑定失败")); return 0; } while (1) { // 检测一些异常,比如突然弹出的对话框,目标窗口被关闭或者掉线等突发情况 // 比如检测到掉线,可考虑通知UI,然后重新运行 CheckException(index); ScriptDelay(index,3000); } } void CheckException(long index) { dmsoft * dm = g_info[index].dm; // 检测窗口是否存在 if (dm->GetWindowState(g_info[index].hwnd,0) == 0) { SetExcepState(index,_T("窗口不见了")); } // 检测窗口是否卡死 // 检测是否掉线 // 检测是否有弹出窗口 // 其他检测 dm->KeyPressChar(_T("s")); }