duilib是一个比较常见的界面库,闲来无事看看别人写的代码,跟自己写的一比,

才看到了差距呀,感觉自己写的乱七八糟,keep moving

CduiString是duilib提供的一个字符串类,功能是够用的,做duilib项目可以直接拿来用

首先看看头文件定义:

 class UILIB_API CDuiString
{
public:
enum { MAX_LOCAL_STRING_LEN = }; CDuiString();
CDuiString(const TCHAR ch);
CDuiString(const CDuiString& src);
CDuiString(LPCTSTR lpsz, int nLen = -);
~CDuiString(); void Empty();
int GetLength() const;
bool IsEmpty() const;
TCHAR GetAt(int nIndex) const;
void Append(LPCTSTR pstr);
void Assign(LPCTSTR pstr, int nLength = -);
LPCTSTR GetData() const; void SetAt(int nIndex, TCHAR ch);
operator LPCTSTR() const; TCHAR operator[] (int nIndex) const;
const CDuiString& operator=(const CDuiString& src);
const CDuiString& operator=(const TCHAR ch);
const CDuiString& operator=(LPCTSTR pstr);
#ifdef _UNICODE
const CDuiString& CDuiString::operator=(LPCSTR lpStr);
const CDuiString& CDuiString::operator+=(LPCSTR lpStr);
#else
const CDuiString& CDuiString::operator=(LPCWSTR lpwStr);
const CDuiString& CDuiString::operator+=(LPCWSTR lpwStr);
#endif
CDuiString operator+(const CDuiString& src) const;
CDuiString operator+(LPCTSTR pstr) const;
const CDuiString& operator+=(const CDuiString& src);
const CDuiString& operator+=(LPCTSTR pstr);
const CDuiString& operator+=(const TCHAR ch); bool operator == (LPCTSTR str) const;
bool operator != (LPCTSTR str) const;
bool operator <= (LPCTSTR str) const;
bool operator < (LPCTSTR str) const;
bool operator >= (LPCTSTR str) const;
bool operator > (LPCTSTR str) const; int Compare(LPCTSTR pstr) const;
int CompareNoCase(LPCTSTR pstr) const; void MakeUpper();
void MakeLower(); CDuiString Left(int nLength) const;
CDuiString Mid(int iPos, int nLength = -) const;
CDuiString Right(int nLength) const; int Find(TCHAR ch, int iPos = ) const;
int Find(LPCTSTR pstr, int iPos = ) const;
int ReverseFind(TCHAR ch) const;
int Replace(LPCTSTR pstrFrom, LPCTSTR pstrTo); int __cdecl Format(LPCTSTR pstrFormat, ...);
int __cdecl SmallFormat(LPCTSTR pstrFormat, ...); protected:
LPTSTR m_pstr;//指向缓冲区的指针,可能指向m_szBuffer,也可能指向别的缓冲区
TCHAR m_szBuffer[MAX_LOCAL_STRING_LEN + ];//储存字符串的字符数组
};

看了定义之后其中的api大部分都会使用了吧

下面抓几个重要的函数来分析一下吧.

 void CDuiString::Append(LPCTSTR pstr)
{
int nNewLength = GetLength() + (int) _tcslen(pstr);
if( nNewLength >= MAX_LOCAL_STRING_LEN ) {
if( m_pstr == m_szBuffer ) {
m_pstr = static_cast<LPTSTR>(malloc((nNewLength + ) * sizeof(TCHAR)));
_tcscpy(m_pstr, m_szBuffer);
_tcscat(m_pstr, pstr);
}
else {
m_pstr = static_cast<LPTSTR>(realloc(m_pstr, (nNewLength + ) * sizeof(TCHAR)));
_tcscat(m_pstr, pstr);
}
}
else {
if( m_pstr != m_szBuffer ) {//防止m_pstr指向别的地方
free(m_pstr);
m_pstr = m_szBuffer;
}
_tcscat(m_szBuffer, pstr);
}
}

Append函数用于在末尾添加字符串,首先判断添加后的的字符串长度是否比默认数组中能储存的长。1.超过了:先判断当前指针是否指向m_szBuffer,是的话则新申请一段内存,将原有的内容复制过去,再加上新增的即可,否的话先释放m_pstr中的内存再复制内容,新增。(注意这里realloc函数会在申请内存给m_pstr后将原有内容复制过去,并且释放原来m_pstr的内存)2.没有超过则直接添加即可

 void CDuiString::Assign(LPCTSTR pstr, int cchMax)
{
if( pstr == NULL ) pstr = _T("");
cchMax = (cchMax < ? (int) _tcslen(pstr) : cchMax);
if( cchMax < MAX_LOCAL_STRING_LEN ) {
if( m_pstr != m_szBuffer ) {
free(m_pstr);
m_pstr = m_szBuffer;
}
}
else if( cchMax > GetLength() || m_pstr == m_szBuffer ) {
if( m_pstr == m_szBuffer ) m_pstr = NULL;//若指向m_szBuffer,不能将其释放,先指向NULL
m_pstr = static_cast<LPTSTR>(realloc(m_pstr, (cchMax + ) * sizeof(TCHAR)));
}
_tcsncpy(m_pstr, pstr, cchMax);
m_pstr[cchMax] = _T('\0');
}

Assign这个函数设计的十分精妙,很多地方都用到,它主要用于重新赋值操作。cchMax表示截取pstr的范围[0,cchMax),若m_szBuffer长度足以存储,则直接赋值,把指针指过去;若不够长,重新申请内存赋值

 int CDuiString::Replace(LPCTSTR pstrFrom, LPCTSTR pstrTo)
{
CDuiString sTemp;
int nCount = ;
int iPos = Find(pstrFrom);
if( iPos < ) return ;
int cchFrom = (int) _tcslen(pstrFrom);
int cchTo = (int) _tcslen(pstrTo);
while( iPos >= ) {
sTemp = Left(iPos);
sTemp += pstrTo;
sTemp += Mid(iPos + cchFrom);
Assign(sTemp);
iPos = Find(pstrFrom, iPos + cchTo);
nCount++;
}
return nCount;
}

替换字符串,首先查找是否存在需要替换的字符串,没有则直接返回0,存在的话,将其分为三段,中间一段替换,用sTemp储存,再重新赋值sTemp,如此往复替换所有的pstrFrom。

 int CDuiString::Format(LPCTSTR pstrFormat, ...)
{
LPTSTR szSprintf = NULL;
va_list argList;
int nLen;
va_start(argList, pstrFormat);
nLen = ::_vsntprintf(NULL, , pstrFormat, argList);
szSprintf = (TCHAR*)malloc((nLen + ) * sizeof(TCHAR));
ZeroMemory(szSprintf, (nLen + ) * sizeof(TCHAR));
int iRet = ::_vsntprintf(szSprintf, nLen + , pstrFormat, argList);
va_end(argList);
Assign(szSprintf);
free(szSprintf);
return iRet;
}

Format函数用于格式化字符串,与常用的printf类似。具体实现来看看,va_list argList, 这个是什么东东,一开始确实没见过,看看其定义

typedef char *  va_list;原来是char*,它是用于指向可变参数的,va_start(argList,pstrFormat)后,argList指向第一个可变参数,接下来调用_vsntprintf(NULL,0,pstrFormat,argList),最后将格式化后的字符串重新赋值回去即可。

想了解关于va_list 可参考这里:http://blog.csdn.net/edonlii/article/details/8497704

其实难点也并非很难,只是一些需要注意的小问题总是会被自己忽略,看别人写的好的代码还是挺有收获的。

DuiLib 源码分析之CDuiString的更多相关文章

  1. Duilib源码分析(六)整体流程

    在<Duilib源码分析(一)整体框架>.<Duilib源码分析(二)控件构造器—CDialogBuilder>以及<Duilib源码分析(三)XML解析器—CMarku ...

  2. Duilib源码分析(一)整体框架

    Duilib界面库是一款由杭州月牙儿网络技术有限公司开发的界面开源库,以viksoe项目下的UiLib库的基础上开发(此后也将对UiLib库进行源码分析):通过XML布局界面,将用户界面和处理逻辑彻底 ...

  3. Duilib源码分析(三)XML解析器—CMarkup

    上一节介绍了控件构造器CDialogBuilder,接下来将分析其XML解析器CMarkup: CMarkup:xml解析器,目前内置支持三种编码格式:UTF8.UNICODE.ASNI,默认为UTF ...

  4. Duilib源码分析(四)绘制管理器—CPaintManagerUI—(前期准备一)

    上节中提到在遍历创建控件树后,执行了以下操作:      1. CDialogBuilder构建各控件对象并形成控件树,并返回第一个控件对象pRoot:     2. m_pm.AttachDialo ...

  5. Duilib源码分析(五)UI布局—Layout与各子控件

    接下来,继续分析duilib之UI布局Layout,目前提供的布局有:VerticalLayout.HorizontalLayout.TileLayout.TabLayout.ChildLayout分 ...

  6. Duilib源码分析(四)绘制管理器—CPaintManagerUI

    接下来,分析uilib.h中的UIManager.h,在正式分析CPaintManagerUI前先了解前面的一些宏.结构: 枚举类型EVENTTYPE_UI:定义了UIManager.h中事件通告类型 ...

  7. Duilib源码分析(四)绘制管理器—CPaintManagerUI—(前期准备三)

    接下来,我们将继续分析UIlib.h文件中其他的文件, UIContainer.h, UIRender.h, WinImplBase.h, UIManager.h,以及其他布局.控件等: 1. UIR ...

  8. Duilib源码分析(四)绘制管理器—CPaintManagerUI—(前期准备二)

    接下来,我们继续分析UIlib.h文件中余下的文件,当然部分文件可能顺序错开分析,这样便于从简单到复杂的整个过程的里面,而避免一开始就出现各种不理解的地方. 1. UIManager.h:UI管理器, ...

  9. Duilib源码分析(四)绘制管理器—CPaintManagerUI—(前期准备四)

    接下来,分析uilib.h中的WinImplBase.h和UIManager.h: WinImplBase.h:窗口实现基类,已实现大部分的工作,基本上窗口类均可直接继承该类,可发现该类继承于多个类, ...

随机推荐

  1. GPS部标平台的架构设计(三) 基于struts+spring+hibernate+ibatis+quartz+mina框架开发GPS平台

    注意,此版本是2014年研发的基于Spring2.5和Struts2的版本,此版本的源码仍然销售,但已不再提供源码升级的服务,因为目前我们开发的主流新版本是2015-2016年近一年推出的基于spri ...

  2. Leetcode: Ternary Expression Parser

    Given a string representing arbitrarily nested ternary expressions, calculate the result of the expr ...

  3. Linux中可用于管道操作的命令总结

    在Linux中药进行稍复杂的操作,通常需要借助管道命令"|"多个命令的组合,形式如下: command 1 |  command 2 |  command 3 -- 在linux中 ...

  4. PHP自带防SQL攻击函数区别

    为了防止SQL注入攻击,PHP自带一个功能可以对输入的字符串进行处理,可以在较底层对输入进行安全上的初步处理,也即Magic Quotes.(php.ini magic_quotes_gpc).如果m ...

  5. WIN10 多用户登录

    WIN10 多用户登录 参考下面链接 http://www.mysysadmintips.com/windows/clients/545-multiple-rdp-remote-desktop-ses ...

  6. iOS 简单的分段下载文件

    首先自己写个请求数据的类 首先.h文件 #import <Foundation/Foundation.h> @interface Downloaders : NSObject<NSU ...

  7. JVM之几种垃圾收集器简单介绍

    本文中的垃圾收集器研究背景为:HotSpot+JDK1.7 一.垃圾收集器概述 如上图所示,垃圾回收算法一共有7个,3个属于年轻代.三个属于年老代,G1属于横跨年轻代和年老代的算法. JVM会从年轻代 ...

  8. asp.net调用客户端WebBrowser 进行网站地址截屏

    在asp.net网站中,如果要实现,在文本框中输入一个URL地址,就把该网页的页面整屏截下来,这段时间一直在研究这一块,在网上查了好多资料.自己又整合了一下. 其实也不是想象中的那么难.主要是通过调用 ...

  9. 对c++ public、protected、private关键字的理解

    首先要明确一下: 1.这三个关键字在两种地方会用到,一个是对类的成员变量和成员函数修饰时(比如私有的成员变量,受保护的成员变量·,公有的函数),还有一种是对继承方式的修饰(比如公有继承,保护继承). ...

  10. 不用git将项目push到码云上

    1.在码云上创建一个项目: 2.打开STS(spring Tool Suite)   新建一个Maven(webapp)项目: 3.打开你的码云账号,把码云上的工程的URL复制: 4.重新在另一个目录 ...