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

行進(jìn)中開(kāi)火

C++夜未眠

MFC中一個(gè)危險(xiǎn)的Bug

 

上次說(shuō)日本海嘯警報(bào)的時(shí)候,程序出錯(cuò)。在解析代碼的時(shí)候,發(fā)現(xiàn)了MFC中的一個(gè)Bug。

一。問(wèn)題的產(chǎn)生。

這個(gè)程序,用來(lái)處理日本各種天氣預(yù)報(bào)數(shù)據(jù),包括災(zāi)害的預(yù)報(bào)。如果地震,臺(tái)風(fēng)之類的自然災(zāi)害到來(lái),程序會(huì)把預(yù)報(bào)數(shù)據(jù)進(jìn)行處理,生成相應(yīng)的警報(bào)信息,并在電視上面顯示滾動(dòng)的字幕來(lái)提示。程序本身,是幾年前公司的其他人寫(xiě)的。里面有涉及到文件讀寫(xiě)的地方,有很多地方,用了MFC中自帶的文件讀寫(xiě)類CStdioFile。

CStdioFile這個(gè)文件讀寫(xiě)類,估計(jì)大家都不陌生。這個(gè)類的父類,是CFile類。CStdioFile類本身的功能也很簡(jiǎn)單。CStdioFile類有一個(gè)成員函數(shù)是ReadString,函數(shù)的定義如下:

    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函數(shù)能直接讀取文本中的一行數(shù)據(jù)到CString中,很方便。讀到文件結(jié)尾,沒(méi)有讀出任何數(shù)據(jù)的時(shí)候,返回FALSE。很簡(jiǎn)單的函數(shù),但恰恰是這個(gè)函數(shù)有Bug。

程序在處理數(shù)據(jù)的時(shí)候,會(huì)生成一些臨時(shí)文件,然后會(huì)讀取這些臨時(shí)文件中的數(shù)據(jù),讀取操作,正是用的CStdioFile的ReadString函數(shù)。讀取流程很簡(jiǎn)單:

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

當(dāng)時(shí)的現(xiàn)象為,讀取到最后一行,總是直接返回FALSE,怎么也讀不出最后一行來(lái)。看了看文件的最后一行,包含2176個(gè)字符的數(shù)據(jù),沒(méi)有換行符。沒(méi)有任何異常啊。當(dāng)時(shí)沒(méi)想到是MFC的Bug,因?yàn)橐郧坝羞@樣那樣的毛病,多數(shù)是預(yù)報(bào)數(shù)據(jù)本身有問(wèn)題,所以這次也是先分析數(shù)據(jù)了。分析來(lái)分析去,沒(méi)發(fā)現(xiàn)這次的數(shù)據(jù)有什么異常。后來(lái)發(fā)現(xiàn)如果最后一行的文件不是2176個(gè)字符,就能正常讀出來(lái)。奇了怪了,2176也不是什么特殊長(zhǎng)度啊。實(shí)驗(yàn)了幾次后,覺(jué)的是在不對(duì)勁。莫非是MFC的Bug?

二。發(fā)現(xiàn)問(wèn)題所在

決定看看MFC的代碼再說(shuō)。做了個(gè)簡(jiǎn)單的測(cè)試程序,跟到MFC代碼里一看,果然是MFC的問(wèn)題!測(cè)試代碼如下:

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

測(cè)試代碼很簡(jiǎn)單,讀text.txt文件中的每一行,然后打印出來(lái)。還是2176個(gè)字符就不行。確定了不是數(shù)據(jù)的問(wèn)題,就是MFC代碼本身的Bug。

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

BOOL CStdioFile::ReadString(CString& rString)
{
    ASSERT_VALID(this);
    rString = &afxChNil;    // empty string without deallocating
    const int nMaxSize = 128;  //臨時(shí)字符串的長(zhǎng)度
    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') // 最后結(jié)果中,去掉回車符
        rString.GetBufferSetLength(nLen-1); 
    return lpszResult != NULL;  // 這里就是Bug的關(guān)鍵。返回值不對(duì)!
}

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

這里就看出問(wèn)題所在了,2176個(gè)字符,正好是128的17倍!也就是說(shuō),只要文件最后一行是128倍數(shù)個(gè)字符,就一定會(huì)返回FALSE。

為什么會(huì)這樣呢,因?yàn)镽eadString在每次讀取128個(gè)字符的時(shí)候,用lpszResult指向讀取到的字符串。如果讀滿了128個(gè)字符,就繼續(xù)讀,如果讀到的字符不夠128個(gè),那么就結(jié)束讀取。

當(dāng)一行數(shù)據(jù)正好為128的倍數(shù),又沒(méi)有回車符的時(shí)候,會(huì)發(fā)生什么呢?比如最后一行數(shù)據(jù)是128個(gè),那么,讀一次128個(gè)字符,會(huì)繼續(xù)讀下一次,但是下一次的讀取,什么也沒(méi)有讀到,lpszResult就指向NULL,最后的返回值,是return lpszResult != NULL; 所以返回FALSE。

但之前讀到的128個(gè)字符,已經(jīng)在CString里面了。也就是說(shuō)實(shí)際上讀取已經(jīng)成功了,但還是返回了FALSE。返回值不恰當(dāng)!

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

這個(gè)Bug,只會(huì)影響到最后一行數(shù)據(jù)。因?yàn)槿绻袚Q行符的存在,lpszResult就不會(huì)為NULL。

三。解決方法

要解決這個(gè)問(wèn)題,也簡(jiǎn)單,修改一下判斷ReadString成功與否的語(yǔ)句:

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

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

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

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

這個(gè)程序,是用VC6.0做的,我有看了看VC2005中的代碼,發(fā)現(xiàn)這個(gè)Bug被修復(fù)了,代碼如下:

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;

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

MFC的這個(gè)Bug比較隱蔽,平常不容易發(fā)現(xiàn),但一旦遇到特殊長(zhǎng)度的數(shù)據(jù),就會(huì)表現(xiàn)異常。所以,在用VC6.0開(kāi)發(fā)的時(shí)候,盡量避免使用ReadString,或者在使用中,多判斷一步讀取出來(lái)的CString長(zhǎng)度。避開(kāi)這個(gè)Bug。

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

評(píng)論

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

解決方法應(yīng)該是避免使用VC6進(jìn)行開(kāi)發(fā)。  回復(fù)  更多評(píng)論   

# re: MFC中一個(gè)危險(xiǎn)的Bug 2010-03-27 23:47 Jakcie

的確應(yīng)該避免用VC6開(kāi)發(fā)。但使用VC6的人,還是不少啊。  回復(fù)  更多評(píng)論   

# re: MFC中一個(gè)危險(xiǎn)的Bug 2010-03-28 11:17 唐風(fēng)

1. 遺留代碼不可替代
2. 升級(jí)開(kāi)發(fā)環(huán)境的預(yù)算和必要性
3. 有決策權(quán)開(kāi)發(fā)人員的“技術(shù)慣性”
使得VC6到現(xiàn)在還這么有“生命”力。感覺(jué)就像IE6一樣,哈哈,有種種的不好,可偏還是有那么多用戶。

反正我是等VS2010出來(lái)就準(zhǔn)備XX的,嘿嘿。

  回復(fù)  更多評(píng)論   

# re: MFC中一個(gè)危險(xiǎn)的Bug 2010-03-28 16:08 Jakcie

是啊。尤其是像這種遺留代碼,的確是沒(méi)有辦法。

好在現(xiàn)在公司的環(huán)境,基本都是VS2005. VS2010,估計(jì)短時(shí)間內(nèi)用不上。  回復(fù)  更多評(píng)論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久精品一区中文字幕| 亚洲福利视频一区| 亚洲视频在线观看视频| 亚洲国产欧美一区二区三区同亚洲 | 欧美一级日韩一级| 国产日韩一区二区三区在线播放| 欧美亚洲免费在线| 欧美亚洲自偷自偷| 亚洲国产色一区| 99国产精品久久久久老师| 国产精品高精视频免费| 久久激情一区| 蜜臀av性久久久久蜜臀aⅴ四虎| 日韩午夜在线电影| 亚洲欧美自拍偷拍| 在线欧美不卡| 一区二区三区免费观看| 国产三区二区一区久久| 欧美国产精品日韩| 国产精品国产福利国产秒拍| 欧美资源在线观看| 欧美激情bt| 久久久久国产精品www| 免费一级欧美片在线播放| 一区二区三区日韩精品视频| 午夜欧美不卡精品aaaaa| 亚洲日本中文字幕| 午夜精品国产| 99热这里只有精品8| 欧美一区二区| 亚洲午夜久久久久久久久电影院| 欧美一区日韩一区| 亚洲最黄网站| 久久精品中文字幕一区二区三区| 亚洲每日在线| 久久久久成人网| 亚洲在线视频网站| 欧美岛国激情| 久热国产精品| 国产字幕视频一区二区| 一区二区精品| 亚洲麻豆av| 久久亚洲电影| 久久久亚洲精品一区二区三区| 欧美精品在线免费播放| 欧美mv日韩mv国产网站| 国产酒店精品激情| 宅男噜噜噜66一区二区66| 亚洲激情国产| 麻豆国产精品va在线观看不卡| 久久爱www.| 国产精品有限公司| 在线综合亚洲| 亚洲一区二区在线看| 欧美精品一区二区三区在线播放| 欧美电影免费观看网站| 农夫在线精品视频免费观看| 亚洲欧洲在线一区| 红杏aⅴ成人免费视频| 亚洲午夜精品国产| 亚洲欧美国产精品桃花| 欧美日本高清一区| 亚洲免费激情| 在线亚洲高清视频| 欧美日韩国产一区二区三区| 亚洲精品一区二区三区樱花 | 欧美成人高清视频| 欧美成年人视频网站欧美| 激情另类综合| 久久久久网站| 欧美刺激性大交免费视频 | 欧美日韩日日骚| 亚洲毛片av在线| 亚洲先锋成人| 国产精品日韩专区| 午夜精品久久久久| 久久中文字幕导航| 亚洲激情av| 欧美日韩久久久久久| 一区二区三欧美| 欧美在线www| 狠狠综合久久av一区二区老牛| 久久久亚洲精品一区二区三区| 免费亚洲网站| 亚洲欧美视频| 久久久99爱| 91久久精品国产91久久性色tv| 欧美日本不卡高清| 亚洲欧美日韩在线| 欧美gay视频激情| 99www免费人成精品| 国产精品美女999| 久久av一区| 亚洲人成网站精品片在线观看| 宅男噜噜噜66一区二区| 国产一区二区欧美| 欧美国产视频在线| 亚洲欧美亚洲| 亚洲国产精品黑人久久久| 亚洲一区网站| 伊大人香蕉综合8在线视| 欧美亚洲一级片| 亚洲电影欧美电影有声小说| 亚洲一区国产精品| 亚洲国产婷婷| 国产欧美视频一区二区三区| 久久综合久久美利坚合众国| 一区二区三区精密机械公司| 免费日韩成人| 午夜亚洲视频| 日韩视频在线播放| 伊人夜夜躁av伊人久久| 国产精品xxxxx| 免费国产一区二区| 亚洲自拍啪啪| 99精品国产高清一区二区| 久久一日本道色综合久久| 亚洲欧美怡红院| 一本色道久久综合狠狠躁的推荐| 狠狠色噜噜狠狠狠狠色吗综合| 国产精品久久久久999| 欧美精品一区二区三区视频 | 一本大道av伊人久久综合| 免费在线观看精品| 国产精品自拍视频| 一区二区三区国产在线| 欧美激情精品久久久久久变态| 久久精品国产亚洲5555| 亚洲永久免费视频| 日韩午夜三级在线| 亚洲精品影院在线观看| 亚洲国产成人在线视频| 精品999日本| 国产一区99| 国产午夜精品一区理论片飘花| 国产精品久久国产精麻豆99网站| 欧美日韩1区2区3区| 欧美成人自拍| 欧美黄色网络| 欧美紧缚bdsm在线视频| 欧美国产日韩xxxxx| 欧美精品福利在线| 欧美精品一二三| 欧美美女喷水视频| 欧美激情一区二区三区四区| 欧美激情1区| 欧美日韩免费观看一区三区 | 久久综合国产精品台湾中文娱乐网| 欧美一区二区三区四区高清 | 欧美成人免费视频| 久久躁日日躁aaaaxxxx| 久久久一本精品99久久精品66| 欧美在线观看网址综合| 久久蜜桃精品| 欧美激情在线| 国产精品成人一区二区三区夜夜夜 | 国内精品久久久久久久影视蜜臀| 国产色视频一区| 今天的高清视频免费播放成人| 在线看成人片| 一本色道久久综合亚洲91| 亚洲午夜精品久久久久久浪潮| 亚洲欧美激情视频在线观看一区二区三区 | 国产欧美一区二区色老头| 国产午夜精品麻豆| 亚洲国产精品成人一区二区| 亚洲免费观看| 欧美在线日韩在线| 欧美不卡视频一区发布| 日韩视频永久免费| 欧美在线www| 欧美久久久久久久久| 国产精品毛片a∨一区二区三区| 国产亚洲一本大道中文在线| 亚洲国产精品尤物yw在线观看| 一区二区电影免费在线观看| 久久www免费人成看片高清 | 欧美一区二区在线视频| 欧美成人蜜桃| 亚洲男人第一网站| 欧美成人国产一区二区| 国产精品一区久久久久| 亚洲日本视频| 久久精品国产999大香线蕉| 91久久精品国产91久久| 午夜国产精品视频| 亚洲欧美国内爽妇网| 久久精品国产欧美亚洲人人爽| 亚洲丶国产丶欧美一区二区三区 | 欧美亚洲免费在线| 欧美激情一区二区三区四区| 亚洲视频一区二区在线观看| 久久九九精品99国产精品| 欧美人与性动交a欧美精品| 好吊日精品视频| 午夜精品视频在线观看| 亚洲国产美女久久久久| 久久精品国产2020观看福利| 欧美视频精品在线| 亚洲精品久久久久|