- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
为(几乎)POD 类中的属性使用带有隐式 getter 和 setter 的模板类对象是一种好习惯吗?
考虑以下模板示例:
template<typename T>
class Attribute
{
protected:
T m_val;
public:
T * getAddress() { return &m_val; }
T get() { return m_val; }
void set(T v) { m_val = v; }
};
及其用法:
class A
{
public:
Attribute<float> floatAttr;
Attribute<int> intAttr;
Attribute<long> longAttr;
};
有了这个,就可以封装数据,但实现开销更少。
这是不好的做法还是好的做法(以及为什么)?
编辑:陈述我在这方面看到的优势。无需手动实现每个 getter setter 函数,但这些函数仍然具有通常的优点:
最佳答案
在其他一些语言中,getter 和 setter 是一种防止实现细节逃逸到接口(interface)中的方法;一旦直接公开一个字段,如果不更改访问该字段的代码中的所有站点,您以后可能无法将其重新实现为属性(使用 getter 和 setter 函数)。
在 C++ 中,这并不(非常)适用。您可以将任何字段的类型更改为覆盖 operator=
的类,并隐式转换为所需的类型(对于“get”端)。 (当然有一些不适用的用途;例如,如果在客户端代码中创建指向该字段的指针或引用 - 尽管我个人会避免这样做并认为这是可疑的做法)。
此外,由于 C++ 是静态类型的,如果您需要通过适当的调用将字段和相应的访问更改为 getter/setter 对,工具(IDE 等)也更容易提供自动重构。
作为证据,这里是您的 Attribute
的改动允许您的“属性”作为直接公开字段的模板(除了 &
将返回属性的地址而不是隐藏字段):
template<typename T>
class Attribute
{
protected:
T m_val;
public:
operator T() { return m_val; }
T &operator=(const T &a) { m_val = a; return m_val; }
};
如果您真的想要,您也可以覆盖 operator&
:
T *operator&() { return &m_val; }
...但是这样做在很大程度上破坏了封装(出于同样的原因,您可能会考虑将 operator=
的返回类型更改为 T
或 void
)。
如果您最初直接公开该字段,则可以将其定义替换为上述模板的实例,并且大多数使用不会受到影响。这说明了为什么在 C++ 中并不总是需要 getter/setter 模式的原因之一。
您自己的解决方案虽然将原始字段封装在 getter/setter 函数后面,但实际上还公开了另一个字段:Attribute<T>
成员(在您的示例中为 floatAttr
等)。为了将其用作封装方式,您依赖于用户不知道(或不关心)属性字段本身的类型;也就是说,您希望没有人这样做:
A a;
Attribute<float> & float_attr = a.floatAttr;
当然,如果他们不这样做并以您预期的方式访问字段,那么以后确实可以通过更改“属性”字段的类型来更改实现:
A a;
float f = a.floatAttr.get();
...所以从这个意义上说,您实现了一些封装;真正的问题是有更好的方法来做到这一点。 :)
最后,值得一提的是,您提出的 Attribute
模板以及我上面显示的替代方案都将字段移动到与原始父类 ( Attribute<T>
) 分开的类 ( A
对于某些 T)。如果要更改实现,现在在某种程度上受此事实的限制;属性对象自然不会引用包含它的对象。例如,假设我有一个类 B
它有一个属性 level
:
class B {
public:
Attribute<int> level;
};
现在假设我稍后添加一个“最低级别”字段,min_level
:
class B {
public:
Attribute<int> level;
Attribute<int> min_level;
};
此外,假设我现在想约束 level
, 在赋值时,值为 min_level
.这不会是直截了当的!虽然我可以给level
具有自定义实现的新类型,它将无法访问 min_level
来自包含对象的值:
class LevelAttribute {
int m_val;
public:
T &operator=(const T &a) {
m_val = std::max(min_level, a); // error - min_level not defined
}
}
要让它工作,您需要将包含对象传递到 LevelAttribute
对象,这意味着存储一个额外的指针。典型的老式 setter 直接在保存字段的类中声明为函数,避免了这个问题。
关于用于自动 getter-setter 方法的 C++ 类模板 - 好的/坏的做法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43069213/
因此,我需要获取在为其赋值时调用的 setter 的名称。像这样: var b = {}; var a = { set hey(value) { b[] = value; } } 我希望 se
我是 Android 编程的新手(~ 2 个月)有必要为几十个不同的变量设置 getter 吗? 例如—— //Yes I realise that this isn't 'dozens' publi
import Control.Lens import Control.Lens.TH data Foo = Foo { _bar, _baz :: Int } makeLenses ''
我有点困惑:我可以覆盖 setter/getter 但仍然使用 super setter/getter 吗?如果是 - 怎么办? 用例: class A { void set value(num
我有一个接收消息的应用程序。消息中存在可编辑的字段。当字段更改时,应将其保存到数据库中。不幸的是,setter 仅在 setter 的范围内更改给定字段的值。知道为什么会发生这种情况吗?这是 gett
C# 中有没有一种方法可以让 setter 从“某物”继承,这样每次为特定基类及其继承者调用 setter 时,我都可以运行一些代码? 我想做的是在我的基类上有一个名为 IsValid 的 bool
可能是一个我无法解决的非常简单的问题 - 我从 C# 开始,需要使用 getter/setter 方法向数组添加值,例如: public partial class Form1 : Form {
这两个属性实现有什么区别? public override string A { get { return "s"; } set { } } public override strin
是否可以使用 abc.abstractproperty 创建一个具体的 getter 但将 setter 抽象为每个继承类的不同。我为每个子类处理不同的 val 设置。 例如。 @abstractpr
我在某处看到类似下面的内容,想知道它是什么意思。我知道他们是getter和setter,但是想知道为什么字符串Type是这样定义的。谢谢你帮助我。 public string Type { get;
Public class Example { private int number; public Example(int number){ this.number =
假设我有这样的代码: public response MyMethod(Request req) { String id = req.getFirst().geId(); } 我已经模拟了主对
允许这样做: public int Age { get; set; } 但是应用程序是否为变量创建/分配空间?我经常这样做 private int age = 0; public int Age {
我有一个条件,我构造字符串 (finalValue) 的方式是基于我在输入中获得的非空值的数量。所以我想知道是否可以用一个不同的参数为字符串 (finalValue) 重载 setter 方法并根据我
例如,在这段代码中 var o = { set a(value) {this.b = value}, get a() {return this.b} } 是否有可能获得对 o.a 的 sett
我一直在努力了解 getter 和 setter,但没有深入了解。我读过 JavaScript Getters and Setters和 Defining Getters and Setters只是没
我想在我的类中添加一个 getter 和 setter。然而,setter 应该接收一个 querySelector,但 getter 返回一个新类型 pageSections。 我的问题是 gett
使用有什么好处: private var _someProp:String; public function set someProp(value:String):void { _somePr
当从域类调用它时,我想在我的 setter 中执行一些操作,而不是从 hibernate 中调用它时。此外,我正在使用 session 工厂,因此我无法使用 @PostLoad 来触发标志! 有人对此
人员类别: public class Person { private String firstName; private String lastName; public Pe
我是一名优秀的程序员,十分优秀!