9

我正在使用 Qt 开发一个科学数据采集应用程序。由于我不是 Qt 方面的资深专家,因此我希望社区就以下问题提供一些架构建议:

该应用程序支持多个硬件采集接口,但我想在这些接口之上提供一个通用 API。每个接口都有一个样本数据类型和一个数据单位。因此,我将来自每个设备的样本向量表示为std::vectorBoost.Units 数量(即std::vector<boost::units::quantity<unit,sample_type> >)。我想使用多播风格的架构,其中每个数据源将新收到的数据广播给 1 个或多个感兴趣的各方。Qt 的 Signal/Slot 机制显然很适合这种风格。所以,我希望每个数据源都发出一个信号

typedef std::vector<boost::units::quantity<unit,sample_type> > SampleVector
signals:
    void samplesAcquired(SampleVector sampleVector);

适用于该设备的单元和 sample_type。由于元对象编译器不支持模板化子QObject类,因此似乎没有办法为定义samplesAcquired信号的所有数据源提供(模板化)基类。换句话说,以下内容不起作用

template<T,U> //sample type and units
class DataSource : public QObject {
  Q_OBJECT
  ...
  public:
    typedef std::vector<boost::units::quantity<U,T> > SampleVector
  signals:
    void samplesAcquired(SampleVector sampleVector);
};

我能想到的最佳选择是两层方法:

template<T,U> //sample type and units
class IAcquiredSamples {
    public:
        typedef std::vector<boost::units::quantity<U,T> > SampleVector
        virtual shared_ptr<SampleVector> acquiredData(TimeStamp ts, unsigned long nsamples);
};

class DataSource : public QObject {
    ...
    signals:
      void samplesAcquired(TimeStamp ts, unsigned long nsamples);
};

samplesAcquired信号现在为采集提供时间戳和样本数量,客户端必须使用IAcquiredSamplesAPI 来检索这些样本。显然,数据源必须同时继承 DataSource 和IAcquiredSamples.

这种方法的缺点似乎是失去了 API 的简单性……如果客户端可以连接 Slot 中获取的样本,那就更好了。能够使用 Qt 的排队连接也将使线程问题更容易,而不必acquiredData在每个子类的方法中管理它们。

另一种可能性是使用QVariant参数。这必然使子类有责任用Q_REGISTER_METATYPE/注册其特定的样本向量类型qRegisterMetaType。没什么大不了的。然而,基类的客户端将无法知道QVariant值类型是什么类型,除非标记结构也与信号一起传递。我认为这个解决方案至少和上面的一样复杂,因为它迫使抽象基类 API 的客户端处理类型系统的一些更复杂的方面。

那么,有没有办法实现模板化的信号参数呢?有没有比我建议的架构更好的架构?

4

4 回答 4

2

有 QVariant 类型 - 您可以在其上创建自定义子类型并将其用作信号http://doc.trolltech.com/qq/qq14-metatypes
中的参数(如果我理解您的权利并且这就是您想要的)
.html#customtypesinqvariant

于 2010-03-16T10:40:35.690 回答
1

对您的两层方法的一种简化是将QObject类作为类模板的非模板基础,即类似于

class DataSourceBase : public QObject {
    Q_OBJECT
    ...
    signals:
      void samplesAcquired(TimeStamp ts, unsigned long nsamples);
};

template<T,U> //sample type and units
class DataSource : public DataSourceBase {
    public:
        typedef std::vector<boost::units::quantity<U,T> > SampleVector
        virtual shared_ptr<SampleVector> acquiredData(TimeStamp ts, unsigned long nsamples);
};

请注意,这种方法的缺点是,由于您不能Q_OBJECT在类模板中使用宏,因此在 Qt 的元对象系统中没有关于它的信息。

于 2010-03-31T23:41:38.443 回答
1

Qt 不喜欢继承自 QObject 的类模板。如果您不需要 QObject 的运行时自省,您可能希望使用Boost.Signals代替,它没有这个问题。

不过,在 Qt 项目中引入 Boost.Signals 库可能会有些挑战。在 Boost.Signals 中,是一个命名空间,signals而 Qt#define是. 在引入 Boost.Signals 之前,您应该确保您的 Qt 项目使用定义的(在 qmake 中)进行编译。signalprotectedQT_NO_KEYWORDSCONFIG += no_keywords

于 2011-04-27T13:26:34.010 回答
1

您实际上可以将 Qt 类与模板一起使用,而无需对代码进行少量修改。

Qt 类和模板的问题是生成元对象信息的 moc 工具,它根本不了解模板,但生成的代码不需要来自 Moc。

您可以使用Verdigris创建您的 C++/QObject 类,该类在没有任何问题的情况下使用模板,绕过此类代码的 moc 步骤。

于 2017-04-05T15:29:31.280 回答