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

woaidongmao

文章均收錄自他人博客,但不喜標題前加-[轉貼],因其丑陋,見諒!~
隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
數據加載中……

“傳遞/轉發”可變參數并通過printf記錄程序日志,徹底告別vsnprintf!:)

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

但是通常我們需要在記錄日志的時候記錄更多的信息,比如說運行時間等,所以我們不能使用一條簡單的printf來完成該操作,另外,為防止日志信息以外丟失,我們最好是在每次printf后立即調用fflush。所以我們通常會使用下面的方法來完成日志記錄操作:

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);
}

從該代碼可以看出,我們必須事先定義好一個我們認為足夠大的緩存已存儲所有可能的數據,這就是使用該方法帶來的inflexibility,究竟多大才算足夠大啊?10240102400?甚至1024000?恐怕你的棧也沒這么大吧!即使你在堆中分配存儲空間也一樣!
接下來我就介紹一種不用預先分配緩存并且能夠接受并輸出任意長度的信息至日至文件中(當然,只要不超過你的系統允許的大小),試想,只要我們在log0中完成了我們想做的任何事(比如輸出日志前綴信息等),并且如果能夠將調用者傳遞給log0的參數原封不動的傳遞給printf的話,即將所有的可變參數按照printf所要求的格式傳遞給它,由它來完成剩下的操作。這是不是就克服了使用預先分配緩存的問題呢?沒錯,接下來所要解決的就是怎樣將這些可變參數傳遞printf

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

也就是說擁有相同聲明的函數,在被調用時,他們所擁有的參數棧(即Stack Frame)的結構是一樣的。所以,只要我們能夠從一個函數A“突然跳轉到另外一個函數B中,那么B所擁有的參數棧和A將是同一份數據,即他們共享了同一份參數棧數據。需要注意的是,這里的跳轉不能使用通常的函數掉用來實現,因為函數被調用時,編譯器會在背后做很多事情,如給我們設置新的ESP指針等等,因此這樣勢必不能達到共享參數棧數據的目的。為了不讓編譯器在函數調用時在背后做任何事情,我們需要使用一個naked函數,在這樣的函數中我們就可以自己利用棧資源,自己控制所有一切。有了這樣的函數后,就可以很輕松,而且很高效的達到我們的目的了。

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
/* 將函數返回地址保存到EBX */
        call    dword ptr [printf]
        sub     esp, 4    /* 1 */


       
/* 調用fflush將數據立即保存到文件中 */
        call    dword ptr [__iob_func]
        add     eax, 0x20

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


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

}

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

當然,這只是這種所謂的棧共享技術的一個應用而已了,掌握了這種技術后,我想你肯定會把它應用到其他更適合的地方。

其實,在VC8中,由于提供了可變參數的宏,所以我們可以通過下面一條簡單的調用來完成日至記錄操作,而且信息還比較完全:

#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) 評論(0)  編輯 收藏 引用 所屬分類: C++ 基礎

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美成人精品不卡视频在线观看| 久久久久国色av免费看影院| 亚洲伦理在线| 在线欧美日韩国产| 伊人久久亚洲美女图片| 伊人久久婷婷| 亚洲茄子视频| 一本色道久久综合狠狠躁的推荐| 日韩午夜免费视频| 国产日韩欧美在线视频观看| 国产一区二区电影在线观看 | 亚洲国产精品久久91精品| 有坂深雪在线一区| 99热在这里有精品免费| 亚洲欧美在线磁力| 麻豆成人精品| 99精品视频免费全部在线| 亚洲一区二区三| 久久精品国产综合| 欧美午夜精品久久久久久孕妇| 国产欧美日韩精品专区| 亚洲人成免费| 久久电影一区| 亚洲精品国产拍免费91在线| 亚洲乱码日产精品bd| 欧美亚洲专区| 欧美性猛交xxxx乱大交退制版| 久久国产精品72免费观看| 欧美88av| 国产一区三区三区| 亚洲性夜色噜噜噜7777| 麻豆精品视频在线| 亚洲影院免费观看| 欧美经典一区二区三区| 伊甸园精品99久久久久久| 亚洲图片欧洲图片av| 欧美freesex8一10精品| 亚洲免费一区二区| 欧美色另类天堂2015| 在线观看视频亚洲| 欧美淫片网站| 国产精品99久久久久久久久| 欧美1区2区3区| 久久久久综合网| 国产亚洲美州欧州综合国| 亚洲视频精品在线| 欧美肥婆bbw| 久久久九九九九| 国产精品视频免费一区| 99在线热播精品免费| 欧美激情亚洲国产| 久久精品亚洲精品国产欧美kt∨| 国产精品视频网| 香蕉免费一区二区三区在线观看| 亚洲精品一区在线观看香蕉| 亚洲日本久久| 免费欧美在线| 最新高清无码专区| 欧美激情第三页| 免费成人黄色av| 欧美日韩精品一区| 亚洲靠逼com| 亚洲欧洲日韩综合二区| 欧美激情精品久久久久久大尺度| 美女亚洲精品| 91久久午夜| 最近中文字幕日韩精品| 欧美激情精品久久久久久| 亚洲人成网站在线观看播放| 欧美bbbxxxxx| 欧美激情一区二区| 亚洲午夜精品久久| 亚洲综合第一| 一区二区在线观看视频在线观看| 鲁大师影院一区二区三区| 久久频这里精品99香蕉| 亚洲精品中文字幕在线| 亚洲乱码精品一二三四区日韩在线 | 性欧美大战久久久久久久久| 亚洲综合国产激情另类一区| 国产资源精品在线观看| 欧美3dxxxxhd| 欧美国产日韩精品免费观看| 在线午夜精品自拍| 亚洲欧美在线一区二区| 在线观看亚洲视频| 亚洲电影观看| 国产精品一二| 欧美国产高清| 国产乱码精品一区二区三区不卡 | 久久大香伊蕉在人线观看热2| 久久久久久久999精品视频| 亚洲韩国青草视频| 野花国产精品入口| 激情久久影院| 亚洲美女免费视频| 国产在线精品自拍| 亚洲精品欧美激情| 国产精品专区h在线观看| 欧美成人精品福利| 国产精品系列在线播放| 欧美韩日视频| 国产精品永久入口久久久| 欧美国产免费| 国产日韩欧美一区| 夜夜夜久久久| 亚洲日本欧美日韩高观看| 亚洲欧美视频在线观看视频| 亚洲精品专区| 久久夜色精品一区| 久久精品亚洲一区二区三区浴池| 欧美日韩精品免费在线观看视频| 久久综合色8888| 国产精品永久免费观看| 在线日本成人| 亚洲在线1234| av成人激情| 免费美女久久99| 老色鬼精品视频在线观看播放| 国产精品欧美一区二区三区奶水 | 欧美中文日韩| 午夜激情综合网| 欧美日韩一区二区国产| 亚洲国产精品女人久久久| 激情婷婷欧美| 欧美在线免费视屏| 欧美一区二区三区免费大片| 欧美午夜片在线免费观看| 亚洲精品一区在线| 一区二区三区不卡视频在线观看| 欧美成人午夜激情在线| 欧美国产日韩视频| 亚洲国产福利在线| 久久久一区二区| 欧美成人性生活| 亚洲国产精品久久久久婷婷老年| 久久久噜久噜久久综合| 欧美成人中文| 99av国产精品欲麻豆| 欧美人成在线| 亚洲视频一二三| 久久国产精品一区二区三区| 国产一区二区三区四区| 久久综合色8888| 亚洲国产成人高清精品| 亚洲作爱视频| 国产精品日韩久久久久| 欧美一激情一区二区三区| 欧美一区永久视频免费观看| 国产亚洲成精品久久| 久久全球大尺度高清视频| 欧美激情自拍| 亚洲图片在线| 国产一区香蕉久久| 久久嫩草精品久久久久| 亚洲日本国产| 久久国产福利| 亚洲国产天堂网精品网站| 欧美日韩第一区| 香蕉成人久久| 欧美激情一区二区| 亚洲天堂av在线免费| 国产伦精品一区二区三区免费迷| 欧美在线视频免费播放| 欧美国产一区二区在线观看| 中文一区二区在线观看| 国产麻豆午夜三级精品| 美女主播一区| 亚洲一区视频| 欧美国产日韩二区| 欧美一区二区性| 日韩午夜免费视频| 国产一区二区日韩精品| 欧美美女视频| 久久精品麻豆| 亚洲视频你懂的| 亚洲丰满少妇videoshd| 羞羞漫画18久久大片| 亚洲精品一区二区三区樱花| 国产噜噜噜噜噜久久久久久久久| 欧美成人国产| 久久精品国产v日韩v亚洲| 99这里有精品| 亚洲国产经典视频| 久久成人一区| 六月天综合网| 亚洲第一在线视频| 欧美日韩国产在线看| 欧美伊人久久| 中国亚洲黄色| 亚洲高清电影| 久久久青草婷婷精品综合日韩 | 久久久久久综合网天天| 一区二区三区欧美日韩| 亚洲高清视频一区| 久久中文在线| 久久av资源网站| 亚洲欧美中文日韩在线| 亚洲色图在线视频| 99在线精品观看|