• 一个月前尝试写了一篇关于QStringLiteral,存盘时MoinMoin罢工了。吸取一点经验,还是写成短篇吧

可是,可是,QString不就是简简单单一个字符串么?能有什么可谈的。真的么...

(本文范围:Qt4)

字符串比较

如果翻看Qt的源码,或者Qt的(正规一点)第三方库,我们很少看到这种代码:

//QString myString
if (myString == "dbzhang800") {
}

取而代之的是

if (myString == QLatin1String("dbzhang800")) {
}

为什么不使用前者?

为什么?

  • QT_NO_CAST_FROM_ASCII

熟悉这个宏的,应该清楚,一旦定义之后,前者将无法通过编译。

  • 同样,下列代码也无法通过编译。
QString s = "China";
QString s2("China");

为什么?续

  • 其实,即使不考虑 QT_NO_CAST_FROM_ASCII,前面的两个比较操作仍然可能会有性能区别。(特别在国内!)

在国内,很多新手喜欢使用

QTextCodec::setCodecForCStrings()

但是大家一般很少考虑到:该语句会直接影响到字符串比较操作的性能

if (myString == "dbzhang800") {
}

??看似完全没什么联系嘛?

答案

看看Qt4是怎么做的(为了清晰起见,进行了删减)。

#ifndef QT_NO_CAST_FROM_ASCII
inline bool QString::operator==(const char *s) const
{
if (QString::codecForCStrings)
return (s1 == QString::fromAscii(s2));
return (*this == QLatin1String(s2));
}
#endif

可是,此处用QLatin1String和QString又有什么区别?

QLatin1String

Manual 中如是说

The QLatin1String class provides a thin wrapper around an US-ASCII/Latin-1 encoded string literal.

Latin1? 看来这个东西,对国内用户也没有多大用,简单说说好了。

我们知道,QString内部存储的是 utf16 字符串。于是

QString s = "dbzhang800";

这些一个简单的10个字符的字符串,至少需要在堆上申请20个字节的存储空间。对于程序中大量使用的单字节Latin1字符来说,这还真的有点浪费。

那么 QLatin1String 是怎么做的,是少申请了一些存储空间么?

确实少了不少:这种"..."字符串本身不就在常用区么?

QLatin1String("dbzhang800");

所以,只需要直接存储它的指针的就够了

class QLatin1String
{
public:
QLatin1String(const char *s) : chars(s) {}
private:
const char *chars;

只是,这还不是最优的...

QLatin1Literal

如果仔细看QString的Manual的话,你应该注意到这个东西的存在。可是这个东西又是干嘛的?和 QLatin1String 有什么瓜葛?

额,...

  • 这是一个私有类,【从Qt4.6开始出现,从Qt5.0开始消失(变成了QLatin1String的别名)】
  • 与QLatin1String的最大区别是:它在构造时直接获取字符串的长度,而QLatin1String只保存一个指针。
class QLatin1Literal
{
public:
template <int N>
QLatin1Literal(const char (&str)[N])
: m_size(N - 1), m_data(str) {} private:
const int m_size;
const char * const m_data;

对比下面的两种用法:

QString type = "long";
QLatin1String("vector<") + type + QLatin1String(">::iterator")

QString type = "long";
QLatin1Literal("vector<") + type + QLatin1Literal(">::iterator")

后者的好处就是,不用调用strlen()来获取latin1字符串的长度。

为何在Qt5中消失了呢?

因为Qt5中引入了全新的 QStringLiteral,而且估计大家会比较喜欢这个东西。毕竟可以用于中文嘛。

一个问题是,在Qt5自身的源码中,何时使用QStringLiteral,何时使用QLatin1String,有时特别让人纠结。因为

QStringLiteral("dbzhang800");
QLatin1String("dbzhang800");

前者占用的常量区比较多,程序体积会稍微大一点。(但对于中文来说,完全没有这个问题)

QString内部不是存储的utf16字符串么?而QStringLiteral就是编译期直接生成utf16字符串。(当然,需要编译器的支持)

参考

QString 乱谈(1)的更多相关文章

  1. [转载]QString 乱谈(3)-Qt5与中文

    原文地址http://blog.csdn.net/dbzhang800/article/details/7542672?reload 两个月前,简单写过QTextCodec中的setCodecForT ...

  2. QString 乱谈(3)-Qt5与中文

    原文请看:http://blog.csdn.net/dbzhang800/article/details/7542672 两个月前,简单写过QTextCodec中的setCodecForTr等终于消失 ...

  3. QString乱谈(2)

    长期以来,很多人都清楚,一旦C++源码中直接使用了中文,这样的源码想要跨平台(I18N)会非常困难. 随着: Windows下:MSVC2010成为主流 Linux下:GCC升级到4.6 C++中的中 ...

  4. 微软ASP.NET技术“乱谈”

    微软ASP.NET技术“乱谈” 2014新年了,顺手写的一点文字,主要谈谈我对当前微软ASP.NET技术的看法,比较随意,大伙儿随便看看吧. 1 当前微软Web平台技术全貌 从2002年发布.NET ...

  5. C++矢量图形库系列(1)——矢量图形库乱谈(转)

    转自:http://blog.sina.com.cn/s/blog_4265e1760100lg03.html 本系列篇章的主要内容是讲解矢量图形库的编译.开发和使用.并不对他们周边的内容做过多的描述 ...

  6. 优测优社区干货精选|老司机乱谈编辑器之神——vim

    文 / 腾讯 吴双 前言 优测小优 有话说: 腾讯优测只有应用测试大神?不不不,我们还有各种研发大牛! *** vim 是一种信仰,我自从2004年有了这个信仰,已经12个年头了.本文介绍了学习vim ...

  7. [转]vnpy乱乱谈 02架构

    vnpy乱乱谈 02架构 转自:http://101.132.65.227/?p=51 听到架构这个词先不要害怕. 其实这部分内容还是挺简单的. 一般而言, 一个交易系统我们可以简单的分成输入, (系 ...

  8. JIT-动态编译与AOT-静态编译:java/ java/ JavaScript/Dart乱谈

    C 和 C++ 之类的编译语言性能远超Java,但是生成的代码只能在有限的几种系统上执行,这就有了Java的存在基础(JVM-跨平台) 早期 Java 运行时所提供的性能级别远低于 C 和 C++ 之 ...

  9. 乱谈Qt事件循环嵌套

    本文旨在说明:QDialog::exec().QMenu::exec()等开启的局部事件循环,易用的背后,还有很多的陷阱... 引子 Qt 是事件驱动的,基本上,每一个Qt程序我们都会通过QCoreA ...

随机推荐

  1. 关于Hystrix

    RPC远程调用过程中如何防止服务雪崩效用 微服务中如何保护服务 Hystrix是一个微服务中关于服务保护框架,在分布式中能够实现对服务容错.出错之后的预备方案 背景 在今天,基于SOA的架构已经大行其 ...

  2. Luogu-3878 [TJOI2010]分金币

    这题和在我长郡考试时的一道题思路差不多...考虑折半搜索,预处理左半边选的方案所产生的数量差值\(x\)以及价值差值\(y\),把\(y\)扔到下标为\(x\)的set里面,然后在搜索右半边,每搜出一 ...

  3. 在shell中使用sendmail发送邮件

    cat > sendmymail.sh #!/bin/bash/usr/sbin/sendmail -t <<EOFFrom: Mail testing <abc@gmail. ...

  4. 描述一下你最常用的编程风格---JAVA

    描述一下你最常用的编程风格---JAVA     描述一下你最常用的编程风格---JAVA   (1)类名首字母应该大写.字段.方法以及对象(句柄)的首字母应小写.对于所有标识符,其中包含的所有单词都 ...

  5. POJ 3167 Cow Patterns (KMP+前缀和)

    题意:给你两串数字,长度分别为n和m,数字大小在[1,25].当后一串数字每个数字的排名位置与前一串数字(任一长度为m的子串)每个数字的排名位置一致时就完全匹配,最后求哪些位置是完全匹配的. 例如:1 ...

  6. HDU 6096 AC自动机

    n个字符串 m个询问 每个询问给出前后缀 并且不重合 问有多少个满足 m挺大 如果在线只能考虑logn的算法 官方题解:对n个串分别存正序倒序 分别按照字典序sort 每一个串就可以被化作一个点 那么 ...

  7. UOJ14 DZY Loves Graph

    DZY开始有 nn 个点,现在他对这 nn 个点进行了 mm 次操作,对于第 ii 个操作(从 11 开始编号)有可能的三种情况: Add a b: 表示在 aa 与 bb 之间连了一条长度为 ii ...

  8. iso不支持document事件

    ios safari游览器除了a.input.button等不支持document事件委托?<body>加上这个样式即可 <style> .clickable-div { cu ...

  9. Thinkpad E440个性化设置:如何/禁用关闭触摸板?

    #如何禁用/关闭触摸版 默认情况下,ThinkPad E440是不支持触摸板的关闭功能,如果要关闭的话,需要去官方下载相应的鼠标驱动 UltraNav. 下载地址:http://think.lenov ...

  10. 一个渣渣tomcat的学习成果.

    //////////////////////////////////////写在前面////////////////////////////////////// 时隔几个月,恢复更新了,之前由于一些私 ...