qt 帮助手册中对函数 escape_expand(string[,string[,string...]])的解释有些让人糊涂。
escape_expand(string[,string[,string...]])是qt的内建replace函数。用于对字符串中的"\n" "\r" "\t" 转义成'\n' '\r' '\t'。
qmake language对string中的字符转义是在词法分析阶段,转义字符包括: ,而对\n,\t,\r不会做转义。如果要在string中嵌入'\n' '\t' '\r',需要用escape_expand(string)函数进行操作。
message(aaa\nbbb)
message(\[ \] \{ \} \( \) \$ \\ \' \" aa\nbb\rcc\tdd)
message(aaa$$escape_expand(\n)bbb)
message(aaa$$escape_expand(\t)bbb)
message(aaa$$escape_expand(\r)bbb)
#输出:
#Project MESSAGE: aaa\nbbb
#Project MESSAGE: [ ] { } ( ) $ \ ' " aa\nbb\rcc\tdd
#Project MESSAGE: aaa
#bbb
#Project MESSAGE: aaa bbb
#Project MESSAGE: aaa
#bbb
下面是qmake中对内建函数 escape_expand(string[,string[,string...]])的实现逻辑。注意这里是语义分析阶段了,语法阶段已经对string按qmake language规则做过了转义字符处理了。也就是说,如果你在pro中string1写入的是\\n,那么在这里,你拿到的string2是\n; 如果在pro中写入string1的是\\\n,在这里你拿到的string2是\\n。escape_expand中的逻辑是对string2做处理。
//D:\Qt\Qt5.12.0\5.12.0\Src\qtbase\qmake\library\qmakebuiltins.cpp
QMakeEvaluator::VisitReturn QMakeEvaluator::evaluateBuiltinExpand(
const QMakeBuiltin &adef, const ProKey &func, const ProStringList &args, ProStringList &ret)
{
......
case E_ESCAPE_EXPAND:
for (int i = 0; i < args.size(); ++i) {
QString str = args.at(i).toQString();
QChar *i_data = str.data();
int i_len = str.length();
for (int x = 0; x < i_len; ++x) {
if (*(i_data+x) == QLatin1Char('\\') && x < i_len-1) {
if (*(i_data+x+1) == QLatin1Char('\\')) { //处理\\r \\\r \\\\\r \\\\\r......的情况,此处直接略过前面多余的\,不做任何处理。
++x;
} else {
struct {
char in, out;
} mapped_quotes[] = {
{ 'n', '\n' },
{ 't', '\t' },
{ 'r', '\r' },
{ 0, 0 }
};
for (int i = 0; mapped_quotes[i].in; ++i) {
if (*(i_data+x+1) == QLatin1Char(mapped_quotes[i].in)) {
*(i_data+x) = QLatin1Char(mapped_quotes[i].out);
if (x < i_len-2)
memmove(i_data+x+1, i_data+x+2, (i_len-x-2)*sizeof(QChar));
--i_len;
break;
}
}
}
}
}
ret.append(ProString(QString(i_data, i_len)).setSource(args.at(i)));
}
break;
......
}