- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试将现有功能封装在大量的类中,以便可以对其进行统一修改(例如互斥、优化、记录等)出于某种原因,我已经想到了(多个)私有(private)继承是可行的方法,但我找不到导致我得出该结论的原因。
问题是:我想做的事情的名称是什么,我在哪里可以看到它正确完成?
我认为这不是:
我也对其他选择持开放态度,但这里的大奖是找到一篇比我聪明得多的人写的关于它的文章(如 la Alexandrescu、Meyers、Sutter 等)
示例代码:
// Original code, this stuff is all over
class SprinkledFunctionality
{
void doSomething()
{
...
int id = 42;
Db* pDb = Db::getDbInstance(); // This should be a reference or have a ptr check IRL
Thing* pThing = pDb->getAThing(id);
...
}
}
// The desired functionality has been extracted into a method, so that's good
class ExtractedFunctionality
{
void doSomething()
{
...
int id = 42;
Thing* pThing = getAThing(id);
...
}
protected:
Thing* getAThing(int id)
{
Db* pDb = Db::getDbInstance();
return pDb->getAThing(id);
}
}
// What I'm trying to do, or want to emulate
class InheritedFunctionality : private DbAccessor
{
void doSomething()
{
...
int id = 42;
Thing* pThing = getAThing(id);
...
}
}
// Now modifying this affects everyone who accesses the DB, which is even better
class DbAccessor
{
public:
Thing* getAThing(int id)
{
// Mutexing the DB access here would save a lot of effort and can't be forgotten
std::cout << "Getting thing #" << id << std::endl; // Logging is easier
Db* pDb = Db::getDbInstance(); // This can now be a ptr check in one place instead of 100+
return = pDb->getAThing(id);
}
}
最佳答案
您可能会忽略的一项有用技术是非虚拟接口(interface) (NVI),这是 Sutter 在其关于虚拟性的著作中创造的。它需要稍微改变您看待它的方式,但旨在解决这些确切的问题。它还从内部解决了这些问题,也就是说,装饰器是关于从外部非侵入性地扩展功能的。
class Foo
{
public:
void something()
{
// can add all the central code you want here for logging,
// mutex locking/unlocking, instrumentation, etc.
...
impl_something();
...
}
private:
virtual void impl_something() = 0;
};
我们的想法是支持您的公共(public)接口(interface)使用非虚函数,但让它们调用在其他地方被覆盖的虚函数(具有私有(private)或 protected 访问权限)。这为您提供了通常通过继承获得的可扩展性,同时保留了中央控制(否则通常会丢失)。
现在 Bar
可以从 Foo
派生并覆盖 impl_something
以提供特定的行为。然而,您保留了 Foo
中的中央控制权,可以添加您喜欢的任何内容并影响 Foo
层次结构中的所有子类。
最初 Foo::something
甚至可能只调用 Foo::impl_something
做任何事情,但这里的值(value)是为将来添加提供的喘息空间任何你想要的中央代码——如果你低头查看一个代码库,它有大量直接依赖于虚拟函数的代码库,这可能会非常尴尬。通过依赖一个公共(public)非虚拟函数,该函数依赖于一个被覆盖的非公共(public)虚拟函数,我们获得了一个中间站点,我们可以在其中添加我们喜欢的所有中央代码。
请注意,这也可能有点矫枉过正。在 SE 中一切都可能过大,因为一个足够简单的程序实际上可能是最容易维护的,如果它只使用全局变量和一个大的 main
函数。所有这些技术都需要权衡取舍,但随着足够的规模、复杂性和不断变化的需求*,优点开始超过缺点。
* 我注意到您在另一个问题中写道,适合这项工作的正确工具应该具有零缺点,但一切往往都有缺点,一切都是权衡取舍。最终决定它是否是一个好的设计决策的是利弊是否大于利弊,并且远见卓识而不是事后诸葛亮来实现所有这些远非易事。
至于你的例子:
// What I'm trying to do, or want to emulate
class InheritedFunctionality : private DbAccessor
{
void doSomething()
{
...
int id = 42;
Thing* pThing = getAThing(id);
...
}
}
...这里的耦合比本示例所需的耦合要紧密得多。它可能比你所展示的更多,这使得私有(private)继承成为必要,但否则组合通常会在没有太多额外努力的情况下大大放松耦合,就像这样:
class ComposedFunctionality
{
...
void doSomething()
{
...
int id = 42;
Thing* pThing = dbAccessor.getAThing(id);
...
}
...
private:
DbAccessor dbAccessor;
};
关于c++ - 通过多重私有(private)继承扩展类——这是一回事吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33988733/
据我所知,在 C# 中,所有字段默认都是私有(private)的,如果没有另外标记的话。 class Foo { private string bar; } class Foo { strin
这个问题在这里已经有了答案: Why is it allowed to access Java private fields via reflection? (7 个答案) 关闭 6 年前。 使用反
在 C#(和许多其他语言)中,访问相同类型的其他实例的私有(private)字段是完全合法的。例如: public class Foo { private bool aBool; pu
使用私有(private)方法通过将一些决策点重构为单独的方法来降低 CC 会降低实际方法的 CC 并易于阅读,但不会减少在测试中获得完整分支覆盖的工作量。 这合理吗?你有什么现场经验? 最佳答案 好
在下面的例子中,模块outer有一个私有(private)类型Private和一个私有(private)内部模块inner。 inner 能够访问Private(因为子模块可以访问其父级的私有(pri
class Person def one @var = 99 self.two end private def two p @var end end p=P
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我在 Azure 中创建了 VNET。我放入了一个子集 Azure Private Link,它在 VNET 之外和另一台虚拟机中调用 Azure Function。 当我尝试通过专用 IP 调用专用
我正在尝试获得良好的 Ruby 编码风格。为防止意外调用具有相同名称的局部变量,我总是在适当的地方使用 self.。但是现在我偶然发现了这个: class MyClass "method" a
今天遇到一个案例类构造函数的奇怪问题。我想将构造函数设为私有(private),看来这不是问题。所以我已经在我的一个项目中尝试过它并且它有效。但在另一个项目中,我可以调用私有(private)构造函数
我想坚持使用记录,并且不想返回对象。所以我想知道是否可以将记录的字段设置为私有(private)?或者创建记录的私有(private)成员。其他具体类型(例如可区分联合)怎么样? 或者,这个要求是否违
我正在通过 Flickr API 进行经过身份验证的调用来访问照片。但我只得到我的公开照片,而没有任何私有(private)照片。 下面给出的是我正在使用的代码, Flickr f; Request
这两个类的行为不同;原因似乎与使用 private[this] 声明而不是 private 有关。有人可以解释一下为什么吗? 私有(private): class Person( private
在 private 中的 1000 秒 private 之后,我想到可能不需要它 public class Outer { private static class Inner { // yo
我有以下代码: class C { private enum E { // ... } } private extension C { func f(e: E)
OOP 语言中是否有object-private 的概念??我的意思是比经典的私有(private)访问限制更多? Private (or class-private) restricts the a
swift 3.0 我知道fileprivate访问级别修饰符将函数/属性的使用限制在声明它的源文件和 private - 仅限于声明的词法范围。但似乎这条规则不适用于扩展。例如。此代码有效: cla
即将推出的 Delphi 版本中启用该功能的功能怎么样? 也许它可能是一个编译器开关,促进所有 ** private **s to ** strict private **小号。 ... 或者它可能是
我可以通过将函数放入类的私有(private)扩展中来创建私有(private)函数,而不是通过不断调用 private func functionName(){} 来创建新的私有(private)函
部署专用端点并需要专用 IP 地址作为输出,但似乎无法正确获取值查询。下面的结果是“模板输出'主机名'无效:语言表达式属性|'privateIPAddress'具有无效的数组索引..(代码:Deplo
我是一名优秀的程序员,十分优秀!