我正在实现一个可以从各种来源(例如ftp,http等)下载文件的类。我从以下界面开始
class IInternetFileDownloader
{
public:
IInternetFileDownloader() = default;
virtual void Download(const std::string& urlToDownloadFrom) = 0;
};
然后,我实现了将从相应端点执行实际下载的类。所以我有一个HttpFileDownloader.h,如下所示:
#include "IInternetFileDownloader.h"
class HttpFileDownloader : public IInternetFileDownloader
{
public:
HttpFileDownloader() = default;
virtual void Download(const std::string& urlToDownloadFrom)
{
// download implementation
}
};
所以我有一个如下的FtpFileDownloader .h
#include "IInternetFileDownloader.h"
class FtpFileDownloader : public IInternetFileDownloader
{
public:
FtpFileDownloader() = default;
virtual void Download(const std::string& urlToDownloadFrom)
{
// download implementation
}
};
我可以如下调用适当的类
#include "IInternetFileDownloader.h"
#include "HttpFileDownloader.h"
int main()
{
std::string downloadFromUrl = "http://www.example.org/xyz.zip";
IInternetFileDownloader* download = new HttpFileDownloader();
download->Download(downloadFromUrl);
}
但是,我不想在这里实例化特定的HttpFileDownloader或FtpFileDownloader。在我看来,应该有另一个可以接受url的类,并且可以根据协议来构造适当的类。这样,客户端代码(main.cpp)无需担心适当类的实例化。我读到了有关工厂和制造商的设计模式的信息,并对在这种情况下最好使用哪种模式感到有些困惑?
答案 0 :(得分:1)
最简单的方法是在IInternetFileDownloader上具有静态函数以实例化正确的子类。
此外,我认为您不需要基类中的默认构造函数,但是您可能需要在虚拟基类中的默认析构函数。这样,我建议的工厂函数CreateDownloader
可以返回指向IInternetFileDownloader
实例的指针(或shared_ptr),该实例稍后将删除。
class IInternetFileDownloader
{
public:
virtual ~IInternetFileDownloader() = default;
virtual void Download(const std::string& urlToDownloadFrom) = 0;
// parses the url to infer the protocol and construct an instance of a derived class
static IInternetFileDownloader* CreateDownloader(const std:string& url);
};
class HttpFileDownloader : public HttpFileDownloader
{
public:
virtual void Download(const std::string& urlToDownloadFrom) override;
};
class FtpFileDownloader : public IInternetFileDownloader
{
public:
virtual void Download(const std::string& urlToDownloadFrom) override;
};
这可能就是我要采用的方法。
这的另一个变体是将“工厂”作为一个单独的类可能是有意义的。一个优点是,它可以在单元测试中提供更好的模拟下载程序实例的功能。
class IInternetFileDownloader
{
public:
virtual ~IInternetFileDownloader() = default;
virtual void Download(const std::string& urlToDownloadFrom) = 0;
};
class InternetFileDownloaderFactory
{
public:
// parses the url to infer the protocol and construct an instance of a derived class
virtual IInternetFileDownloader* CreateDownloader(const std:string& url);
};
class InternetFileDownloaderFactoryMock : public InternetFileDownloaderFactory
{
public:
IInternetFileDownloader* CreateDownloader(const std:string& url) override
{
return new MockFileDownloaderFactoryMock();
}
};
class MockFileDownloaderFactoryMock : public IInternetFileDownloader
{
public:
virtual void Download(const std::string& urlToDownloadFrom)
{
// do nothing or simulate a download
}
};