• <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>

            行進(jìn)中開火

            C++夜未眠

            MFC中一個危險的Bug

             

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

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

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

            CStdioFile這個文件讀寫類,估計大家都不陌生。這個類的父類,是CFile類。CStdioFile類本身的功能也很簡單。CStdioFile類有一個成員函數(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é)尾,沒有讀出任何數(shù)據(jù)的時候,返回FALSE。很簡單的函數(shù),但恰恰是這個函數(shù)有Bug。

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

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

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

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

            決定看看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個字符就不行。確定了不是數(shù)據(jù)的問題,就是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') // 最后結(jié)果中,去掉回車符
            
                    rString.GetBufferSetLength(nLen-1); 
            
                return lpszResult != NULL;  // 這里就是Bug的關(guān)鍵。返回值不對!
            
            }

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

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

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

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

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

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

            這個Bug,只會影響到最后一行數(shù)據(jù)。因為如果有換行符的存在,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中的代碼,發(fā)現(xiàn)這個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;

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

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

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

            評論

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

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

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

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

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

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

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

              回復(fù)  更多評論   

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

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

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

            国产农村妇女毛片精品久久| 久久国产精品波多野结衣AV| 亚洲国产精品久久久久婷婷软件| 精品国产乱码久久久久久人妻| 亚洲а∨天堂久久精品9966| A级毛片无码久久精品免费| 国产亚洲精品自在久久| 久久棈精品久久久久久噜噜| 久久久久AV综合网成人| 狠狠色丁香久久婷婷综合五月| 日韩AV无码久久一区二区 | 国产成人久久777777| 欧美精品一区二区精品久久 | 久久免费视频网站| 精品久久久无码人妻中文字幕豆芽 | 亚洲伊人久久精品影院| 色婷婷久久综合中文久久蜜桃av| 综合久久一区二区三区 | 久久99国产综合精品免费| 91精品国产综合久久精品| 免费精品99久久国产综合精品| 丰满少妇人妻久久久久久4| 久久久久综合国产欧美一区二区| 亚洲一区精品伊人久久伊人| 99久久综合国产精品免费| 久久亚洲精精品中文字幕| 久久免费精品一区二区| 亚洲伊人久久成综合人影院 | 精品久久人人爽天天玩人人妻| 久久国产视频99电影| 久久久久99这里有精品10 | 人妻无码αv中文字幕久久琪琪布 人妻无码久久一区二区三区免费 人妻无码中文久久久久专区 | 思思久久好好热精品国产| 日产精品99久久久久久| 国产三级观看久久| 国产精品成人久久久| 久久免费精品视频| 18禁黄久久久AAA片| 久久成人影院精品777| 久久无码AV一区二区三区| 天天久久狠狠色综合|