• 关于CString的=操作符


    检查自已的代码时,发现一个比较弱智的错误,是关于转码的,和CString的=操作符有关。
      为了能简单重现,我模拟了一下代码:

      //未有UNICODE或_UNICODE定义
      USES_CONVERSION;
      CString sTest(_T("test中文"));
      sTest = A2W(sTest);
      ::MessageBoxW(NULL,(LPCWSTR)sTest.GetBuffer(0),L"test中文标题",MB_OK);

      输出:
          对话框的内容区为乱码。

      我的想象中:A2W取得宽字符串,然后赋值给CString,CString中应该保存的是宽字符串。
      显然,并不如我想,想是没用的,仔细看一下出错的原因吧。

      先看看CString=操作符源码:
      #ifdef _UNICODE
      ……
      #else //!_UNICODE
      const CString& CString::operator=(LPCWSTR lpsz)
      {
         int nSrcLen = lpsz != NULL ? wcslen(lpsz) : 0;
         AllocBeforeWrite(nSrcLen*2);
         _wcstombsz(m_pchData, lpsz, (nSrcLen*2)+1);
         ReleaseBuffer();
         return *this;
      }
      哇,_wcstombsz在干什么?
      再看看它的代码:
      int AFX_CDECL _wcstombsz(char* mbstr, const wchar_t* wcstr, size_t count)
      {
         if (count == 0 && mbstr != NULL)
            return 0;
         int result = ::WideCharToMultiByte(CP_ACP, 0, wcstr, -1,
             mbstr, count, NULL, NULL);
         ASSERT(mbstr == NULL || result <= (int)count);
         if (result > 0)
             mbstr[result-1] = 0;
         return result;
      }
      嘿,居然是在转码噢。
      好了,答案找到了,CString的 = 操作符居然会智能的做转码,所以上面代码的sTest最终得到的仍然是ANSI字符串。因此,输出是乱码。

      结论:CString = 操作符是有转码功能的,会将右侧的字符转为TCHAR的当前编码方式。
      换句话说,如果你想实现A2T或是W2T,那你根本不需要用ATL的宏,直接用CString的=就可以了,呵呵。

      好了,改一下前文的代码,不能想当然,老老实实的做吧。
      USES_CONVERSION;
      CString sTest(_T("test中文"));
      LPWSTR lpwszTest = A2W(sTest);
      ::MessageBoxW(NULL,lpwszTest.GetBuffer(0),L"test中文标题",MB_OK);

  • 相关阅读:
    我所理解的三次握手
    网络舆情——初步了解
    【转载】位运算的密码
    【转载】基础排序算法简介
    【原创】关于hashcode和equals的不同实现对HashMap和HashSet集合类的影响的探究
    【原创】Java移位运算
    【原创】MapReduce计数器
    【原创】Hadoop机架感知对性能调优的理解
    【原创】一个复制本地文件到Hadoop文件系统的实例
    【转载】JAVA IO 流的总结
  • 原文地址:https://www.cnblogs.com/lzjsky/p/1947809.html
Copyright © 2020-2023  润新知