?
?因為是隨便一想就想出的方法,沒來得及用搜索引擎了解相關的情況,老神仙們見笑了。
??? 通常,程序把一些設置,配置保存在一個專門的文件中或者注冊表中,但在一些情況下,這樣非常不方便。比如偷OICQ號的那種程序(沒用過,也叫不上名來),提前設置好郵箱,然后放在別人機子上運行,通過某種方法取得OICQ密碼,最后把密碼發往指定信箱。把設置的郵箱保存在一個配置文件中一起放到別人機子上,就不如保存在執行文件本身中用起來方便。這只是舉個例子,在一些情況下,把配置,或者一些信息保存在可執行文件本身中是很有用處的。
??? 下面我通過一個例子,來講一種實現的方法,這種方法簡單易用。
??? 兩個程序,一個叫 SettingInMe ,一個叫 SetYou 。SettingInMe 讀取程序中的一個字符串,并把這個字符串的內容,用一個 MessageBox 顯示出來,SettingInMe 相當于讀取設置的程序,那個字符串就相當于我們的設置。SetYou 設置那個SettingInMe 中的字符串,相當于設置另一個程序中的設置或者參數。運行 SettingInMe 可以看到程序中原來的設置。退出 SettingInMe ,運行放在同一文件夾下的 SetYou 隨便輸入一些字符,然后退出 SetYou ,再運行 SettingInMe 看看,就會發現里面的設置被改了。
??? 大家可以先下載這兩個可執行文件試一下。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用來存放參數的空間長度,不要超過。
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
char* lpText = MagicStr+cbMatchStr;
MessageBox(NULL,lpText,"Yes!",NULL);
return 0;
}
??? 申請了一個全局變量 MagicStr ,我們要把設置放在它的里面,SettingInMe 會使用這個全局變量中的內容做為設置。程序編譯之后,這個全局變量會在 EXE 文件中占相同的空間,只要設置程序 SetYou 能找到這個全局變量的在文件中的位置,然后寫入相應的內容,就可以達到我們的目的。由于全局變量在編譯好的 EXE 文件中也要占相同的空間,覆蓋其中的內容,只是改變了全局變量的值,并不會破壞可執行程序。在這里我們用這個全局變量的前55個字節的內容,做為搜索串,來標明這個全局變量在文件中的位置,這個搜索串之后的256個字節,用來存放設置。關于這個,后面將有更多的討論。
??? 下面看一下 SetYou 如何找到這個全局變量和設置它的內容,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,字符串的結束符1,共56字節
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字節,然后在內存中進行比較...
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() 會截取范圍內部分。
EndDialog(hwndDlg,TRUE);
break;
case IDCANCEL:
PostQuitMessage(0);
break;
}
break;
}
return 0;
}
??? SetYou 首先用 CreateFile 打開相同文件夾下的 SettingInMe.exe 然后開始一個字節一個字節的讀文件中的內容,每讀一個字節,就把這個字節和用來標明全局變量位置的那個全局變量開始處的字符串進行比較,如此一直讀到文件結束。讀到一個字節,比較是否和標志串中第i個(最開始是0)相同,相同的話,就把i++;接著讀和比較(再比較就是標志串中的第二個字節了),又相同,又i++,又...直到一部分相同之后,有一個不同,把i重新設為0,然后接著讀文件,向后尋找。或者所讀內容和整個標志串都相同,我們找到了要找的那個全局變量。找到全局變量之后,我們可以用 WriteFile 來改變這個全局變量中的內容。在 SetYou 這個程序中,會彈出一個對話框,來取得輸入串。在 SetYou 中 WriteFile(hFile,SetStr,strlen(SetStr)),最后那個參數是個strlen,也就是說只會覆蓋設置串中前面的部分。如果這里的參數是256(也就是sizeof(SetStr))的話,就會覆蓋整個設置。
??? 這樣,我們在一個程序中靜態申請一個全局變量,來在EXE文件中保留一定的空間,來作為保存設置,配置,參數,信息的空間。并在這個空間的開始處放一個標志串,用來查找。另一個程序,查找那個標志串,找到這個全局變量的空間,對其相應位置進行設置,就實現了將EXE程序的設置保存在EXE文件本身。
??? 需要說明的是,程序一旦編譯之后,EXE文件中的內容就固定了,所以文件中要不然有其他地方和標志串一樣,要不然沒有其他地方和標志串一樣。如果有地方和標志串內容一樣的話,只要換個標志串就可以了。只要標志串有一定長度,要和其他內容不一樣是很容易的。更進一步說,編譯之后,EXE文件內容固定,全局變量在EXE文件中的位置也是固定的,只要能找到這個位置,是沒有必要有一個標志串的,本文例子用標志串,只是為了便于說明和理解。
??? 最后總結一下,程序中靜態申請一塊全局空間,使用這個空間中的內容作為參數,設置,配置或者某些信息。在編譯生成的EXE文件中就相應的也存在相同大小的一個空間。在另一個程序中打開這個程序的EXE文件,用某種方法找到這塊空間在EXE文件中的位置,就可以進行設置。
??? 唉,這篇文章寫的真爛啊,大家湊合看吧。本來還說拿它得個諾貝爾文學獎玩玩,看來也沒戲了。
終
下載SettingInMe和SetYou的源程序?
下載SettingInMe的可執行文件
下載SetYou的可執行文件
Trackback: http://tb.donews.net/TrackBack.aspx?PostId=1044749