そのライブラリ内のすべてのオブジェクトにわたって静的ライブラリ内でその後使用されるクラスへのポインタを受け入れる適切な方法は何ですか?
古典的なケースはロガーで、ここでは例として使用します。
シングルトンは、ロガーを扱うときによく使用されます。静的ライブラリ内で完全に定義されている場合、これは問題ありません。ただし、ロガーがライブラリの外部にある場合に対処する適切な方法は何ですか? インスタンスを渡しながら、ライブラリ内のすべてのクラスで使用するにはどうすればよいでしょうか?
それなら、あなたが言うDIを使いますか?私の場合、一部のオブジェクトがライブラリの外部で作成されているため、DI を使用できません。
では、正しいパターンとは?
これは、私が達成しようとしていることの概念を示すコードです。
#include <string>
#include <vector>
#include <memory>
#include <iostream>
//=====================================================
// Logger Libraray
//=====================================================
class Logger {
public:
void Write( std::string message ) {
std::cout << message.c_str() << std::endl;
};
};
//=====================================================
// Internal Libraray
//=====================================================
class Base {
public:
Base() {};
void SetLogger( std::shared_ptr<Logger> logger ) {
logger_ = logger;
};
void Log( std::string message ) {
if( logger_ != nullptr )
logger_->Write( message );
};
private:
static std::shared_ptr<Logger> logger_;
};
class DerivedA : public Base {
public:
DerivedA() {
Log("In derived A");
}
};
class DerivedB : public Base {
public:
DerivedB() {
Log("In derived B");
}
};
//=====================================================
// Library External Interface
//=====================================================
class Item : public Base {
public:
std::string Name;
Item() {
Log( "Item created externally" );
};
};
class Library : public Base {
public:
Library() {};
std::vector<Item> ItemList;
void AssignLog( std::shared_ptr<Logger> logger ) {
SetLogger( logger );
};
void process( ) {
DerivedA a;
DerivedB b;
}
};
//=====================================================
// External Program
//=====================================================
int main() {
Logger log;
log.Write("test");
Library lib;
lib.AssignLog( std::make_shared<Logger>(log));
lib.process();
Item item;
item.Name = "Sample";
lib.ItemList.push_back(item);
}