1

我想用 C++ 编写一个简单灵活的 ftp 服务器,它可以用一个类参数化来处理服务器初始化时提供的用户(检查登录名和密码,传递文件等)。

所以我想出了这个整洁(所以我想)的想法:

class FtpDelegate
{
public:
    FtpDelegate() {}
    virtual ~FtpDelegate() {}
    virtual bool login(QString username, QString password) = 0;
    // ...
};

class DummyDelegate : public FtpDelegate
{
public:
    virtual bool login(QString username, QString password)
    {
        return true;
    }
};

template<class Delegate>
class FtpServer : public QObject, Derived_from<Delegate, FtpDelegate>
{
    Q_OBJECT
public:
    explicit FtpServer(const QHostAddress &address = QHostAddress::Any,
                       quint16 port = 21,
                       QObject *parent = 0);

public slots:
    void newConnection();

private:
    QTcpServer *server;
    QHostAddress address;
};

template <class Delegate>
void FtpServer<Delegate>::newConnection()
{
    FtpDelegate *delegate = new Delegate();
    new FtpConnection (delegate, server->nextPendingConnection(), address, this);
}

class FtpConnection : public QObject
{
    Q_OBJECT
public:
    explicit FtpConnection(FtpDelegate *delegate,
                           QTcpSocket *socket,
                           const QHostAddress &address,
                           QObject *parent = 0);

public slots:
    void newDataConnection();

private:
    QTcpSocket *socket;
    QTcpServer *dataServer; // needed to transfer data to user
    QTcpSocket *dataSocket;
};


// server initialization
FtpServer<DummyDelegate> ftpServer();

然后(你可能已经看到了)砰!

Error: Template classes not supported by Q_OBJECT

很可能还有其他错误或误解,因为我才刚刚开始学习 C++ 模板机制(以及 Qt)。

我的问题是:在不使用丑陋的技巧(如传递函数指针或需要为每个具体的 FtpDelegate 的派生类创建工厂实现)的情况下,使其工作的最佳方法是什么。也许有一些我看不到的聪明的设计模式。最终,如果它是最好的选择,我可以重写网络机制来提升。

4

1 回答 1

2

无法创建模板 Q_OBJECT 类(请参阅内容和答案)。

您应该使用运行时继承,而不是使用静态继承,并注入从 FtpDelegate 类继承的对象。


看起来 FtpServer 实际上是一个创建连接的工厂。从您的问题来看,我不明白为什么它必须是 Q_OBJECT 类。因此,您可能需要重新考虑您的设计,并简化该类。


什么是让它工作的最佳方法,而不使用像传递函数指针或需要为每个具体的 FtpDelegate 派生类创建工厂实现这样的丑陋黑客。

最好的方法可能是拥有一个创建类型实例的工厂类FtpDelegate。但是您在发布的代码中有很多问题,如果不知道所有血淋淋的细节,就不可能说出更多。

于 2013-11-05T16:25:00.077 回答