热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

c++编译器名字查找之ADL(argumentdepentmentlookup)规则

ADL,参数相关查找,也称作为Koenig查找(以AndrewKoenig的名字命名,有兴趣可以看ScottMeyer的文章

ADL,参数相关查找,也称作为Koenig查找(以Andrew Koenig的名字命名,有兴趣可以看Scott Meyer的文章The Most Important C++ People...Ever),是指在编译器对无限定域的函数调用进行名字查找时,所应用的一种查找规则。


f(x, y, z); // unqualified
N::f(x, y, z); // qualified




上面的函数调用,第一个f 就是无限定域的函数调用,第二个则限定了在名字空间N里面,也是说使用了完全限定名。
我们首先来看一个函数所在的域的分类:
1:类域(函数作为某个类的成员函数(静态或非静态))
2:名字空间域
3:全局域
而Koenig查找,它的规则就是当编译器对无限定域的函数调用进行名字查找时,除了当前名字空间域以外,也会把函数参数类型所处的名字空间加入查找的范围。
Herb提供的解释(Exceptional C++, Item 31)

Koenig Lookup(simplified): If you supply a function argument of class type (here x, of type A::X), then to look up the correct function name the compiler considers matching names in the namespace (here A) containing the argument's type.

请看下面的例程:
#include 
using namespace std;
 
namespace Koenig
{
    class KoenigArg
    {
    public:
         ostream& print(ostream& outconst
         {
                 out<          }

 
         KoenigArg(int member &#61; 5) : member_(member){}

    private:
         int member_;
    }
;
 

    inline ostream& operator<<(ostream& outconst KoenigArg& kArg)
    {
         return kArg.print(out);
    }

}

 
int main()
{
    Koenig::KoenigArg karg(10);
    cout<  
    char c;cin>>c;

    return 0;
}


我们通常都会写如上的代码&#xff0c;使用operator<<打印对象的状态&#xff0c;但是ostream& operator<<(ostream& out, const KoenigArg& kArg) 的定义是处于名字空间Koenig&#xff0c;为什么编译器在解析main函数&#xff08;全局域&#xff09;里面的operator<<调用时&#xff0c;它能够正确定位到Koenig名字空间里面的operator<<&#xff1f;这是因为根据Koenig查找规则&#xff0c;编译器需要把参数类型KoenigArg所在的名字空间Koenig也加入对operator<<调用的名字查找范围中。

 

如果没有Koenig查找规则&#xff0c;我们就无法直接写cout<

template void print(const T& value)
{
    std::cout< }

 
print(karg);
很显然&#xff0c;你的模版代码根本无法确认T是来自那个名字空间&#xff0c;直到编译器对模版实例化&#xff08;print(karg); 被调用&#xff09;。
对Koenig查找规则的一个异议是&#xff0c;由于Koenig查找规则的存在&#xff0c;处于某个名字空间的函数调用的重载决议会受到另外一个名字空间的自由函数所影响&#xff0c;仅仅是由于它使用了另外一个名字空间的类型作为参数。在这样的规则下&#xff0c;名字空间看起来不像我们一般所想象的那样是完全封闭和独立的。
我们应该怎么解释这样的异议呢&#xff1f;这样隐讳的影响或者依赖性是合理的吗&#xff1f;Herb认为&#xff0c;如果我们把另外一个名字空间的自由函数&#xff08;非类成员函数&#xff09;也看作是它所涉及的类型的接口的一部分&#xff0c;很显然&#xff0c;它应该参与这样的重载决议&#xff0c;这样的跨越名字空间的影响是合理的。从而导出了Herb在传统类定义之上的一个更详细和完整的解释&#xff08;请参考Exceptional C&#43;&#43;, Item 32&#xff09;。
传统的类定义&#xff1a;

A class describes a set of data, along with the functions that operate on that data.

一个类描述了数据的集合以及操作这些数据的函数。
Herb的类定义&#xff0c;称之为接口准则&#xff08;Interface Principle&#xff09;&#xff1a;
For a class X, all functions, including free functions, that both
"Mention" X
Are "supplied with" X

are logically part of X, because they form part of the interface of X.

 
对应类X来说&#xff0c;所有函数&#xff0c;包括自由函数&#xff0c;只要它们
         提及X&#xff08;跟X有关&#xff09;
         X一起提供
都在逻辑上被认为是X的一部分&#xff0c;因为它们是X的接口的一部分。
关于Koenig查找&#xff0c;我们该说的都说了吗&#xff1f;其实未然&#xff0c;之前所描述的只是Koenig查找一般可能发生的状况&#xff0c;当Koenig查找规则和C&#43;&#43;原来的Ordinal Lookup&#xff08;OL&#xff0c;顺序查找规则&#xff09;混合在一起的时候&#xff0c;它们之间的组合所产生的状况要比之前的例子复杂的多……








转自&#xff1a;

http://blog.csdn.net/rogeryi/article/details/1448606


推荐阅读
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Python正则表达式学习记录及常用方法
    本文记录了学习Python正则表达式的过程,介绍了re模块的常用方法re.search,并解释了rawstring的作用。正则表达式是一种方便检查字符串匹配模式的工具,通过本文的学习可以掌握Python中使用正则表达式的基本方法。 ... [详细]
  • 本文介绍了UVALive6575题目Odd and Even Zeroes的解法,使用了数位dp和找规律的方法。阶乘的定义和性质被介绍,并给出了一些例子。其中,部分阶乘的尾零个数为奇数,部分为偶数。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Html5-Canvas实现简易的抽奖转盘效果
    本文介绍了如何使用Html5和Canvas标签来实现简易的抽奖转盘效果,同时使用了jQueryRotate.js旋转插件。文章中给出了主要的html和css代码,并展示了实现的基本效果。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
author-avatar
mobiledu2502885807
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有