青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

woaidongmao

文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見(jiàn)諒!~
隨筆 - 1469, 文章 - 0, 評(píng)論 - 661, 引用 - 0
數(shù)據(jù)加載中……

“傳遞/轉(zhuǎn)發(fā)”可變參數(shù)并通過(guò)printf記錄程序日志,徹底告別vsnprintf!:)

通常我們需要在程序中輸出部分日志信息,并把它記錄到文件中。在這種情況下,使用printf可以為我們帶了很大方便。因?yàn)?span lang="EN-US">printf卻省情況下是向stdout即控制臺(tái)屏幕輸出信息,在GUI程序中,我們看不到printf的輸出結(jié)果,但是我們可以將該輸出重定向到指定的文件中。即使用freopen(“c:\\yourlog.log”, “a+”,stdout)或通過(guò)yourapp.exe > c:\yourlog.log完成輸出重定向操作。

但是通常我們需要在記錄日志的時(shí)候記錄更多的信息,比如說(shuō)運(yùn)行時(shí)間等,所以我們不能使用一條簡(jiǎn)單的printf來(lái)完成該操作,另外,為防止日志信息以外丟失,我們最好是在每次printf后立即調(diào)用fflush。所以我們通常會(huì)使用下面的方法來(lái)完成日志記錄操作:

void __cdecl log0 (const char* _Format, ...)
{
   
char buff[10240], tm[80];
    va_list vl;
    va_start (vl, _Format);

    _strtime (tm);
    vsnprintf (buff, 10240, _Format, vl);
    printf ("%s - %s", tm, buff);
    fflush (stdout);

    va_end (vl);
}

從該代碼可以看出,我們必須事先定義好一個(gè)我們認(rèn)為足夠大的緩存已存儲(chǔ)所有可能的數(shù)據(jù),這就是使用該方法帶來(lái)的inflexibility,究竟多大才算足夠大啊?10240102400?甚至1024000?恐怕你的棧也沒(méi)這么大吧!即使你在堆中分配存儲(chǔ)空間也一樣!
接下來(lái)我就介紹一種不用預(yù)先分配緩存并且能夠接受并輸出任意長(zhǎng)度的信息至日至文件中(當(dāng)然,只要不超過(guò)你的系統(tǒng)允許的大小),試想,只要我們?cè)?span lang="EN-US">log0
中完成了我們想做的任何事(比如輸出日志前綴信息等),并且如果能夠?qū)⒄{(diào)用者傳遞給log0的參數(shù)原封不動(dòng)的傳遞給printf的話,即將所有的可變參數(shù)按照printf所要求的格式傳遞給它,由它來(lái)完成剩下的操作。這是不是就克服了使用預(yù)先分配緩存的問(wèn)題呢?沒(méi)錯(cuò),接下來(lái)所要解決的就是怎樣將這些可變參數(shù)傳遞printf

由于在log0內(nèi)部,我們不知道調(diào)用者究竟傳遞了多少個(gè)參數(shù)進(jìn)來(lái),所以我們不能按照通常所用的按照參數(shù)名的方式將參數(shù)傳遞給printf。但是,先別急,看看log0的函數(shù)聲明,他是不是和printf的聲明完全一致呢(事實(shí)上只要是log0中的參數(shù)和printf中的部分一致也可,如void log1(char* filename, int len, char* _Format, ...)也可。)?

也就是說(shuō)擁有相同聲明的函數(shù),在被調(diào)用時(shí),他們所擁有的參數(shù)棧(即Stack Frame)的結(jié)構(gòu)是一樣的。所以,只要我們能夠從一個(gè)函數(shù)A“突然跳轉(zhuǎn)到另外一個(gè)函數(shù)B中,那么B所擁有的參數(shù)棧和A將是同一份數(shù)據(jù),即他們共享了同一份參數(shù)棧數(shù)據(jù)。需要注意的是,這里的跳轉(zhuǎn)不能使用通常的函數(shù)掉用來(lái)實(shí)現(xiàn),因?yàn)楹瘮?shù)被調(diào)用時(shí),編譯器會(huì)在背后做很多事情,如給我們?cè)O(shè)置新的ESP指針等等,因此這樣勢(shì)必不能達(dá)到共享參數(shù)棧數(shù)據(jù)的目的。為了不讓編譯器在函數(shù)調(diào)用時(shí)在背后做任何事情,我們需要使用一個(gè)naked函數(shù),在這樣的函數(shù)中我們就可以自己利用棧資源,自己控制所有一切。有了這樣的函數(shù)后,就可以很輕松,而且很高效的達(dá)到我們的目的了。

void mkprefix ()
{  
char buff[80];
    _strtime (buff);
    printf ("%s - ", buff);
}


__declspec (naked)
void __cdecl
xprintf (
const char* _Format, ...)
{
    __asm
    {   call    dword ptr [mkprefix]

        pop     ebx
/* 將函數(shù)返回地址保存到EBX */
        call    dword ptr [printf]
        sub     esp, 4    /* 1 */


       
/* 調(diào)用fflush將數(shù)據(jù)立即保存到文件中 */
        call    dword ptr [__iob_func]
        add     eax, 0x20

        push    eax
        call    dword ptr [fflush]
        add     esp, 4  /* 2 */


        mov     dword ptr [esp], ebx
/* 恢復(fù)函數(shù)返回地址 */
        ret
    }

}

代碼中在12處分別對(duì)ESP4后又加4,所以這兩處的代碼完全可以忽略,在這里加上是為了更好的理解函數(shù)調(diào)用的機(jī)制(即在函數(shù)調(diào)用后需要修正ESP,即所謂的Stack clean-up)。你可以將mkprefix 作的足夠復(fù)雜已記錄更多的信息,甚至我們可以通過(guò)log0將參數(shù)傳遞到mkprefix 中,向log1那樣。不過(guò)這樣處理起來(lái)就稍復(fù)雜點(diǎn),為簡(jiǎn)單起見(jiàn),就不再講述這種方法了。

當(dāng)然,這只是這種所謂的棧共享技術(shù)的一個(gè)應(yīng)用而已了,掌握了這種技術(shù)后,我想你肯定會(huì)把它應(yīng)用到其他更適合的地方。

其實(shí),在VC8中,由于提供了可變參數(shù)的宏,所以我們可以通過(guò)下面一條簡(jiǎn)單的調(diào)用來(lái)完成日至記錄操作,而且信息還比較完全:

#define TRACE(fmt, ...) printf ("%s (%s:%d) - "##fmt, mkprefix(), __FILE__, __LINE__, __VA_ARGS__)

TRACE ("This is a debug information, a = %d, b = %s. ", 234, "xxxxx");

 

posted on 2009-08-12 13:01 肥仔 閱讀(835) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): C++ 基礎(chǔ)

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            麻豆freexxxx性91精品| 国产视频在线观看一区二区| 在线观看成人小视频| 亚洲视频国产视频| 亚洲免费高清| 亚洲天堂偷拍| 久久不见久久见免费视频1| 午夜视频一区在线观看| 久久精品官网| 欧美激情视频在线播放| 国产精品三上| 91久久亚洲| 午夜日韩在线观看| 蜜桃av一区二区三区| 免费h精品视频在线播放| 亚洲女同同性videoxma| 欧美激情精品久久久久久久变态| 麻豆精品在线观看| 欧美日韩一区二区三区视频 | 亚洲欧美偷拍卡通变态| 先锋影院在线亚洲| 欧美激情bt| 香蕉久久夜色| 欧美日韩国产精品一区| 国内伊人久久久久久网站视频| 久久久亚洲精品一区二区三区| 欧美成人精品一区二区| 中文久久精品| 欧美a级一区二区| 国产日韩欧美日韩| 夜夜嗨av一区二区三区| 久久综合久久久久88| 日韩一级大片在线| 久久久亚洲影院你懂的| 国产精品久久久久77777| 亚洲国产精品一区二区第一页| 午夜欧美大片免费观看| 亚洲激情综合| 亚洲自拍高清| 欧美日韩极品在线观看一区| 亚洲电影免费观看高清| 久久精品亚洲热| 一本大道久久精品懂色aⅴ| 男人天堂欧美日韩| 在线观看成人av| 久久精品国产免费| 亚洲一区三区在线观看| 欧美天堂在线观看| 中日韩高清电影网| 欧美激情小视频| 久久午夜激情| 影音先锋中文字幕一区| 久久婷婷成人综合色| 亚洲欧美日韩一区二区在线| 国产精品国产三级国产专播精品人| 日韩亚洲成人av在线| 亚洲国产欧美在线人成| 欧美jizz19性欧美| 亚洲伦理中文字幕| 最近看过的日韩成人| 欧美精品福利| 亚洲天堂免费观看| 亚洲视频视频在线| 国产欧美精品一区aⅴ影院| 欧美一区免费视频| 欧美亚洲视频| 在线免费不卡视频| 亚洲福利免费| 欧美日韩国内| 久久国产天堂福利天堂| 久久久久久亚洲综合影院红桃 | 在线观看中文字幕亚洲| 欧美亚洲一区二区在线| 先锋亚洲精品| 国内精品视频在线播放| 欧美成人按摩| 欧美日韩国产系列| 欧美一区精品| 免费在线看成人av| 亚洲视频高清| 欧美一级成年大片在线观看| 黄色成人av网| 欧美成年人视频网站| 亚洲区中文字幕| 日韩视频在线一区二区| 国产欧美一区二区三区在线看蜜臀 | 国产日韩欧美制服另类| 裸体一区二区三区| 欧美激情按摩在线| 久久av一区二区三区漫画| 久久精品国产综合精品| 999亚洲国产精| 亚洲欧美日韩成人| 亚洲国产天堂久久国产91| 99这里有精品| 在线国产日韩| 亚洲欧美激情在线视频| 亚洲人成网站精品片在线观看 | 激情91久久| 99国内精品久久久久久久软件| 国产日韩综合一区二区性色av| 欧美激情亚洲激情| 国产亚洲欧美中文| 亚洲精品免费在线| 伊人成综合网伊人222| 夜夜嗨av色综合久久久综合网| 极品中文字幕一区| 午夜精品免费在线| 一区二区精品在线| 美女被久久久| 久久精品最新地址| 国产精品久久久久秋霞鲁丝| 亚洲国产精品成人综合| 国内精品久久久久久久果冻传媒| 夜夜爽av福利精品导航| 亚洲精品久久久久久久久久久| 久久成人这里只有精品| 欧美一区二区三区视频在线| 欧美日韩亚洲免费| 亚洲人体一区| 亚洲经典自拍| 免费观看欧美在线视频的网站| 久久精品视频在线播放| 国产精品综合久久久| 亚洲最新在线| 一区二区三区日韩欧美精品| 欧美韩国日本综合| 亚洲电影视频在线| 亚洲日本激情| 欧美精品播放| 99视频+国产日韩欧美| 99在线热播精品免费| 欧美日韩精品三区| 99re国产精品| 亚洲夜间福利| 国产精品视频九色porn| 亚洲欧美大片| 久久国产日韩| 在线成人h网| 欧美va亚洲va国产综合| 最新中文字幕亚洲| 久久精品亚洲精品国产欧美kt∨| 国产精品综合久久久| 亚洲在线视频免费观看| 先锋a资源在线看亚洲| 国产精品亚洲综合一区在线观看| 亚洲欧美成人网| 久久视频在线免费观看| 亚洲成人在线视频播放| 免费成人高清视频| 亚洲看片网站| 久久国产精品电影| 亚洲承认在线| 欧美日韩成人| 午夜精品电影| 欧美国产先锋| 亚洲伊人久久综合| 国产亚洲午夜高清国产拍精品| 久久综合精品国产一区二区三区| 亚洲欧洲另类| 午夜久久tv| 亚洲高清影视| 国产精品第一区| 久久久久久久高潮| aa成人免费视频| 久久嫩草精品久久久久| 一本色道久久| 狠狠色香婷婷久久亚洲精品| 欧美精品免费在线观看| 亚洲女爱视频在线| 亚洲高清123| 欧美一区观看| 日韩一区二区免费高清| 国户精品久久久久久久久久久不卡| 美日韩在线观看| 亚洲欧美激情视频| 亚洲精品日韩久久| 麻豆成人在线| 香蕉成人伊视频在线观看| 亚洲福利视频在线| 国产欧美日韩另类视频免费观看| 欧美国产三级| 久久免费99精品久久久久久| 亚洲视频精选在线| 91久久精品美女高潮| 久久精品日韩| 亚洲一区国产视频| 亚洲精品视频二区| 一区视频在线| 国产欧美日韩精品一区| 欧美日韩亚洲三区| 欧美成人一二三| 久久久免费精品视频| 午夜视频一区二区| 亚洲午夜免费福利视频| 99精品视频一区| 日韩一级精品| 日韩一区二区电影网| 欧美日韩一区二区免费视频| 免费日韩av电影|