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

行進中開火

C++夜未眠

MFC中一個危險的Bug

 

上次說日本海嘯警報的時候,程序出錯。在解析代碼的時候,發現了MFC中的一個Bug。

一。問題的產生。

這個程序,用來處理日本各種天氣預報數據,包括災害的預報。如果地震,臺風之類的自然災害到來,程序會把預報數據進行處理,生成相應的警報信息,并在電視上面顯示滾動的字幕來提示。程序本身,是幾年前公司的其他人寫的。里面有涉及到文件讀寫的地方,有很多地方,用了MFC中自帶的文件讀寫類CStdioFile。

CStdioFile這個文件讀寫類,估計大家都不陌生。這個類的父類,是CFile類。CStdioFile類本身的功能也很簡單。CStdioFile類有一個成員函數是ReadString,函數的定義如下:

    virtual LPTSTR ReadString(__out_ecount_z(nMax) LPTSTR lpsz, __in UINT nMax);
    virtual BOOL ReadString(CString& rString);
MSDN定義如下http://msdn.microsoft.com/library/x5t0zfyf(VS.80).aspx
BOOL ReadString(CString& rString);
throw( CFileException );
Return Value
A pointer to the buffer containing the text data. NULL if end-of-file was reached without reading any data; or if boolean, FALSE if end-of-file was reached without reading any data.

ReadString函數能直接讀取文本中的一行數據到CString中,很方便。讀到文件結尾,沒有讀出任何數據的時候,返回FALSE。很簡單的函數,但恰恰是這個函數有Bug。

程序在處理數據的時候,會生成一些臨時文件,然后會讀取這些臨時文件中的數據,讀取操作,正是用的CStdioFile的ReadString函數。讀取流程很簡單:

while(dFile.ReadString(Str_temp))
{
    doSomething();
}

當時的現象為,讀取到最后一行,總是直接返回FALSE,怎么也讀不出最后一行來。看了看文件的最后一行,包含2176個字符的數據,沒有換行符。沒有任何異常啊。當時沒想到是MFC的Bug,因為以前有這樣那樣的毛病,多數是預報數據本身有問題,所以這次也是先分析數據了。分析來分析去,沒發現這次的數據有什么異常。后來發現如果最后一行的文件不是2176個字符,就能正常讀出來。奇了怪了,2176也不是什么特殊長度啊。實驗了幾次后,覺的是在不對勁。莫非是MFC的Bug?

二。發現問題所在

決定看看MFC的代碼再說。做了個簡單的測試程序,跟到MFC代碼里一看,果然是MFC的問題!測試代碼如下:

    CStdioFile  dFile;
    dFile.Open("text.txt",CFile::modeRead);
    CString str;
    while (dFile.ReadString(str) != FALSE )
    {
        printf("%s", str);
    }
    dFile.Close();

測試代碼很簡單,讀text.txt文件中的每一行,然后打印出來。還是2176個字符就不行。確定了不是數據的問題,就是MFC代碼本身的Bug。

MFC的ReadString代碼如下:(中文是我加的注釋)

BOOL CStdioFile::ReadString(CString& rString)
{
    ASSERT_VALID(this);
    rString = &afxChNil;    // empty string without deallocating
    const int nMaxSize = 128;  //臨時字符串的長度
    LPTSTR lpsz = rString.GetBuffer(nMaxSize);  //保存每次讀取到的字符串到CString中
    LPTSTR lpszResult;  //指向每次讀到的字符串
    int nLen = 0;
    for (;;)
    {
        lpszResult = _fgetts(lpsz, nMaxSize+1, m_pStream); //讀取操作
        rString.ReleaseBuffer();
        // handle error/eof case
        if (lpszResult == NULL && !feof(m_pStream))
        {
            clearerr(m_pStream);
            AfxThrowFileException(CFileException::generic, _doserrno,
                m_strFileName);
        }
        // if string is read completely or EOF
        if (lpszResult == NULL ||
            (nLen = lstrlen(lpsz)) < nMaxSize ||
            lpsz[nLen-1] == '\n')
            break;
        nLen = rString.GetLength();
        lpsz = rString.GetBuffer(nMaxSize + nLen) + nLen; //位置后移
    }
    // remove '\n' from end of string if present
    lpsz = rString.GetBuffer(0);
    nLen = rString.GetLength();
    if (nLen != 0 && lpsz[nLen-1] == '\n') // 最后結果中,去掉回車符
        rString.GetBufferSetLength(nLen-1); 
    return lpszResult != NULL;  // 這里就是Bug的關鍵。返回值不對!
}

可以看到,ReadString的底層,是用fgets來讀取文件的。在內部,每次讀取128個字符到CString中,然后位置后移,反復讀取128個字符,直到遇到回車符或者文件結束。最后把回車符去掉,返回一個CString。其中,lpszResult也指向每次讀出的字符串。

這里就看出問題所在了,2176個字符,正好是128的17倍!也就是說,只要文件最后一行是128倍數個字符,就一定會返回FALSE。

為什么會這樣呢,因為ReadString在每次讀取128個字符的時候,用lpszResult指向讀取到的字符串。如果讀滿了128個字符,就繼續讀,如果讀到的字符不夠128個,那么就結束讀取。

當一行數據正好為128的倍數,又沒有回車符的時候,會發生什么呢?比如最后一行數據是128個,那么,讀一次128個字符,會繼續讀下一次,但是下一次的讀取,什么也沒有讀到,lpszResult就指向NULL,最后的返回值,是return lpszResult != NULL; 所以返回FALSE。

但之前讀到的128個字符,已經在CString里面了。也就是說實際上讀取已經成功了,但還是返回了FALSE。返回值不恰當!

Bug的描述:當文件的最后一行數據,正好是128的倍數個字符的時候,用ReadString讀取,一定會返回FALSE。但實際上讀取是成功的,返回的CString中的數據是正確的!(VC6.0中存在這個Bug,VS2005中,沒有這個Bug)

這個Bug,只會影響到最后一行數據。因為如果有換行符的存在,lpszResult就不會為NULL。

三。解決方法

要解決這個問題,也簡單,修改一下判斷ReadString成功與否的語句:

while (dFile.ReadString(str) != FALSE || str.GetLength() != 0)

在返回FALSE的情況下,CString的長度不為0,就不算讀取失敗。或者這樣:

if(!dFile.ReadString(str) && str.GetLength() == 0)

在返回FALSE并且CString的長度為0,則算讀取失敗,否則就是讀取成功。

這個程序,是用VC6.0做的,我有看了看VC2005中的代碼,發現這個Bug被修復了,代碼如下:

BOOL CStdioFile::ReadString(CString& rString)
{
    ASSERT_VALID(this);
    rString = _T("");    // empty string without deallocating
    const int nMaxSize = 128;
    LPTSTR lpsz = rString.GetBuffer(nMaxSize);
    LPTSTR lpszResult;
    int nLen = 0;
    for (;;)
    {
        lpszResult = _fgetts(lpsz, nMaxSize+1, m_pStream);
        rString.ReleaseBuffer();
        // handle error/eof case
        if (lpszResult == NULL && !feof(m_pStream))
        {
            Afx_clearerr_s(m_pStream);
            AfxThrowFileException(CFileException::genericException, _doserrno,
                m_strFileName);
        }
        // if string is read completely or EOF
        if (lpszResult == NULL ||
            (nLen = (int)lstrlen(lpsz)) < nMaxSize ||
            lpsz[nLen-1] == '\n')
            break;
        nLen = rString.GetLength();
        lpsz = rString.GetBuffer(nMaxSize + nLen) + nLen;
    }
    // remove '\n' from end of string if present
    lpsz = rString.GetBuffer(0);
    nLen = rString.GetLength();
    if (nLen != 0 && lpsz[nLen-1] == '\n')
        rString.GetBufferSetLength(nLen-1);
    return nLen != 0; //返回值變了!
}

我們看到,VC2005中,讀取部分的代碼與VC6.0中的代碼完全一樣。不一樣的地方只是返回值的部分。VC2005的ReadString中,返回值為

return nLen != 0;

也就是說,只要讀出的CString的長度不為0就為讀取成功。與我修改后的方法完全一致。就這樣向客戶解釋,然后修改了。悲劇的是,幾年前所有程序中所有使用ReadString函數的地方,都要進行修改。。。

MFC的這個Bug比較隱蔽,平常不容易發現,但一旦遇到特殊長度的數據,就會表現異常。所以,在用VC6.0開發的時候,盡量避免使用ReadString,或者在使用中,多判斷一步讀取出來的CString長度。避開這個Bug。

posted on 2010-03-26 22:15 Jakcie 閱讀(2327) 評論(4)  編輯 收藏 引用 所屬分類: C++ & CWindows & MFC

評論

# re: MFC中一個危險的Bug 2010-03-27 22:16 陳梓瀚(vczh)

解決方法應該是避免使用VC6進行開發。  回復  更多評論   

# re: MFC中一個危險的Bug 2010-03-27 23:47 Jakcie

的確應該避免用VC6開發。但使用VC6的人,還是不少啊。  回復  更多評論   

# re: MFC中一個危險的Bug 2010-03-28 11:17 唐風

1. 遺留代碼不可替代
2. 升級開發環境的預算和必要性
3. 有決策權開發人員的“技術慣性”
使得VC6到現在還這么有“生命”力。感覺就像IE6一樣,哈哈,有種種的不好,可偏還是有那么多用戶。

反正我是等VS2010出來就準備XX的,嘿嘿。

  回復  更多評論   

# re: MFC中一個危險的Bug 2010-03-28 16:08 Jakcie

是啊。尤其是像這種遺留代碼,的確是沒有辦法。

好在現在公司的環境,基本都是VS2005. VS2010,估計短時間內用不上。  回復  更多評論   


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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网站| 欧美视频在线观看 亚洲欧| 亚洲乱码一区二区| 亚洲欧美日韩一区二区三区在线| 国产精品欧美在线| 久久久精品欧美丰满| 欧美77777| 在线亚洲观看| 国产一区二区三区丝袜| 美女视频黄 久久| 亚洲精品网站在线播放gif| 亚洲欧美综合另类中字| 韩国视频理论视频久久| 欧美劲爆第一页| 亚洲午夜伦理| 久热精品视频在线| 妖精视频成人观看www| 国产精品推荐精品| 玖玖玖免费嫩草在线影院一区| 亚洲精品人人| 久久夜色精品国产亚洲aⅴ| 日韩亚洲欧美成人一区| 国产亚洲激情在线| 欧美巨乳波霸| 久久久久久久一区二区三区| 日韩系列在线| 免费久久99精品国产自| 国产精品99久久久久久久vr| 韩国v欧美v日本v亚洲v| 欧美视频一区| 免费短视频成人日韩| 亚洲欧美日韩国产综合在线| 亚洲韩国日本中文字幕| 久久国产精品99久久久久久老狼| 亚洲久久一区| 黄色av成人| 国产精品萝li| 欧美日韩国产精品自在自线| 久久精品国产在热久久| 亚洲中字黄色| 亚洲乱亚洲高清| 亚洲第一级黄色片| 久久综合色一综合色88| 午夜日韩在线观看| 亚洲一区二区三区免费在线观看| 亚洲国产成人av在线| 国产午夜精品久久久久久免费视| 欧美午夜理伦三级在线观看| 欧美波霸影院| 裸体一区二区| 久久露脸国产精品| 香蕉免费一区二区三区在线观看 | 亚洲尤物在线| 亚洲国产成人在线| 欧美成人精品在线| 久久婷婷激情| 久久精品亚洲国产奇米99| 亚洲伊人观看| 亚洲综合国产激情另类一区| 99人久久精品视频最新地址| 亚洲欧洲三级| 亚洲精品综合在线| 亚洲欧洲一区二区三区久久| 尤物视频一区二区| 在线观看日韩一区| 有坂深雪在线一区| 在线精品视频一区二区| 在线欧美三区| 亚洲片国产一区一级在线观看| 亚洲福利视频免费观看| 亚洲成色777777女色窝| 亚洲国产一区二区三区a毛片| 亚洲国产一二三| 亚洲欧洲三级| 一区二区免费看| 一区二区三区高清不卡| 亚洲曰本av电影| 亚洲欧美资源在线| 久久国产欧美精品| 欧美/亚洲一区| 欧美成人乱码一区二区三区| 亚洲经典视频在线观看| 日韩亚洲成人av在线| 亚洲午夜电影在线观看| 亚洲欧美综合精品久久成人| 香蕉久久夜色精品国产使用方法| 久久久久久伊人| 欧美暴力喷水在线| 欧美视频免费在线观看| 国产区精品视频| 亚洲国产日韩欧美在线图片| 一本久久青青| 欧美一区免费视频| 免费成人av| 日韩亚洲一区在线播放| 午夜国产精品视频| 老司机免费视频一区二区| 欧美精品久久一区二区| 国产精品国产三级国产专播品爱网| 国产欧美日韩精品丝袜高跟鞋| 国产综合欧美| 日韩亚洲一区二区| 欧美中文在线观看国产| 免费在线播放第一区高清av| 亚洲欧洲在线观看| 午夜伦欧美伦电影理论片| 男同欧美伦乱| 国产伦理精品不卡| 亚洲国产一区视频| 午夜精品在线视频| 欧美成人亚洲成人| 亚洲午夜精品一区二区三区他趣| 久久久久久色| 国产精品私人影院| 亚洲片在线资源| 久久成人免费| 99亚洲一区二区| 麻豆成人小视频| 国产伦精品一区二区三区视频孕妇| 最新成人在线| 久久国产主播| 日韩午夜剧场| 男人天堂欧美日韩| 国产综合一区二区| 亚洲综合日韩在线| 欧美激情中文字幕一区二区| 亚洲欧美国产精品专区久久| 欧美日本中文字幕| 在线观看精品| 久久精品麻豆| 中日韩视频在线观看| 欧美激情视频在线播放| 精品成人一区| 久久激情五月丁香伊人| 一本色道久久88综合日韩精品| 久久一区二区三区四区| 国产伦精品一区二区三区高清版| 国产精品99久久久久久www| 亚洲国产成人精品久久| 久久精品99久久香蕉国产色戒| 国产精品美女久久| 亚洲在线1234| 9国产精品视频| 欧美激情91| 亚洲三级免费电影| 欧美激情a∨在线视频播放| 欧美伊久线香蕉线新在线| 国产精品成人免费| 亚洲一品av免费观看| 亚洲日本欧美在线| 欧美精品在线一区二区| 亚洲麻豆视频| 亚洲经典一区| 欧美区一区二| 艳妇臀荡乳欲伦亚洲一区| 亚洲国产精品一区二区第四页av | 欧美色123| 一区二区三区高清| 日韩一级免费| 欧美三级视频在线播放| 亚洲小说欧美另类婷婷| 一区二区三区欧美| 国产精品亚洲美女av网站| 欧美一级精品大片| 欧美一区二区三区久久精品 | 亚洲第一二三四五区| 欧美顶级艳妇交换群宴| 女人香蕉久久**毛片精品| 亚洲国产欧美在线人成| 亚洲国产欧美在线人成| 欧美精品一区二区在线观看 | 亚洲影院免费观看| 国产精品一区二区三区久久久| 欧美在线视频网站| 久久精品日产第一区二区三区| 尤物在线观看一区| 亚洲国产精品va在看黑人| 欧美精品999| 亚洲欧美综合网| 久久国产精品一区二区三区| 一区二区三区在线免费观看| 亚洲第一页在线| 欧美视频中文一区二区三区在线观看 | 国产欧美高清| 葵司免费一区二区三区四区五区| 美女诱惑一区| 亚洲综合精品四区| 欧美专区日韩视频| 亚洲日本欧美在线| 在线视频日韩精品| 狠狠色香婷婷久久亚洲精品| 欧美激情亚洲| 国产精品久久久久7777婷婷| 久久天堂精品| 欧美日韩免费视频|