Qt Creator插件制作小插曲:有关QT_NO_CAST_FROM_ASCII的注意事项



  • 这两天制作了两个Qt Creator增强套装的两个插件,其实也是非常简单的,但是其实花了我超过四天的时间,为什么呢?因为我之前很长一段时间都是在Linux下开发的,一切安好,没有任何问题,但是到了Windows下,各种问题就暴露出来了。首先呢,就是——

    1、Qt Creator源码中,默认是打开QT_NO_CAST_FROM_ASCII这个宏的,看文档,这个宏就是禁用一切来自双引号或者单引号的字符串字面量传入QString的函数。但是我们裸写字符串不是非常平常的一件事情么?这里是Qt Creator断了我们一条道路,解决的方法还是有的,我这里提供两个方法,欢迎提出新的方法:

    include($$QTCREATOR_SOURCES/src/qtcreatorplugin.pri)
    

    这一句后面添加

    DEFINES -= QT_NO_CAST_FROM_ASCII
    

    如果还有必要的话,那么还需要添加

    DEFINES -= QT_NO_CAST_TO_ASCII
    

    这些都是解除设定宏的好方法

    1. 使用QStringLiteral宏来包裹字面量
      我还设计了两个宏,分别对应识别单个字符和字符串。他们分别是:
    #define CHAR( c ) QChar( ushort( c ) )
    #define TEXT QStringLiteral
    

    这些都是非常方便的宏,可以很方便地替换掉裸写的字面量。

    2、由于Windows下编译器有MinGW以及MSVC,我们要注意我们编写的插件是C++插件,是编译器相关的,比如说,MinGW编写的插件无法在MSVC编译器编译的Qt Creator上使用。遗憾的是,Qt Creator官方编译的都是MSVC的,而且有一个现象,不同的MSVC版本编写的Qt Creator依赖的Qt库,有一些函数的地址还不能通用。比如说和上面一个例子非常相似的例子,那就是我使用MSVC2015编译的Qt Creator插件,其中用到了QString::utf8()函数,但就是这个函数,无法在官方MSVC2013编译的Qt Creator中成功载入,提示“无法找到模块”。为什么呢?看看QString::utf8()函数源码吧:

    #if defined(Q_COMPILER_REF_QUALIFIERS) && !defined(QT_COMPILING_QSTRING_COMPAT_CPP)
        QByteArray toLatin1() const & Q_REQUIRED_RESULT
        { return toLatin1_helper(*this); }
        QByteArray toLatin1() && Q_REQUIRED_RESULT
        { return toLatin1_helper_inplace(*this); }
        QByteArray toUtf8() const & Q_REQUIRED_RESULT
        { return toUtf8_helper(*this); }
        QByteArray toUtf8() && Q_REQUIRED_RESULT
        { return toUtf8_helper(*this); }
        QByteArray toLocal8Bit() const & Q_REQUIRED_RESULT
        { return toLocal8Bit_helper(constData(), size()); }
        QByteArray toLocal8Bit() && Q_REQUIRED_RESULT
        { return toLocal8Bit_helper(constData(), size()); }
    #else
        QByteArray toLatin1() const Q_REQUIRED_RESULT;
        QByteArray toUtf8() const Q_REQUIRED_RESULT;
        QByteArray toLocal8Bit() const Q_REQUIRED_RESULT;
    #endif
    

    看出来什么吗?不同的预编译宏会导致不同版本的toUtf8()函数被编译,难怪Qt Creator发现QString::utf8()函数和自己Qt库中的这个函数地址不一致会报“无法找到模块”错误呢。

    那,有没有解决的方法呢?由于Json格式化,一定要这样的函数,因此我打算看看它的源码,是否能够找到替代的实现方法。没错,就是

    toUtf8_helper(*this);
    

    中,我找到了

    QUtf8::convertFromUnicode()
    

    方法
    终于找到了这个方法,又偶然发现QTextCodec::fromUnicode()函数中隐含调用了上述函数,于是我这么操作:

        QTextCodec* utf8Codec = QTextCodec::codecForName( "UTF-8" );
    
        QString plainText = ui->jsonEdit->toPlainText( );
        QByteArray converted = QJsonDocument::fromJson( utf8Codec->fromUnicode( plainText ) ).
                toJson( QJsonDocument::Indented );
        plainText = utf8Codec->toUnicode( converted );
    

    顺利实现效果!

    开发Qt Creator的过程中,每一步都是一个坑。但万变不离其宗,只要你有耐心,找到规律,找到核心问题,相信你一定和我一样,最终会找到问题的答案的。



  • @jiangcaiyang123Qt Creator插件制作小插曲:有关QT_NO_CAST_FROM_ASCII的注意事项 中说:

    plainText

    1. plainText 是一个编码 不是字符呢????
      怎么合适的方法转成 QByteArray,再 toUnicode( converted );
    2. 一定要DEFINES -= QT_NO_CAST_FROM_ASCII


  • @jiangcaiyang123Qt Creator插件制作小插曲:有关QT_NO_CAST_FROM_ASCII的注意事项 中说:
    //编码=0xcb9d,编号
    QTextCodec 能把GB2312任意编码转换成 Unicode编码吗?
    还是只能 由Unicode编码找到GB2312编码?
    也就是说只能 一个汉字 QString s="字";Qt存储为Unicode,QTextCodec::codecForName( "GB2312" ); fromUnicode,Unicode编码 装成了 GB2312编码,

    这样的话 由GB2312编码 转换成 Unicode编码 只能通过查表了吗????

    也就是说
    Unicode 提取区位码 可以找到 GB2312 编码;
    而由 GB2312 编码 不能反查 区 位码, 找到Unicode编码???

    也就是说 x+y=z 求z 只有一个解
    但是 z= x+y 求 x,y 可以有多种组合????



  • 那么 QTextCodec::toUnicode 做了些什么事情呢???

    QJsonDocument是5.0以上的 4.7 还用不了?
    Header:
    #include <QJsonDocument>
    qmake:
    QT += core
    Since:
    Qt 5.0



  • @MOMO QTextCodec是可以将一种编码转换为unicode,然后再转换为另外一种编码格式。



  • @MOMO 现在4.7不用了,尽量迁移到5吧。因为最新的是5.7了。


 

最近的回复

  • 我似乎找到原因了, 似乎是因为骁龙845是64位架构, 但我不确定,构建一个arm64-v8a的qt versions应该能证明一下, 那么问题来了.....

    阅读更多
  • @大黄老鼠 要启用深度测试:

    glEnable( GL_DEPTH_TEST );

    阅读更多
  • @青山白云 纹理滤波的选择,可以让纹理放大或者缩小的时候避免失真。还有一个叫做Anisotropic的扩展,中文名是各向异性采样,是更高级也是更好看的一个。

    阅读更多
  • 纹理滤波

    在纹理映射的过程中,如果图元的大小不等于纹理的大小,OpenGL便会对纹理进行缩放以适应图元的尺寸。我们可以通过设置纹理滤镜来决定OpenGL对某个纹理采用的放大、缩小的算法

    滤镜 描述 GL_NEAREST 取最邻近像素 GL_LINEAR 线性内部插值 GL_NEAREST_MIPMAP_NEAREST 最近多贴图等级的最邻近像素 GL_NEAREST_MIPMAP_LINEAR 在最近多贴图等级的内部线性插值 GL_LINEAR_MIPMAP_NEAREST 在最近多贴图等级的外部线性插值 GL_LINEAR_MIPMAP_LINEAR 在最近多贴图等级的外部和内部线性插值

    阅读更多

关注我们

微博
QQ群











召唤蕾姆