admin管理员组文章数量:1344334
The following code compiles (in C++20 mode) on gcc version <= 14, and on clang version <=17, not on clang 18 and gcc trunk ():
#include <iostream>
#include <concepts>
struct A {
void f() { std::cout << "A::f\n"; }
};
template <class T>
struct B : public A {
// Intention: use B::f if T is int. Use A::f otherwise.
using A::f;
void f() requires std::same_as<T, int> {
std::cout << "B::f";
}
};
int main() {
B<int>().f(); // <-- clang 18 and gcc trunk: "error: call to 'f' is ambiguous"
}
Clang's error is the following, and gcc's is very similar:
<source>:18:14: error: call to member function 'f' is ambiguous
18 | B<int>().f(); // <-- clang 18: "error: call to member function 'f' is ambiguous"
| ~~~~~~~~~^
<source>:5:10: note: candidate function
5 | void f() { std::cout << "A::f\n"; }
| ^
<source>:12:10: note: candidate function
12 | void f() requires std::same_as<T, int> {
| ^
Both I, and gcc version <= 14, and clang version <= 17, think that the call seems non-ambiguous because B::f
is more constrained than A::f
, so B::f
can be preferred. But if both compilers generate the error in more recent versions, I do realize that they are probably right and there is something I don't understand...
Which is the correct behavior, according to the standard? Why can't it resolve the call by choosing the more constrained
f
?I see that I can make this work by replacing
using A::f;
byvoid f() { A::f(); }
inB
, and this works also in my real world code. But this pattern could potentially be harder to apply if the base class contains many overloads of the function, and/or the function prototypes are complex. Is there a workaround that does not require duplicating all prototypes from the base class into the derived class?
The following code compiles (in C++20 mode) on gcc version <= 14, and on clang version <=17, not on clang 18 and gcc trunk (https://godbolt./z/hvre4x8Pc):
#include <iostream>
#include <concepts>
struct A {
void f() { std::cout << "A::f\n"; }
};
template <class T>
struct B : public A {
// Intention: use B::f if T is int. Use A::f otherwise.
using A::f;
void f() requires std::same_as<T, int> {
std::cout << "B::f";
}
};
int main() {
B<int>().f(); // <-- clang 18 and gcc trunk: "error: call to 'f' is ambiguous"
}
Clang's error is the following, and gcc's is very similar:
<source>:18:14: error: call to member function 'f' is ambiguous
18 | B<int>().f(); // <-- clang 18: "error: call to member function 'f' is ambiguous"
| ~~~~~~~~~^
<source>:5:10: note: candidate function
5 | void f() { std::cout << "A::f\n"; }
| ^
<source>:12:10: note: candidate function
12 | void f() requires std::same_as<T, int> {
| ^
Both I, and gcc version <= 14, and clang version <= 17, think that the call seems non-ambiguous because B::f
is more constrained than A::f
, so B::f
can be preferred. But if both compilers generate the error in more recent versions, I do realize that they are probably right and there is something I don't understand...
Which is the correct behavior, according to the standard? Why can't it resolve the call by choosing the more constrained
f
?I see that I can make this work by replacing
using A::f;
byvoid f() { A::f(); }
inB
, and this works also in my real world code. But this pattern could potentially be harder to apply if the base class contains many overloads of the function, and/or the function prototypes are complex. Is there a workaround that does not require duplicating all prototypes from the base class into the derived class?
- There are a similar "issue" with SFINAE case (so template with special wording), not sure how the wording evolve with constraints. – Jarod42 Commented yesterday
- 1 What language version are you asking the compilers to use? It's not clear from "gcc version <= 14, and clang version <= 17" whether those are compiler versions or language versions. – Jesper Juhl Commented yesterday
- Good point about language vs compiler version. I meant compiler versions, and in both cases compiling in C++20 mode. Clarified the question. – Sven Sandberg Commented yesterday
2 Answers
Reset to default 2This behavior change is related to CWG2789, which made this example ambiguous.
Per [over.match.funcs.general]/4, the types of the implicit object parameters of both candidates are considered to be the same in this case, so neither function is better than the other based on the ranking of implicit conversion sequences.
The only thing distinguishing them, then, is the requires-clause. In C++20 (and C++23) as published, the relevant tiebreaker ([over.match.best.general]/2.6) reads:
... a viable function F1 is defined to be a better function than another viable function F2 if [...]
- F1 and F2 are non-template functions with the same parameter-type-lists, and F1 is more constrained than F2 according to the partial ordering of constraints described in [temp.constr.order]
(The referenced subclause considers a declaration with associated constraints to be more constrained than one without, as you'd expect.)
CWG2789 changed this to:
- F1 and F2 are non-template functions and
- they have the same non-object-parameter-type-lists, and
- if they are member functions, both are direct members of the same class, and
- if both are non-static member functions, they have the same types for their object parameters, and
- F1 is more constrained than F2 according to the partial ordering of constraints described in [temp.constr.order]
Since the two f
are not direct members of the same class, this bullet no longer applies, and the call is ambiguous.
Yes, this should work: [over.match.funcs.general]/4 specifies that implicit-object member functions are not distinguished by their originating class during overload resolution (so the B&
that is *this
isn’t a better match for B
’s member function than for A
’s). Perhaps the compilers are confused by their support for explicit-object member functions added in recent versions.
本文标签:
版权声明:本文标题:c++ - Constraint does not disambiguate function in base class from function in derived class, in clang 18gcc trunk - Stack Overf 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743789248a2539256.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论