我遇到了我认为是PHP应用程序的循环依赖问题。如果这不正确,请告诉我。情况如下:
两个类,LogManager和DBSession。
DBSession用于与数据库交互,LogManager用于登录文件。两者都广泛用于我的应用程序中。创建DBSession实例时,必须通过构造函数参数为其提供LogManager实例。这是因为DBSession有时会将信息记录到文件中,并将使用LogManager实例执行此操作。
现在,我想扩展LogManager,以便它也可以登录到数据库表,而不是文本文件。当然,我更喜欢重复使用现有的课程,但我很快意识到这会带来一个有趣的情况。
DBSession已经需要一个LogManager实例来构建。如果我想在LogManager中重用DBSession类,它现在需要一个DBSession实例。我怎样才能满足这两个要求?显然,我的方法肯定有问题。
你怎么建议我解决这个问题?
先谢谢你们。
答案 0 :(得分:8)
不要扩展LogManager,让它成为聚合类型。并且延迟选择要记录的位置,即:
$logManager = new LogManager();
$dbSession = new DbSession($logManager);
$logManager->add(new FileLog($filename) );
$logManager->add(new DBLog($dbSession) );
FileLog和DBLog当然共享一个通用接口。 这是Observer模式的应用程序,其中add()是“subscribe”操作,FileLog / DBLog是日志记录事件的观察者。 (这样你也可以在许多地方保存日志。)
Owen edit :调整为php语法。
答案 1 :(得分:2)
其中一个对象实际上并不需要另一个:你猜对了,它就是DBSession。修改该对象,以便在构造之后将记录器附加到它。
答案 2 :(得分:2)
为什么要求LogManager对象创建DbSession对象,如果它有时只写入文件?懒惰加载它只在你需要它时。另外,在我看来,两者应该相互独立。每个人都可以在需要时实例化另一个。
答案 3 :(得分:1)
也许您可以应用某些模式,例如Singleton Pattern,以确保您只有一个LogManager类的实例。