gpt4 book ai didi

C++ 重载 getter 两次,一次返回指针,一次返回 const 引用,失败

转载 作者:搜寻专家 更新时间:2023-10-31 00:11:28 24 4
gpt4 key购买 nike

正如标题所说,我试图重载一个 getter 以返回一个指向成员变量的指针(mutator 方法)和一个常量引用(inspector 方法)。

请注意,上面链接中的示例使用了一个引用和一个常量引用,我不想要这个。

#include <vector>

class A {
public:
A() : v() {}
const std::vector<int>& get_v() const {return v;}
std::vector<int>* get_v() {return &v;}
private:
std::vector<int> v;
};

int main() {
A a;
a.get_v()->size(); // ok
a.get_v().size(); // error: request for member ‘size’ in ‘a.A::get_v()’,
// which is of pointer type ‘std::vector<int>*’
// (maybe you meant to use ‘->’ ?)
}

在我尝试使用它并抛出上面给出的错误描述之前,它似乎工作正常。这种不当行为是否有原因(以及解决方法)?

最佳答案

It seems to work ok until I try to use it throwing the error description given above.

a.get_v().size();

那段代码显然是错误的,因为std::vector<int>* get_v()返回一个指针。关于尝试访问指针成员的错误消息非常清楚,这是您不能做的事情。要修复它,您需要使用箭头运算符取消引用指针,就像您在上面的行中所做的那样。

How can you say which one is used?

一个重载是非常量的,只能在非常量对象上调用。另一个是 const,可以在 const 和非常量对象上调用,但永远不会在非常量对象上调用它,因为非常量重载是首选。它是首选,因为它不需要将非常量对象参数转换为 const。对 const 成员函数的调用将需要这样的转换。不需要参数转换的重载优于那些需要转换的重载。因此,对于非常量对象参数,总是调用非常量版本。

the following also fails

std::vector<int>& v = a.get_v();

a仍然是非常量,因此选择了返回指针的重载。您不能将非指针引用绑定(bind)到指针。


如果您想调用返回非 const 对象上的 const 引用的函数,那么您必须给它另一个名称,这样它就没有优先于它的重载。


Do you have a reference for this?

我会引用 cppreference.com因为这是非常基本的东西——尽管规则有点复杂。如果您怀疑它的正确性,请查阅该标准。

首先,关于成员函数重载的一些知识:

If any candidate function is a member function (static or non-static), but not a constructor, it is treated as if it has an extra parameter (implicit object parameter) which represents the object for which they are called and appears before the first of the actual parameters.

我认为很明显这两个重载都是候选者,因为它们具有相同的名称。它们也是可行的,因为它们具有正确数量的参数并且它们是可转换的。那么,哪种重载是首选?

For each pair of viable function F1 and F2, the implicit conversion sequences from the i-th parameter to i-th argument are ranked to determine which one is better

F1 is determined to be a better function than F2 if implicit conversions for all arguments of F1 are not worse than the implicit conversions for all arguments of F2, and

1) there is at least one argument of F1 whose implicit conversion is better than the corresponding implicit conversion for that argument of F2

...

好的,所以更好的转换序列是首选...哪个更好?让我们首先弄清楚转换序列是什么。唯一的参数是隐式对象参数。传递参数的类型是非常量 A .

作为非常量的重载具有非常量类型的隐式对象参数 A (它在这里并不真正相关,但它在实践中作为指针传递)。非常量 A不需要转换为非常量 A因为它是完全匹配的。这将被视为身份转换。

作为 const 的重载具有类型为 const A 的隐式对象参数. A可以隐式转换为 const A .

两次转化的排名相同。在这种情况下,需要遵循一长串规则。这些都不适用,直到最后一部分:

f) Or, if not that, S1 and S2 only differ in qualification conversion, and the cv-qualification of the result of S1 is a subset of the cv-qualification of the result of S2

恒等式转换是常量转换的一个子集。因此它更好。我认为身份也可能更好,因为 cv-conversion 在计算身份转换时需要两次转换......不过我找不到关于它的规则。

关于C++ 重载 getter 两次,一次返回指针,一次返回 const 引用,失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33912091/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com