免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2234 | 回复: 6
打印 上一主题 下一主题

函数名称查找ADL的一个问题,和我预想的不太一样. [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-06-05 11:33 |只看该作者 |倒序浏览
10可用积分
下面这段代码是从网上摘下来的的,我自己做了一点修改。在名称空间X里面加入了一个我的函数f。我本来期待f(N:的时候调用的是我的函数,但是还是调用的N::里面的f

  1. namespace X{
  2.     template<typename T> void f(T);
  3.     void f(int){ // 我新加的函数,为什么没有匹配?
  4.         std::cout << "X::f\n";
  5.     }
  6. }
  7. namespace N{
  8.     using namespace X;
  9.     enum E { e1 };
  10.     void f(E){
  11.         std::cout << "N::f(N::E) called\n";
  12.     }
  13. }
  14. void f(int){
  15.     std::cout << "::f(int) called\n";
  16. }
  17. int main(){
  18.     ::f(N::e1);  // qualified function name: no ADL
  19.     f(N::e1);    // ordinary lookup finds ::f() and ADL finds N::f(),
  20. }                //  the latter is preferred
复制代码
奇怪的是,既然ADL按照限定名-->类型-->函数参数的名称空间的优先顺序来查找,那么既然N包含了X,为什么不能匹配X::f?

最佳答案

查看完整内容

回复 4# donet8 就是要你看这个效果啊!!你看,你用的参数是个枚举,那么最匹配的是int还是枚举呢?显然是枚举啊,那你的函数怎么可能被调用呢?那么,如果有两个枚举的话,就有二义性了,这证明你的函数的确被找到了,我们只需要关注这个即可。那么,接下来你就看,为啥用using X::f可以,用using namespace X;不可以了。如果我没记错的话,using namespace X的时候,实际上还是有个层次在里面的,会先在本名字空间找,找不 ...

论坛徽章:
5
狮子座
日期:2013-08-20 10:12:24午马
日期:2013-11-23 18:04:102015年辞旧岁徽章
日期:2015-03-03 16:54:152015亚冠之德黑兰石油
日期:2015-06-29 18:11:1115-16赛季CBA联赛之新疆
日期:2024-02-21 10:00:53
2 [报告]
发表于 2012-06-05 11:33 |只看该作者
回复 4# donet8


    就是要你看这个效果啊!!

你看,你用的参数是个枚举,那么最匹配的是int还是枚举呢?显然是枚举啊,那你的函数怎么可能被调用呢?

那么,如果有两个枚举的话,就有二义性了,这证明你的函数的确被找到了,我们只需要关注这个即可。

那么,接下来你就看,为啥用using X::f可以,用using namespace X;不可以了。

如果我没记错的话,using namespace X的时候,实际上还是有个层次在里面的,会先在本名字空间找,找不到才会去上层。

而using X::f直接把名字导入当前空间,在当前空间查找的时候就能找到它。

论坛徽章:
14
巨蟹座
日期:2013-11-19 14:09:4615-16赛季CBA联赛之青岛
日期:2016-07-05 12:36:0515-16赛季CBA联赛之广东
日期:2016-06-29 11:45:542015亚冠之全北现代
日期:2015-07-22 08:09:472015年辞旧岁徽章
日期:2015-03-03 16:54:15巨蟹座
日期:2014-12-29 08:22:29射手座
日期:2014-12-05 08:20:39狮子座
日期:2014-11-05 12:33:52寅虎
日期:2014-08-13 09:01:31巳蛇
日期:2014-06-16 16:29:52技术图书徽章
日期:2014-04-15 08:44:01天蝎座
日期:2014-03-11 13:06:45
3 [报告]
发表于 2012-06-05 12:11 |只看该作者
那么既然N包含了X
------ 你认为 “using namespace X;” 是 “包含了X”?谁忽悠你了^_^

论坛徽章:
5
狮子座
日期:2013-08-20 10:12:24午马
日期:2013-11-23 18:04:102015年辞旧岁徽章
日期:2015-03-03 16:54:152015亚冠之德黑兰石油
日期:2015-06-29 18:11:1115-16赛季CBA联赛之新疆
日期:2024-02-21 10:00:53
4 [报告]
发表于 2012-06-05 13:35 |只看该作者
试试这个:
  1. #include <iostream>

  2. namespace N {
  3.     enum E { e1 };
  4. }

  5. namespace X{
  6.     template<typename T> void f(T);
  7.     void f(N::E, int = 0){ // 我新加的函数,为什么没有匹配?
  8.         std::cout << "X::f\n";
  9.     }
  10. }
  11. namespace N{
  12.     using namespace X;
  13.     using X::f;
  14.     void f(E){
  15.         std::cout << "N::f(N::E) called\n";
  16.     }
  17. }

  18. void f(int){
  19.     std::cout << "::f(int) called\n";
  20. }
  21. int main(){
  22.     ::f(N::e1);  // qualified function name: no ADL
  23.     f(N::e1);    // ordinary lookup finds ::f() and ADL finds N::f(),
  24. }                //  the latter is preferred
复制代码

论坛徽章:
0
5 [报告]
发表于 2012-06-05 14:24 |只看该作者
starwing83 发表于 2012-06-05 13:35
试试这个:


这个不行,在f(N::E);这一行出错: error C2668: “f”: 对重载函数的调用不明确. 应该是同时找到了两个匹配,所以重载决议失败了吧。

论坛徽章:
0
6 [报告]
发表于 2012-06-05 14:25 |只看该作者
bruceteen 发表于 2012-06-05 12:11
那么既然N包含了X
------ 你认为 “using namespace X;” 是 “包含了X”?谁忽悠你了^_^


哦,那应该怎么说呢? 一个namespace包含了另一个namespace,那么X里面我增加的f会不会加入重载决议当中?

论坛徽章:
0
7 [报告]
发表于 2012-06-05 16:10 |只看该作者
starwing83 发表于 2012-06-05 15:00
回复 4# donet8


嗯,明白多了。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP