typedef
創(chuàng)新互聯(lián)公司專注于企業(yè)營銷型網(wǎng)站建設(shè)、網(wǎng)站重做改版、新都網(wǎng)站定制設(shè)計(jì)、自適應(yīng)品牌網(wǎng)站建設(shè)、H5高端網(wǎng)站建設(shè)、商城網(wǎng)站開發(fā)、集團(tuán)公司官網(wǎng)建設(shè)、外貿(mào)網(wǎng)站建設(shè)、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁設(shè)計(jì)等建站業(yè)務(wù),價(jià)格優(yōu)惠性價(jià)比高,為新都等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。
struct
{
int
a;
void
(*pshow)(int);
}TMP;
void
func(TMP
*tmp)
{
if(tmp-a
10)//如果a10,則執(zhí)行回調(diào)函數(shù)。
{
(tmp-pshow)(tmp-a);
}
}
void
show(int
a)
{
printf("a的值是%d\n",a);
}
void
main()
{
TMP
test;
test.a
=
1;
test.pshow
=
show;
func(test);
}
這只是舉例,一般回調(diào)函數(shù)的用法為:
甲方進(jìn)行結(jié)構(gòu)體的定義(成員中包括回調(diào)函數(shù)的指針)
乙方定義結(jié)構(gòu)體變量,并向甲方注冊,
甲方收集N個(gè)乙方的注冊形成結(jié)構(gòu)體鏈表,在某個(gè)特定時(shí)刻遍歷鏈表,進(jìn)行回調(diào)。
callback Function
回調(diào)函數(shù)是應(yīng)用程序提供給Windows系統(tǒng)DLL或其它DLL調(diào)用的函數(shù),一般用于截獲消息、獲取系統(tǒng)信息或處理異步事件。應(yīng)用程序把回調(diào)函數(shù)的地址指針告訴DLL,而DLL在適當(dāng)?shù)臅r(shí)候會調(diào)用該函數(shù)?;卣{(diào)函數(shù)必須遵守事先規(guī)定好的參數(shù)格式和傳遞方式,否則DLL一調(diào)用它就會引起程序或系統(tǒng)的崩潰。通常情況下,回調(diào)函數(shù)采用標(biāo)準(zhǔn)WindowsAPI的調(diào)用方式,即__stdcall,當(dāng)然,DLL編制者可以自己定義調(diào)用方式,但客戶程序也必須遵守相同的規(guī)定。在__stdcall方式下,函數(shù)的參數(shù)按從右到左的順序壓入堆棧,除了明確指明是指針或引用外,參數(shù)都按值傳遞,函數(shù)返回之前自己負(fù)責(zé)把參數(shù)從堆棧中彈出。
理解回調(diào)函數(shù)!
程序在調(diào)用一個(gè)函數(shù)(function)時(shí)(通常指api).相當(dāng)于程序(program)呼叫(Call)了一個(gè)函數(shù)(function)關(guān)系表示如下:
call(調(diào)用)
program --------------------→ dll
程序在調(diào)用一個(gè)函數(shù)時(shí),將自己的函數(shù)的地址作為參數(shù)傳遞給程序調(diào)用的函數(shù)時(shí)(那么這個(gè)自己的函數(shù)稱回調(diào)函數(shù)).需要回調(diào)函數(shù)的 DLL 函數(shù)往往是一些必須重復(fù)執(zhí)行某些操作的函數(shù).關(guān)系表示如下:
call(調(diào)用)
program --------------------→ dll
↑ ¦
¦_______________________________¦
callback(回調(diào))
當(dāng)你調(diào)用的函數(shù)在傳遞返回值給回調(diào)函數(shù)時(shí),你就可以利用回調(diào)函數(shù)來處理或完成一定的操作。至于如何定義自己的回調(diào)函數(shù),跟具體使用的API函數(shù)有關(guān),很多不同類別的回調(diào)函數(shù)有各種各樣的參數(shù),有關(guān)這些參數(shù)的描述一般在幫助中有說明回調(diào)函數(shù)的參數(shù)和返回值等.其實(shí)簡單說回調(diào)函數(shù)就是你所寫的函數(shù)滿足一定條件后,被DLL調(diào)用!
也有這樣的說法(比較容易理解):
回調(diào)函數(shù)就好像是一個(gè)中斷處理函數(shù),系統(tǒng)在符合你設(shè)定的條件時(shí)自動(dòng)調(diào)用。為此,你需要做三件事:
1. 聲明;
2. 定義;
3. 設(shè)置觸發(fā)條件,就是在你的函數(shù)中把你的回調(diào)函數(shù)名稱轉(zhuǎn)化為地址作為一個(gè)參數(shù),以便于DLL調(diào)用。
回調(diào)函數(shù),顧名思義,也就是等該函數(shù)執(zhí)行完了,會回去調(diào)用我們傳進(jìn)去的函數(shù)。
用到回調(diào)函數(shù)的地方有不少,像我見過的:SQLite中的一個(gè)函數(shù),sqlite_exec函數(shù)名有沒有記錯(cuò)我沒什么印象了。待這個(gè)函數(shù)執(zhí)行完畢后,會去調(diào)用我傳進(jìn)去的一個(gè)函數(shù),一般回調(diào)函數(shù)都是有自己的參數(shù)列表格式的,再利用這個(gè)格式從回調(diào)函數(shù)中獲取到我們需要的一些值。
看一下這個(gè)例子吧,我是這樣理解的:
#include
"iostream.h"
#include
"windows.h"
typedef
void
(CALLBACK
*MyFun)(void);//回調(diào)函數(shù)定義
void
CALLBACK
callback()
//
回調(diào)函數(shù)
{
cout"****callback****\n";
}
void
Call_CallBack(MyFun
mycb)
{
cout"****Call_CallBack****\n";
mycb();
cout"__________________\n";
}
void
main()
{
Call_CallBack(callback);
}
//
其他人需要修改的話只要修改callback函數(shù)里的內(nèi)容就行了,一般sdk封裝后都會有回調(diào),這樣他人在調(diào)用sdk的時(shí)候就可以實(shí)現(xiàn)回調(diào)函數(shù)里的內(nèi)容。
如果你需要理論的,網(wǎng)上搜回調(diào)函數(shù),內(nèi)容哈多隨便看
你的DLL注入每意義,應(yīng)為你的DLL中的代碼不會被有效執(zhí)行。
DLL注入主要還是要HOOK系統(tǒng)API,但是NP是不允許你HOOK的。
你所謂的回調(diào)函數(shù)沒有意義。
兩進(jìn)程通訊,windows有很多方法,
SendMessage就很簡單。你的DLL在共享內(nèi)存段加入句柄,主程序調(diào)用,并給句柄負(fù)值,DLL就可以直接SendMessageg給主窗口了。
嫌通訊數(shù)據(jù)不夠大,可以使用CreateFileMapping,創(chuàng)建內(nèi)存映像文件也可以。
最后,你試過你的DLL可以直接注入到游戲進(jìn)程?NP無反應(yīng)?
回調(diào)函數(shù) 就是上層調(diào)用 設(shè)置下去
底層通過函數(shù)指針調(diào)用上層函數(shù)
多文件中才有用 單文件可以模擬
比如
#include?stdio.h
typedef?void?(*pFuncCb)?(int);//定義回調(diào)函數(shù)。
void?callback1(int?a)
{
printf("callback?function1?is?called?and?parameter?=?%d\n",?a);//打印1
}
void?callback2(int?a)
{
printf("callback?function2?is?called?and?parameter?=?%d\n",?a);//打印2
}
pFuncCb?callback_function;
void?lowerFunc(int?n)
{
int?i;
for(i?=?n;?i??n+10;?i?++)
if(callback_function)?callback_function(i);
}
int?main()
{
callback_function?=?callback1;
lowerFunc(1);//?會打印十次?打印1,?1到10
callback_function?=NULL;
lowerFunc(10);//沒有打印。
callback_function?=?callback2;
lowerFunc(100);//?會打印十次?打印2,?100到110
return?0;
}