QStringLieral是Qt5中新引入的一个用来从“字符串常量”创建QString对象的宏(字符串常量指在源码中由双引号包含的字符串)。在这篇博客我讲解释它的的内部实现和工作原理。
提要
让我们从它的使用环境开始说起:假设你想要在Qt5中从字符串常量初始化一个QString对象,你应该这样:
大多数情况:
(1)使用QStringLiteral(“某字符串”) --如果它最终转会换成QString的话
(2)使用QLatin1String(“某字符串”) --如果使用的函数有支持QLatin1String的重载(比如operator==, operator+, startWith, replace等)的话
我把这段话放在最开始是为了那些不怎么想了解其具体技术细节的人着想。继续阅读你将了解QStringLiteral是如何工作的。
QString的工作方式
QString和Qt中的其他类一样,是一个”隐式共享类“。它唯一的数据成员就是一个指向其“私有”数据的指针。QStringData由malloc函数分配空间,并且在其后(同一块内存块)分配了足够的空间来存放实际的字符数据。
//为了此博客的目标做了简化
struct QStringData {
QtPrivate::RefCount ref; // 对QAtomicInt进行封装
int size; // 字符串的大小
uint alloc : 31 ; // 该字符串数据之后预留的内存数
uint capacityReserved : 1 ; // reserve()使用到的内部细节
qptrdiff offset; // 数据的偏移量 (通常是 sizeof(QStringData))
inline ushort *data()
{ return reinterpret_cast < ushort *>( reinterpret_cast <char *>( this ) + offset); }
};
// ...
class QString {
QStringData *d;
public :
// ... 公共 API ...
};
offset是指向QStringData相对数据的指针。在Qt4中它是一个实际的指针。稍后我们会讲到为什么这个指针发生了变化。在字符串中保存的实际数据是UTF-16编码的,这意味着每一个字符都占用了两个字节。
文字与转换
字符串常量是指直接在源码中用引号包起来的字符串。 这有一些例子。(假设action,string和filename都是QString类型)
o->setObjectName( "MyObject" );
if (action == "rename" )
string.replace( "%FileName%" , filenam