注册 登录
编程论坛 VC++/MFC

关于MFC使用 Unicode 字符集出现乱码的问题!!

ozc2701 发布于 2010-08-08 00:07, 10133 次点击
我用MFC写了一个对话框。
在工程属性里使用Unicode字符集。然后加入一个编辑框空间。在文本编辑框内输入汉字。然后给空间关联了一个CString的变量,然后保存到TXT文件里是乱码。
但是如果在工程属性里使用多字节字符集的话。重复上述步骤,没有乱码出现。。。
求解。。。
为什么使用多字节字符集的话,对话框的按钮都变成Win98那种模式的了?而使用Unicode字符集则有立体感。。
这让我很难抉择啊。。。我是个画面党。。那种按钮确实太难看了。。。
有什么两全的办法吗??
望高人给出解答。。。
谢谢。。。
(注:使用两种字符集都能正常的打开一个TXT文件。。)
打开方式的代码是:
    CFileDialog fileDlg(TRUE);
    fileDlg.m_ofn.lpstrTitle=TEXT("打开文件");
    fileDlg.m_ofn.lpstrFilter=TEXT("文本文件(*.txt)\0*.txt\0\0");
    if(IDOK==fileDlg.DoModal())
    {
        CFile file(fileDlg.GetPathName(),CFile::modeRead);
        char* ch=new char[file.GetLength()+1];
        ch[file.GetLength()]=0;
        file.Read(ch,file.GetLength());
        file.Close();
        m_Text=ch;
    }
    UpdateData(FALSE);

m_Text是控件关联的CString变量,直接使用会出错。。。因为找不出什么原因,我new了一个char数组。然后吧char数组的内容给了m_Text。。。。
15 回复
#2
ozc27012010-08-08 00:08
写入文件的代码是:
    UpdateData();
    CFileDialog fileDlg(FALSE);
    fileDlg.m_ofn.lpstrTitle=TEXT("保存文件");
    fileDlg.m_ofn.lpstrFilter=TEXT("文本文件(*.txt)\0*.txt\0\0");
    fileDlg.m_ofn.lpstrDefExt=TEXT("txt");
    if (IDOK==fileDlg.DoModal())
    {
        CFile file(fileDlg.GetPathName(),CFile::modeCreate | CFile::modeWrite);
        file.Write(m_Text,m_Text.GetLength());
        file.Close();
    }
#3
东海一鱼2010-08-08 13:51
写入Unicode文本时,要在文件头部加入Unicode文本标志0XFEFF。
#4
ozc27012010-08-08 14:07
回复 3楼 东海一鱼
高人能详细点吗?
#5
东海一鱼2010-08-08 14:51
程序代码:
#include <stdio.h>

#ifndef _UNICODE
#define _UNICODE            //使用UNICODE编码
#endif

#include <Afx.h>           //为了使用CString类

const int UNICODE_TXT_FLG = 0xFEFF;  //UNICODE文本标示

int main()
{
    FILE* WriteF;
   

    CString Wstr = _T("一个测试写入文本");
    WriteF = fopen("d:\\test.txt","w+");

    if(WriteF)
    {
        fwrite(&UNICODE_TXT_FLG,2,1,WriteF);  //写入头部
        fwrite(Wstr.GetBuffer(10),Wstr.GetLength() * 2,1,WriteF);
        fclose(WriteF);
    }      
   
    return 0;
}

#6
ozc27012010-08-08 15:12
回复 5楼 东海一鱼
如果用CFile类该怎么写?
我用CFile类的Write写 还是乱码。。。
#7
东海一鱼2010-08-08 15:25
程序代码:
#include <stdio.h>

#ifndef _UNICODE
#define _UNICODE
#endif

#include <Afx.h>



const int UNICODE_TXT_FLG = 0xFEFF;

// int main()
// {
//     FILE* WriteF;
//   
//
//     CString Wstr = _T("一个测试写入文本");
//     WriteF = fopen("d:\\test.txt","w+");
//
//     if(WriteF)
//     {
//         fwrite(&UNICODE_TXT_FLG,2,1,WriteF);
//         fwrite(Wstr.GetBuffer(10),Wstr.GetLength() * 2,1,WriteF);
//         fclose(WriteF);
//     }      
//   
//     return 0;
// }

int main()
{
    CFile WriteF;
    CString Wstr = _T("一个测试写入文本");

    WriteF.Open(_T("d:\\test.txt"),CFile::modeCreate | CFile::modeWrite);
    if(WriteF)
    {
        WriteF.Write(&UNICODE_TXT_FLG,2);
        WriteF.Write(Wstr.GetBuffer(10),Wstr.GetLength()*2);
        WriteF.Flush();
        WriteF.Close();      
    }
    return 0;

}

#8
ozc27012010-08-08 16:07
回复 7楼 东海一鱼
谢谢您,写入问题解决了。。
不过。。。华丽丽的读取又出错了。。。
用原来的方法是乱码。。
改成这样出现的是空白 - -|
CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
        ReadF.SeekToBegin();
        ReadF.Read(m_Text.GetBuffer(10),m_Text.GetLength()*2);
        ReadF.Close();

改成这样还是空白
ReadF.Open(fileDlg.GetPathName(),CFile::modeRead);
ReadF.Read(m_Text.GetBuffer(10),m_Text.GetLength()*2)
ReadF.Close();


对字符问题是一窍不通啊。。。。无奈。。
#9
ozc27012010-08-08 16:29
回复 7楼 东海一鱼
不好意思  是我疏忽了。。。
        ReadF.Read(m_Text.GetBuffer(10),m_Text.GetLength()*2);
写错了 - -|
应该是   ReadF.Read(m_Text.GetBuffer(10),ReadF.GetLength());
不过提示触发了一个断点。。。
得再研究研究了。。。。
#10
ozc27012010-08-08 16:41

        m_Text.Empty();
        CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
        TCHAR* temp=new TCHAR[ReadF.GetLength()/2+1];
        ReadF.Read(temp,ReadF.GetLength());
        temp[ReadF.GetLength()/2]=0;
        ReadF.Close();
        m_Text=temp;
        delete temp;

这样的话能正常的写入和打开Unicode字符集的TXT文件了。。
可是那些非Unicode的文件就变成乱码了。。。
是不是应该加个判断语句???应该怎么加??
#11
东海一鱼2010-08-08 16:46
按二进制读取文件前两字节,判断是否是0xff,0xfe。如是就按UNICODE文本处理。否则按ASCII文本处理。
#12
ozc27012010-08-08 17:13
这个判断该怎么加呢????
#13
ozc27012010-08-08 17:27
    CFileDialog fileDlg(TRUE);
    fileDlg.m_ofn.lpstrTitle=TEXT("打开文件");
    fileDlg.m_ofn.lpstrFilter=TEXT("文本文件(*.txt)\0*.txt\0\0");
    if(IDOK==fileDlg.DoModal())
    {
        m_Text.Empty();
        CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
        TCHAR* temp=new TCHAR[ReadF.GetLength()/2+1];
        ReadF.Read(temp,ReadF.GetLength());
        temp[ReadF.GetLength()/2]=0;
        ReadF.Close();
        if (temp[0]==0xFEFF)
        {
            m_Text=temp;
            delete temp;
        }
        else
        {
            CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
            char* temp=new char[ReadF.GetLength()+1];
            ReadF.Read(temp,ReadF.GetLength());
            temp[ReadF.GetLength()]=0;
            ReadF.Close();
            m_Text=temp;
            delete temp;
        }
        UpdateData(FALSE);
    }




写完了。。。正常了。。。
代码可能有啰嗦的地方。。
望指出缺点。
#14
东海一鱼2010-08-08 18:12
以下是引用ozc2701在2010-8-8 17:27:28的发言:

    CFileDialog fileDlg(TRUE);
    fileDlg.m_ofn.lpstrTitle=TEXT("打开文件");
    fileDlg.m_ofn.lpstrFilter=TEXT("文本文件(*.txt)\0*.txt\0\0");
    if(IDOK==fileDlg.DoModal())
    {
        m_Text.Empty();
        CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
        TCHAR* temp=new TCHAR[ReadF.GetLength()/2+1];
        ReadF.Read(temp,ReadF.GetLength());
        temp[ReadF.GetLength()/2]=0;
        ReadF.Close();
        if (temp[0]==0xFEFF)
        {
            m_Text=temp;
            delete temp;
        }
        else
        {
            CFile ReadF(fileDlg.GetPathName(),CFile::modeRead);
            char* temp=new char[ReadF.GetLength()+1];
            ReadF.Read(temp,ReadF.GetLength());
            temp[ReadF.GetLength()]=0;
            ReadF.Close();
            m_Text=temp;
            delete temp;
        }
        UpdateData(FALSE);
    }




写完了。。。正常了。。。
代码可能有啰嗦的地方。。
望指出缺点。
   很好,自己完成的才有成就感。
   你可以试着一次读取文件。然后根据文件头来做不同的处理。而不用分两次。
#15
yunlongdell2011-05-20 11:34
把编程的环境更换成Unicode环境,在工程里的设置C++的预处理编译定义框中加_UNICODE在连接选项卡中更改输出的入口点,换成wWinMainCRTStartup
然后就开始Unicode编程    相关的字符函数都将变化。有点麻烦。
#16
jianghuai_j2014-07-11 10:29
小端机,mfc读取Unicode(be)文件显示在edit control是乱码,如果正常显示怎么解决
1