?
?因?yàn)槭请S便一想就想出的方法,沒來得及用搜索引擎了解相關(guān)的情況,老神仙們見笑了。
??? 通常,程序把一些設(shè)置,配置保存在一個專門的文件中或者注冊表中,但在一些情況下,這樣非常不方便。比如偷OICQ號的那種程序(沒用過,也叫不上名來),提前設(shè)置好郵箱,然后放在別人機(jī)子上運(yùn)行,通過某種方法取得OICQ密碼,最后把密碼發(fā)往指定信箱。把設(shè)置的郵箱保存在一個配置文件中一起放到別人機(jī)子上,就不如保存在執(zhí)行文件本身中用起來方便。這只是舉個例子,在一些情況下,把配置,或者一些信息保存在可執(zhí)行文件本身中是很有用處的。
??? 下面我通過一個例子,來講一種實(shí)現(xiàn)的方法,這種方法簡單易用。
??? 兩個程序,一個叫 SettingInMe ,一個叫 SetYou 。SettingInMe 讀取程序中的一個字符串,并把這個字符串的內(nèi)容,用一個 MessageBox 顯示出來,SettingInMe 相當(dāng)于讀取設(shè)置的程序,那個字符串就相當(dāng)于我們的設(shè)置。SetYou 設(shè)置那個SettingInMe 中的字符串,相當(dāng)于設(shè)置另一個程序中的設(shè)置或者參數(shù)。運(yùn)行 SettingInMe 可以看到程序中原來的設(shè)置。退出 SettingInMe ,運(yùn)行放在同一文件夾下的 SetYou 隨便輸入一些字符,然后退出 SetYou ,再運(yùn)行 SettingInMe 看看,就會發(fā)現(xiàn)里面的設(shè)置被改了。
??? 大家可以先下載這兩個可執(zhí)行文件試一下。SettingInMe.exeSetYou.exe
??? 首先看一下 SettingInMe 的源程序。
SettingInMe
___________________________________________________________________
#include <windows.h>
#include <stdio.h>
#define cbMatchStr 55 //strlen("I'm JIURL,My HomePage is http://jiurl.yeah.net. Salute!")
char MagicStr[cbMatchStr+256]="I'm JIURL,My HomePage is http://jiurl.yeah.net. Salute!http://jiurl.yeah.net\n\nhttp://jiurl.nease.net\n\njiurl@mail.china.com";
//256用來存放參數(shù)的空間長度,不要超過。
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
char* lpText = MagicStr+cbMatchStr;
MessageBox(NULL,lpText,"Yes!",NULL);
return 0;
}
??? 申請了一個全局變量 MagicStr ,我們要把設(shè)置放在它的里面,SettingInMe 會使用這個全局變量中的內(nèi)容做為設(shè)置。程序編譯之后,這個全局變量會在 EXE 文件中占相同的空間,只要設(shè)置程序 SetYou 能找到這個全局變量的在文件中的位置,然后寫入相應(yīng)的內(nèi)容,就可以達(dá)到我們的目的。由于全局變量在編譯好的 EXE 文件中也要占相同的空間,覆蓋其中的內(nèi)容,只是改變了全局變量的值,并不會破壞可執(zhí)行程序。在這里我們用這個全局變量的前55個字節(jié)的內(nèi)容,做為搜索串,來標(biāo)明這個全局變量在文件中的位置,這個搜索串之后的256個字節(jié),用來存放設(shè)置。關(guān)于這個,后面將有更多的討論。
??? 下面看一下 SetYou 如何找到這個全局變量和設(shè)置它的內(nèi)容,SetYou 的源程序。
SetYou
___________________________________________________________________
#include <windows.h>
#include <stdio.h>
#include "resource.h"
INT_PTR CALLBACK DialogProc(HWND,UINT,WPARAM,LPARAM);
char SetStr[256];
#define cbMatchStr 55//strlen("I'm JIURL,My HomePage is http://jiurl.yeah.net. Salute!")
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
HANDLE hFile;
char Err[256];
char MatchStr[cbMatchStr+1]="I'm JIURL,My HomePage is http://jiurl.yeah.net. Salute!";
//字符串長度55,字符串的結(jié)束符1,共56字節(jié)
hFile = CreateFile("SettingInMe.exe",GENERIC_WRITE|GENERIC_READ,0,
NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
//SettingInMe.exe 和 本程序要在同一目錄下
if(hFile==INVALID_HANDLE_VALUE)
{
sprintf(Err,"CreatFile Error: 0x%08x !",GetLastError());
MessageBox(NULL,Err,"CoNgRaTuLaTiOn!",NULL);
return 0;
}
char Data;
DWORD nRead;
int retval;
int i=0;
//這種效率不好,不過比較直觀,
//如果一次讀入1024字節(jié),然后在內(nèi)存中進(jìn)行比較...
while(1)
{
retval = ReadFile(hFile,&Data,1,&nRead,NULL);
if(retval==0||nRead==0)
{
MessageBox(NULL,"Not Found Place to Set","CoNgRaTuLaTiOn!",NULL);
break;
}
if(Data==MatchStr[i])i++;
else i=0;
if(i==cbMatchStr)
{
DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG), NULL, (DLGPROC)DialogProc);
retval = WriteFile(hFile,SetStr,strlen(SetStr),
&nRead,NULL);
if(retval!=0)
MessageBox(NULL,"Set OK!","Yes!",NULL);
else
{
sprintf(Err,"WriteFile Error: 0x%08x !",GetLastError());
MessageBox(NULL,Err,"CoNgRaTuLaTiOn!",NULL);
}
break;
}
}
MessageBox(NULL,"OVER","OVER",NULL);
CloseHandle(hFile);
return 0;
}
INT_PTR CALLBACK DialogProc(HWND hwndDlg,UINT uMsg,
WPARAM wParam,LPARAM lParam)
{?
switch(uMsg)
{
case WM_COMMAND:
switch(wParam)
{
case IDOK:
GetDlgItemText(hwndDlg,IDC_IN,SetStr,sizeof(SetStr));
//SetStr[]的長度不要超過256。
//在對話框中輸入的串要是超過256,
//GetDlgItemText() 會截取范圍內(nèi)部分。
EndDialog(hwndDlg,TRUE);
break;
case IDCANCEL:
PostQuitMessage(0);
break;
}
break;
}
return 0;
}
??? SetYou 首先用 CreateFile 打開相同文件夾下的 SettingInMe.exe 然后開始一個字節(jié)一個字節(jié)的讀文件中的內(nèi)容,每讀一個字節(jié),就把這個字節(jié)和用來標(biāo)明全局變量位置的那個全局變量開始處的字符串進(jìn)行比較,如此一直讀到文件結(jié)束。讀到一個字節(jié),比較是否和標(biāo)志串中第i個(最開始是0)相同,相同的話,就把i++;接著讀和比較(再比較就是標(biāo)志串中的第二個字節(jié)了),又相同,又i++,又...直到一部分相同之后,有一個不同,把i重新設(shè)為0,然后接著讀文件,向后尋找?;蛘咚x內(nèi)容和整個標(biāo)志串都相同,我們找到了要找的那個全局變量。找到全局變量之后,我們可以用 WriteFile 來改變這個全局變量中的內(nèi)容。在 SetYou 這個程序中,會彈出一個對話框,來取得輸入串。在 SetYou 中 WriteFile(hFile,SetStr,strlen(SetStr)),最后那個參數(shù)是個strlen,也就是說只會覆蓋設(shè)置串中前面的部分。如果這里的參數(shù)是256(也就是sizeof(SetStr))的話,就會覆蓋整個設(shè)置。
??? 這樣,我們在一個程序中靜態(tài)申請一個全局變量,來在EXE文件中保留一定的空間,來作為保存設(shè)置,配置,參數(shù),信息的空間。并在這個空間的開始處放一個標(biāo)志串,用來查找。另一個程序,查找那個標(biāo)志串,找到這個全局變量的空間,對其相應(yīng)位置進(jìn)行設(shè)置,就實(shí)現(xiàn)了將EXE程序的設(shè)置保存在EXE文件本身。
??? 需要說明的是,程序一旦編譯之后,EXE文件中的內(nèi)容就固定了,所以文件中要不然有其他地方和標(biāo)志串一樣,要不然沒有其他地方和標(biāo)志串一樣。如果有地方和標(biāo)志串內(nèi)容一樣的話,只要換個標(biāo)志串就可以了。只要標(biāo)志串有一定長度,要和其他內(nèi)容不一樣是很容易的。更進(jìn)一步說,編譯之后,EXE文件內(nèi)容固定,全局變量在EXE文件中的位置也是固定的,只要能找到這個位置,是沒有必要有一個標(biāo)志串的,本文例子用標(biāo)志串,只是為了便于說明和理解。
??? 最后總結(jié)一下,程序中靜態(tài)申請一塊全局空間,使用這個空間中的內(nèi)容作為參數(shù),設(shè)置,配置或者某些信息。在編譯生成的EXE文件中就相應(yīng)的也存在相同大小的一個空間。在另一個程序中打開這個程序的EXE文件,用某種方法找到這塊空間在EXE文件中的位置,就可以進(jìn)行設(shè)置。
??? 唉,這篇文章寫的真爛啊,大家湊合看吧。本來還說拿它得個諾貝爾文學(xué)獎玩玩,看來也沒戲了。
終
下載SettingInMe和SetYou的源程序?
下載SettingInMe的可執(zhí)行文件
下載SetYou的可執(zhí)行文件
Trackback: http://tb.donews.net/TrackBack.aspx?PostId=1044749