跳转至

Qt5 中的字符串

原文: http://zetcode.com/gui/qt5/strings/

在本章中,我们将使用字符串。 Qt5 具有用于处理字符串的QString类。 它非常强大,并且具有多种方法。

QString类提供 Unicode 字符串。 它将字符串存储为 16 位QChars。 每个QChar对应一个 Unicode 4.0 字符。 与许多其他编程语言中的字符串不同,可以修改QString

在本章的示例中,我们将不需要 Qt GUI 模块。 我们将创建命令行程序。 由于默认情况下包括 Qt GUI,因此我们可以通过在项目文件中添加QT -= gui声明来禁用它。

第一个例子

在第一个示例中,我们将使用QString类的一些基本方法。

basic.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString a = "love";

   a.append(" chess");
   a.prepend("I ");

   out << a << endl;
   out << "The a string has " << a.count() 
       << " characters" << endl;

   out << a.toUpper() << endl;    
   out << a.toLower() << endl;

   return 0;
}

在代码示例中,我们启动QString。 我们附加并添加一些其他文字。 我们打印字符串的长度。 最后,我们以大写和小写形式显示修改后的字符串。

QString a = "love";

启动QString

a.append(" chess");
a.prepend("I ");

我们将文本附加和添加到初始字符串之前。 该字符串就地修改。

out << a << endl;

终端上印有I love chess

out << "The a string has " << a.count() 
    << " characters" << endl;

count()方法返回字符串中的字符数。 length()size()方法是等效的。

out << a.toUpper() << endl;    
out << a.toLower() << endl;

这两个方法返回字符串的大写和小写副本。 他们不修改字符串,而是返回新的修改后的字符串副本。

输出:

$ ./basic 
I love chess
The a string has 12 characters
I LOVE CHESS
i love chess

初始化字符串

QString可以通过多种方式初始化。

init.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString str1 = "The night train";
   out << str1 << endl;

   QString str2("A yellow rose");
   out << str2 << endl;

   std::string s1 = "A blue sky";
   QString str3 = s1.c_str(); 
   out << str3 << endl;

   std::string s2 = "A thick fog";
   QString str4 = QString::fromLatin1(s2.data(), s2.size());
   out << str4 << endl;

   char s3[] = "A deep forest";
   QString str5(s3);
   out << str5 << endl;

   return 0;
}

我们介绍了五种初始化QString的方法。

QString str1 = "The night train";

这是在计算机语言中初始化字符串的传统方式。

QString str2("A yellow rose");

这是启动QString的对象方式。

std::string s1 = "A blue sky";
QString str3 = s1.c_str(); 

我们有一个来自 C++ 标准库的字符串对象。 我们使用其c_str()方法来生成以空字符结尾的字符序列。 该字符数组(字符串的经典 C 表示形式)可以分配给QString变量。

std::string s2 = "A thick fog";
QString str4 = QString::fromLatin1(s2.data(), s2.size());

在这些代码行中,我们将标准 C++ 字符串转换为QString。 我们利用fromLatin1()方法。 它需要一个指向字符数组的指针。 指针通过std::stringdata()方法返回。 第二个参数是std::string的大小。

char s3[] = "A deep forest";
QString str5(s3);

这是一个 C 字符串; 它是一个字符数组。 QString构造器之一可以将char数组作为参数。

输出:

$ ./init 
The night train
A yellow rose
A blue sky
A thick fog
A deep forest

访问字符串元素

QStringQChars的序列。 可以使用[]运算符或at()方法访问字符串的元素。

access.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString a = "Eagle";

   out << a[0] << endl;
   out << a[4] << endl;

   out << a.at(0) << endl;

   if (a.at(5).isNull()) {
     out << "Outside the range of the string" << endl;  
   }      

   return 0;
}

我们从特定的QString打印一些单独的字符。

out << a[0] << endl;
out << a[4] << endl;

我们打印字符串的第一个和第五个元素。

out << a.at(0) << endl;

使用at()方法,我们检索字符串的第一个字符。

if (a.at(5).isNull()) {
    out << "Outside the range of the string" << endl;  
}      

如果我们尝试访问字符串字符范围之外的字符,则at()方法返回null

输出:

$ ./access 
E
e
E
Outside the range of the string

字符串长度

有三种获取字符串长度的方法。 size()count()length()方法。 所有人都做同样的事情。 它们返回指定字符串中的字符数。

length.cpp

#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  QString s1 = "Eagle";
  QString s2 = "Eagle\n";
  QString s3 = "Eagle ";
  QString s4 = "орел";

  out << s1.length() << endl;
  out << s2.length() << endl;
  out << s3.length() << endl;
  out << s4.length() << endl;

  return 0;
}

我们得到四个字符串的大小。

QString s2 = "Eagle\n";
QString s3 = "Eagle ";

这两个字符串中的每个字符串都有一个白色字符。

QString s4 = "орел";

此字符串由俄语字母组成。

输出:

$ ./length 
5
6
6
4

从输出中我们可以看到length()方法也计算了白色字符。

字符串构建

动态字符串构建使我们可以用实际值替换特定的控制字符。 我们使用arg()方法进行插值。

building.cpp

#include <QTextStream>

int main() {

   QTextStream out(stdout);

   QString s1 = "There are %1 white roses";
   int n = 12;

   out << s1.arg(n) << endl;

   QString s2 = "The tree is %1 m high";
   double h = 5.65;

   out << s2.arg(h) << endl;

   QString s3 = "We have %1 lemons and %2 oranges";
   int ln = 12;
   int on = 4;

   out << s3.arg(ln).arg(on) << endl;

   return 0;
}

将要替换的标记以%字符开头。 下一个字符是指定自变量的数字。 一个字符串可以有多个参数。 arg()方法已重载,它可以接受整数,长数字,字符和QChar等。

QString s1 = "There are %1 white roses";
int n = 12;

%1是我们计划替换的标记。 我们定义了一个整数。

out << s1.arg(n) << endl;

arg()方法采用整数。 %1标记被n变量的值替换。

QString s2 = "The tree is %1 m high";
double h = 5.65;

out << s2.arg(h) << endl;

这三行对于一个双数执行相同的操作。 正确的arg()方法将被自动调用。

QString s3 = "We have %1 lemons and %2 oranges";
int ln = 12;
int on = 4;

out << s3.arg(ln).arg(on) << endl;

我们可以有多个控制字符。 %1引用第一个参数,%2引用第二个参数。 arg()方法在连续的链中调用。

输出:

$ ./building 
There are 12 white roses
The tree is 5.65 m high
We have 12 lemons and 4 oranges

子串

在进行文本处理时,我们需要找到普通字符串的子字符串。 我们有left()mid()right()方法可供我们使用。

substrings.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString str = "The night train";

   out << str.right(5) << endl;
   out << str.left(9) << endl;
   out << str.mid(4, 5) << endl;

   QString str2("The big apple");
   QStringRef sub(&str2, 0, 7);

   out << sub.toString() << endl;

   return 0;
}

我们将使用这三种方法查找给定字符串的一些子字符串。

out << str.right(5) << endl;

通过right()方法,我们获得str字符串的最右边五个字符。 将打印train

out << str.left(9) << endl;

使用left()方法,我们获得str字符串的最左边九个字符。 将打印night

out << str.mid(4, 5) << endl;

使用mid()方法,我们从第 4 个位置开始得到 5 个字符。 打印night

QString str2("The big apple");
QStringRef sub(&str2, 0, 7);

QStringRef类是QString的只读版本。 在这里,我们创建str2字符串一部分的QStringRef。 第二个参数是位置,第三个参数是子字符串的长度。

输出:

$ ./substrings 
train
The night
night
The big

遍历字符串

QStringQChars组成。 我们可以遍历QString以访问字符串的每个元素。

looping.cpp

#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  QString str = "There are many stars.";

  foreach (QChar qc, str) {
    out << qc << " ";  
  }

  out << endl;

  for (QChar *it=str.begin(); it!=str.end(); ++it) {
    out << *it << " " ;
  }

  out << endl;

  for (int i = 0; i < str.size(); ++i) {
    out << str.at(i) << " ";    
  }

  out << endl;

  return 0;
}

我们展示了通过QString的三种方法。 在将字母打印到终端时,我们在字母之间添加空格字符。

foreach (QChar qc, str) {
  out << qc << " ";  
}

foreach关键字是 C++ 语言的 Qt 扩展。 关键字的第一个参数是字符串元素,第二个参数是字符串。

for (QChar *it=str.begin(); it!=str.end(); ++it) {
  out << *it << " " ;
}

在此代码中,我们使用迭代器遍历字符串。

for (int i = 0; i < str.size(); ++i) {
  out << str.at(i) << " ";    
}

我们计算字符串的大小,并使用at()方法访问字符串元素。

输出:

$ ./looping
T h e r e   a r e   m a n y   s t a r s . 
T h e r e   a r e   m a n y   s t a r s . 
T h e r e   a r e   m a n y   s t a r s . 

字符串比较

QString::compare()静态方法用于比较两个字符串。 该方法返回一个整数。 如果返回的值小于零,则第一个字符串小于第二个字符串。 如果返回零,则两个字符串相等。 最后,如果返回的值大于零,则第一个字符串大于第二个字符串。 “较少”是指字符串的特定字符在字符表中位于另一个字符之前。 比较字符串的方法如下:比较两个字符串的第一个字符; 如果它们相等,则比较以下两个字符,直到找到一些不同的字符或发现所有字符都匹配为止。

comparing.cpp

#include <QTextStream>

#define STR_EQUAL 0

int main(void) {

   QTextStream out(stdout);

   QString a = "Rain";
   QString b = "rain";
   QString c = "rain\n";

   if (QString::compare(a, b) == STR_EQUAL) {
     out << "a, b are equal" << endl;
   } else {
     out << "a, b are not equal" << endl;
   }

   out << "In case insensitive comparison:" << endl;

   if (QString::compare(a, b, Qt::CaseInsensitive) == STR_EQUAL) {
     out << "a, b are equal" << endl;
   } else {
     out << "a, b are not equal" << endl;
   }     

   if (QString::compare(b, c) == STR_EQUAL) {
     out << "b, c are equal" << endl;
   } else {
     out << "b, c are not equal" << endl;
   }   

   c.chop(1);

   out << "After removing the new line character" << endl;

   if (QString::compare(b, c) == STR_EQUAL) {
     out << "b, c are equal" << endl;
   } else {
     out << "b, c are not equal" << endl;
   }            

   return 0;
}

我们将使用compare()方法进行区分大小写和不区分大小写的比较。

#define STR_EQUAL 0

为了更好的代码清晰度,我们定义STR_EQUAL常量。

QString a = "Rain";
QString b = "rain";
QString c = "rain\n";

我们将比较这三个字符串。

if (QString::compare(a, b) == STR_EQUAL) {
    out << "a, b are equal" << endl;
} else {
    out << "a, b are not equal" << endl;
}

我们比较ab字符串,它们不相等。 它们的第一个字符不同。

if (QString::compare(a, b, Qt::CaseInsensitive) == STR_EQUAL) {
    out << "a, b are equal" << endl;
} else {
    out << "a, b are not equal" << endl;
}     

如果不区分大小写,则字符串相等。 Qt::CaseInsensitive使比较大小写不敏感。

c.chop(1);

chop()方法从c字符串中删除最后一个字符。 现在bc字符串相等。

输出:

$ ./comparing 
a, b are not equal
In case insensitive comparison:
a, b are equal
b, c are not equal
After removing the new line character
b, c are equal

转换字符串

字符串通常需要转换为其他数据类型,反之亦然。 toInt()toFloat()toLong()是将字符串转换为整数,浮点数和长整数的三种QString方法。 (还有更多这样的方法。)setNum()方法将各种数字数据类型转换为字符串。 该方法已重载,并且自动调用了正确的方法。

Output

#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  QString s1 = "12";
  QString s2 = "15";
  QString s3, s4;

  out << s1.toInt() + s2.toInt() << endl;

  int n1 = 30;
  int n2 = 40;

  out << s3.setNum(n1) + s4.setNum(n2) << endl;

  return 0;
}

在示例中,我们将两个字符串转换为整数并将其相加。 然后,我们将两个整数转换为字符串并将其连接起来。

out << s1.toInt() + s2.toInt() << endl;

toInt()方法将字符串转换为整数。 我们添加两个转换为字符串的数字。

out << s3.setNum(n1) + s4.setNum(n2) << endl; 

在这种情况下,setNum()方法将整数转换为字符串。 我们连接两个字符串。

输出:

$ ./converts 
27
3040

字符

字符分为各种类别:数字,字母,空格和标点符号。 每个QString都由QChar组成。 QChar具有isDigit()isLetter()isSpace()isPunct()方法来执行作业。

letters.cpp

#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  int digits  = 0;
  int letters = 0;
  int spaces  = 0;
  int puncts  = 0;

  QString str = "7 white, 3 red roses.";

  foreach(QChar s, str) {

    if (s.isDigit()) {
      digits++;
    } else if (s.isLetter()) {
      letters++;
    } else if (s.isSpace()) {
      spaces++;
    } else if (s.isPunct()) {
      puncts++;
    }    
  }  

  out << QString("There are %1 characters").arg(str.count()) << endl;
  out << QString("There are %1 letters").arg(letters) << endl;
  out << QString("There are %1 digits").arg(digits) << endl;
  out << QString("There are %1 spaces").arg(spaces) << endl;
  out << QString("There are %1 punctuation characters").arg(puncts) << endl;

  return 0;
}

在示例中,我们定义了一个简单的句子。 我们将计算句子中的数字,字母,空格和标点符号的数量。

int digits  = 0;
int letters = 0;
int spaces  = 0;
int puncts  = 0;

我们为每个字符类别定义一个整数变量。

QString str = "7 white, 3 red roses.";

这是要检查的句子。

foreach(QChar s, str) {

  if (s.isDigit()) {
    digits++;
  } else if (s.isLetter()) {
    letters++;
  } else if (s.isSpace()) {
    spaces++;
  } else if (s.isPunct()) {
    puncts++;
  }    
}  

我们使用foreach关键字来浏览QString。 每个元素都是QChar。 我们使用QChar类的方法来确定字符的类别。

out << QString("There are %1 characters").arg(str.count()) << endl;
out << QString("There are %1 letters").arg(letters) << endl;
out << QString("There are %1 digits").arg(digits) << endl;
out << QString("There are %1 spaces").arg(spaces) << endl;
out << QString("There are %1 punctuation characters").arg(puncts) << endl;

使用字符串插值,我们将数字打印到终端。

输出:

$ ./letters 
There are 21 characters
There are 13 letters
There are 2 digits
There are 4 spaces
There are 2 punctuation characters

修改字符串

某些方法(例如toLower()方法)返回原始字符串的新修改副本。 其他方法可就地修改字符串。 我们将介绍其中的一些。

modify.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString str = "Lovely";   
   str.append(" season");

   out << str << endl;

   str.remove(10, 3);
   out << str << endl;

   str.replace(7, 3, "girl");
   out << str << endl;

   str.clear();

   if (str.isEmpty()) {
     out << "The string is empty" << endl;
   }

   return 0;
}

我们描述了四种就地修改字符串的方法。

str.append(" season");

append()方法在字符串的末尾添加一个新字符串。

str.remove(10, 3);

remove()方法从位置 10 开始从字符串中删除 3 个字符。

str.replace(7, 3, "girl");

replace()方法用指定的字符串替换从7位置开始的3字符。

str.clear();

clear()方法清除字符串。

输出:

$ ./modify 
Lovely season
Lovely sea
Lovely girl
The string is empty

在本章中,我们使用了 Qt5 中的字符串。

对齐字符串

拥有整洁的输出是常见的要求。 我们可以使用leftJustified()rightJustified()方法来对齐字符串。

right_align.cpp

#include <QTextStream>

int main(void) {

   QTextStream out(stdout);

   QString field1 = "Name: ";
   QString field2 = "Occupation: ";
   QString field3 = "Residence: ";
   QString field4 = "Marital status: ";

   int width = field4.size();

   out << field1.rightJustified(width, ' ') << "Robert\n";
   out << field2.rightJustified(width, ' ') << "programmer\n";
   out << field3.rightJustified(width, ' ') << "New York\n";
   out << field4.rightJustified(width, ' ') << "single\n";

   return 0;
}

该示例将字段字符串向右对齐。

int width = field4.size();

我们计算最宽字符串的大小。

out << field1.rightJustified(width, ' ') << "Robert\n";

rightJustified()方法返回具有width字符的字符串。 如果字符串较短,则其余部分将使用提供的字符填充。 在我们的情况下,它是一个空格字符。

输出:

$ ./right_align 
          Name: Robert
    Occupation: programmer
     Residence: New York
Marital status: single

转义字符

Qt5 具有toHtmlEscaped()方法,该方法将纯文本字符串转换为将 HTML 元字符<>&"替换为 HTML 命名实体的 HTML 字符串。

$ cat cprog.c 
#include <stdio.h>

int main(void) {

    for (int i=1; i<=10; i++) {
        printf("Bottle %d\n", i);
    }
}

此 C 包含 HTML 元字符。

html_escape.cpp

#include <QTextStream>
#include <QFile>

int main(void) {

    QTextStream out(stdout);

    QFile file("cprog.c");

    if (!file.open(QIODevice::ReadOnly)) {

        qWarning("Cannot open file for reading");
        return 1;
    }

    QTextStream in(&file);

    QString allText = in.readAll();    
    out << allText.toHtmlEscaped() << endl;

    file.close();

    return 0;
}

该示例读取一个 C 程序,并将元字符替换为其命名的实体。

输出:

$ ./html_escape 
#include <stdio.h>

int main(void) {

    for (int i=1; i<=10; i++) {
        printf(&quot;Bottle %d\n&quot;, i);
    }
}

在本章中,我们使用了 Qt5 中的字符串。



回到顶部