我正在使用 Qt 开发科学数据采集应用程序。由于我不是 Qt 的专家,我希望社区就以下问题提供一些架构建议:
该应用程序支持多个硬件采集接口(interface),但我想在这些接口(interface)之上提供一个通用 API。每个接口(interface)都有一个样本数据类型和一个数据单位。所以我将每个设备的样本 vector 表示为 std::vector
Boost.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);
适用于该设备的单位和样本类型。自心动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
信号现在给出了采集的时间戳和样本数量,客户端必须使用 IAcquiredSamples
检索这些样本的 API。显然,数据源必须同时继承 DataSource 和 IAcquiredSamples
。 .
这种方法的缺点似乎是失去了 API 的简单性……如果客户端可以连接 Slot 中的采集样本,那就更好了。能够使用 Qt 的排队连接也将使线程问题变得更容易,而不必在 acquiredData
中管理它们。每个子类中的方法。
另一种可能性是使用 QVariant
争论。这必然使子类有责任使用 Q_REGISTER_METATYPE
注册其特定的样本 vector 类型。/qRegisterMetaType
.没什么大不了的。然而,基类的客户将无法知道 QVariant
是什么类型。值类型是,除非标记结构也与信号一起传递。我认为此解决方案至少与上述解决方案一样令人费解,因为它迫使抽象基类 API 的客户端处理类型系统的一些较棘手的方面。
那么,有没有办法实现模板化信号参数呢?是否有比我提出的架构更好的架构?
最佳答案
有 QVariant 类型——您可以在其上创建自定义子类型
并将其用作信号中的参数(如果我理解您的权利并且这就是您想要的)
http://doc.trolltech.com/qq/qq14-metatypes.html#customtypesinqvariant
关于c++ - 具有特定于子类的模板化参数类型的 Qt SIGNAL 架构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2450781/