记录一次从MinGw转到MSVC编译器的错误经历

14 篇文章 14 订阅

MinGW和MSVC兼容度并不那么好,由于中文的问题,sa一直使用的是MinGW来进行编译,但说实话,在windows上MinGW编译出来的程序在体积和速度上和MSVC还是有点差距的,因此,sa最终版打算使用msvc编译器。
于是,前几天用Qt5.9 MSVC2015版进行了一下编译结果发现了许多问题,有语法的问题,也有非常讨厌的链接问题。

下面是一些记录:

  • MinGW比MSVC宽松很多,类似于需要返回值得函数,如果不返回值MinGW不会产生错误,只会进行警告,MSVC则会直接报错

  • 编写库时,模板类是不能导出的,我在编写时并没留意这点,模板类也进行了导出,MSVC会报错,MinGW毫无提示直接忽略,这点MSVC貌似处理更合理

然后遇到了该死的链接问题,首先遇到了莫名其妙的重定义问题error LNK2005

  • 在链接上原本能通过MinGW成功链接,但在MSVC却出现了error LNK2005错误,sa使用的绘图引擎是qwt,在signACommonUI库中定义了如下函数:

完整代码见:master/src/signACommonUI/Chart2D/SAFigureOptCommands.h 的SAFigureReplaceSeriesDataInIndexsCommand

template<typename T,typename TQwtSeries>
class SAFigureReplaceSeriesDataInIndexsCommand : public SAFigureOptCommand
{
public:
    SAFigureReplaceSeriesDataInIndexsCommand(SAChart2D* chart
                                        ,QwtSeriesStore<T> *curve
                                        ,const QString &cmdName
                                        ,const QVector<int>& inRangIndexs
                                        ,const QVector<T>& inRangNewData
                                        , QUndoCommand *parent = Q_NULLPTR);
    SAFigureReplaceSeriesDataInIndexsCommand(SAChart2D* chart
                                        ,QwtSeriesStore<T> *curve
                                        ,const QString &cmdName
                                        ,const QVector<int>& inRangIndexs
                                        ,const QVector<T>& inRangOldData
                                        ,const QVector<T>& inRangNewData
                                        , QUndoCommand *parent = Q_NULLPTR);

    virtual void redo();
    virtual void undo();
private:
    QVector<T> m_inRangOldData;
    QVector<int> m_inRangIndexs;
    QVector<T> m_inRangNewData;
    QwtSeriesStore<T> *m_curve;
};
...

///
/// \brief 序列数据QPointF的替换
///
class SA_COMMON_UI_EXPORT SAFigureReplaceXYSeriesDataInIndexsCommand
        : public SAFigureReplaceSeriesDataInIndexsCommand<QPointF,QwtPointSeriesData>
{
public:
    using SAFigureReplaceSeriesDataInIndexsCommand::SAFigureReplaceSeriesDataInIndexsCommand;
};

///
/// \brief 序列数据QwtPlotMultiBarChart的替换
///
class SA_COMMON_UI_EXPORT SAFigureReplaceMultiBarSeriesDataInIndexsCommand
        : public SAFigureReplaceSeriesDataInIndexsCommand<QwtSetSample,QwtSetSeriesData>
{
public:
  using SAFigureReplaceSeriesDataInIndexsCommand::SAFigureReplaceSeriesDataInIndexsCommand;
};

链接错误提示:

qwtd.lib(qwtd.dll) : error LNK2005: "public: void __thiscall QwtSeriesStore<class QwtSetSample>::setData(class QwtSeriesData<class QwtSetSample> *)" (?setData@?$QwtSeriesStore@VQwtSetSample@@@@QAEXPAV?$QwtSeriesData@VQwtSetSample@@@@@Z) 已经在 SAFigureOptCommands.obj 中定义
....

这里所有调用QwtSeriesStoresetData`处都有链接错误提示

看提示就是重定义了QwtSeriesStore<class QwtSetSample>::setData函数,这个函数在模板类的redo()函数里调用:

template<typename T,typename TQwtSeries>
void SAFigureReplaceSeriesDataCommand<T,TQwtSeries>::redo()
{
    ......
    m_curve->setData(new TQwtSeries(curveDatas));
}

完整代码见:SAFigureOptCommands.h 的SAFigureReplaceSeriesDataInIndexsCommand

说明编译器认为我又定义过QwtSeriesStore<class QwtSetSample>::setData函数,看一下信息,说的是qwtd.lib中的QwtSeriesStoresetData函数在我写的这个类之后编译的,也就是说,编译器先编译了我写的这个类,再去看qwtd.lib发现有重复符号。

解决方法就是引用qwtd.lib含有这些实例化模板的类的头文件,让编译器知道,这些模板的实例化已经在qwtd.lib中存在了。

终于是把signACommonUI库编译完成,接下来到主程序signA,结果又有链接错误:LNK2019LNK2001

mainwindow.obj:-1: error: LNK2019: 无法解析的外部符号 "__declspec(dllimport) public: __thiscall SAWaitCursor::SAWaitCursor(void)" (__imp_??0SAWaitCursor@@QAE@XZ),该符号在函数 "private: void __thiscall MainWindow::initPlugin(void)" (?initPlugin@MainWindow@@AAEXXZ) 中被引用
SAChartDatasViewWidget.obj:-1: error: LNK2001: 无法解析的外部符号 "__declspec(dllimport) public: __thiscall SAWaitCursor::SAWaitCursor(void)" (__imp_??0SAWaitCursor@@QAE@XZ)

一看就是编译器没找到SAWaitCursor,具体代码见:master/src/signACommonUI/SAWaitCursor.hmaster/src/signACommonUI/SAWaitCursor.cpp

在你看到之前其实只有一个头文件:


class SA_COMMON_UI_EXPORT SAWaitCursor
{
public:
    SAWaitCursor()
    {
        QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
    }
    ~SAWaitCursor()
    {
        QApplication::restoreOverrideCursor();
    }
    void setWaitCursor()
    {
        QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
    }
    void setCursor(const QCursor cur = QCursor(Qt::WaitCursor))
    {
        QApplication::setOverrideCursor(cur);
    }
    void release()
    {
        QApplication::restoreOverrideCursor();
    }
};


#ifndef SA_SET_AUTO_WAIT_CURSOR
#define SA_SET_AUTO_WAIT_CURSOR() \
    SAWaitCursor __sa__wait__cursor;\
    Q_UNUSED(__sa__wait__cursor);
#endif

略一思索,可能是没有cpp导致的问题,于是新建对应的cpp文件,并把在头文件实现的内容写到cpp中,编译出来的lib和原来的大小居然也不一样,这样顺利完成了signA的编译

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
切换Qt MinGW编译器MSVC编译器需要进行以下步骤: 1. 首先,将项目中的编码问题解决。可以使用Notepad++将文件编码转换为usc-2 little endian编码或utf-8-bom编码格式,以解决中文问题。 2. 解决编译器报错问题。如果在编译过程中遇到类似于“qalgorithms.h:847: error: C3615: constexpr 函数 ‘qCountLeadingZeroBits’ 不会生成常数表达式”这样的错误,可以参考链接\[1\]中提供的解决办法。 3. 解决乱码问题。如果在运行程序时遇到乱码,可以在代码中添加以下语句来设置字符集: ``` #if _MSC_VER >= 1600 #pragma execution_character_set("utf-8") #endif ``` 这样可以确保程序以UTF-8字符集运行。 4. 切换编译器。将Qt MinGW编译器切换为MSVC编译器,可以下载Windows下的OpenCV,并解压后找到build文件夹。在build文件夹中,选择与你使用的Visual Studio版本对应的文件夹(如VC14或VC15),这些文件夹中已经包含了编译好的库,可以直接使用而无需再进行编译\[2\]。 请注意,切换编译器可能会导致一些依赖库的不兼容问题,需要确保项目中的其他依赖项与新的编译器兼容。 参考资料: \[1\] https://codereview.qt-project.org/c/qt/qtbase/+/236948/2/src/corelib/tools/qalgorithms.h#864 \[2\] https://blog.csdn.net/cn_cc/article/details/122220232 \[3\] https://blog.csdn.net/weixin_43943902/article/details/103654051 #### 引用[.reference_title] - *1* [QT MinGw编译项目代码转msvc编译器报错等问题](https://blog.csdn.net/cn_cc/article/details/122221036)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Qt编译器MinGWMSVC的区别](https://blog.csdn.net/qq_43445867/article/details/127889570)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

尘中远

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值