C++11:using 的各类做用

C++11中using关键字的主要做用是:为一个模板库定义一个别名。html

文章连接:派生类中使用using别名改变基类成员的访问权限 
ios

1、《Effective Modern C++》里有比较完整的解释ide

各个做用函数

/*定义别名*/
 template<class T>
 using Tlist = std::list<T>;
 
 using Tlist = std::list<char>;
 Tlist listChar;
 
 //typedef void (*df)()
  using  df = void(*)();
 /*使用外部构造*/
 using A::A;

 /*引用外部类型*/
using typename A;

2、Using 关键字的做用:重载父类函数
spa

1.在当前文件中引入命名空间.net

     这是咱们最熟悉的用法,例如:using namespace std;htm

2.在子类中使用 using 声明引入基类成员名称(参见C++ primer)blog

private或者protected继承时,基类成员的访问级别在派生类中更受限:继承

class Base {
public:
std::size_t size() const { return n; }
protected:
std::size_t n;
};
class Derived : private Base { . . . };

在这一继承层次中,成员函数 size 在 Base 中为 public,但在 Derived 中为 private。为了使 size 在 Derived 中成为 public,能够在 Derived 的 public
部分增长一个 using 声明。以下这样改变 Derived 的定义,可使 size 成员可以被用户访问,并使 n 可以被 Derived的派生类访问:ci

class Derived : private Base {
public:
using Base::size;
protected:
using Base::n;
// ...
};

另外,当子类中的成员函数和基类同名时,子类中重定义的成员函数将隐藏基类中的版本,即便函数原型不一样也是如此(隐藏条件见下面)。

若是基类中成员函数有多个重载版本,派生类能够重定义所继承的 0 个或多个版本,可是经过派生类型只能访问派生类中重定义的那些版本,因此若是派生类想经过自身类型使用全部的重载版本,则派生类必须要么重定义全部重载版本要么一个也不重定义。有时类须要仅仅重定义一个重载集中某些版本的行为,而且想要继承其余版本的含义,在这种状况下,为了重定义须要特化的某个版本而不得不重定义每个基类版本,可能会使人厌烦。能够在派生类中为重载成员名称提供 using 声明(为基类成员函数名称而做的 using 声明将该函数的全部重载实例加到派生类的做用域),使派生类不用重定义所继承的每个基类版本。一个 using 声明只能指定一个名字,不能指定形参表,使用using声明将名字加入做用域以后,派生类只须要重定义本类型确实必须定义的那些函数,对其余版本可使用继承的定义。

“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则以下:

1、若是派生类的函数与基类的函数同名,可是参数不一样。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆)

2、若是派生类的函数与基类的函数同名,而且参数也相同,可是基类函数没有virtual关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)

#include "StdAfx.h"
#include <iostream>
using namespace std;
class Base
{
public:    
   void menfcn()
  {
     cout<<"Base function"<<endl; 
  }
    void menfcn(int n)
    {
     cout<< cout<<"Base function with int"<<endl; 
    }
};

class Derived : Base
{
public:    
using Base::menfcn;//using声明只能指定一个名字,不能带形参表    
int menfcn(int)
{ cout<< cout<<"Derived function with int"<<endl; }
};
int main()
{    Base b; 
     Derived d;   
  b.menfcn();   
  d.menfcn();//若是去掉Derived类中的using声明,会出现错误:error C2660: 'Derived::menfcn' : function does not take 0 arguments    std::cin.ignore(std::cin.gcount()+1);//清空缓冲区    std::cin.get();//暂停程序执行  
}

3、须要注意的状况

子类中using引入基类函数时须要注意的状况
class base{
public:
 void test(){
  cout << "base::test()" << endl;
 }
 void test(int){
  cout << "base::test(int)" << endl;
 }
};
class derived : public base{
public:
 void test(){
  cout << "derived::test()" << endl;
 }
};

此时derived::test()会隐藏(hide)父类中的两个test重载函数(base::test()和base::test(int)),所以咱们为子类中加上一个using声明:
class derived : public base{
public:
 void test(){
  cout << "derived::test()" << endl;
 }
 using base::test;//此声明放在test前面和后面效果都同样
};

如今会不会出现下面所述的状况呢?
---------------------------------------------------------------------------------------------------------------
既然using base::test将父类中的两个test函数都引入子类,则子类中就至关于有了一个void test()函数,因此咱们在子类中从新定义的void test()函数将会和从父类中引入的void test()函数发生冲突,进而出现“重定义”错误。
---------------------------------------------------------------------------------------------------------------
答案是:不会!
此时,子类中从新定义的void test()函数将“顶替”从父类中引入的void test()函数。
(PS:从父类中引入的另一个void test(int)函数则没有发生变化(仍然是父类中的函数实现)。)
相似的另一种状况以下,此时加入了virtual:
class base{
public:
 virtual void test(){
  cout << "base::test()" << endl;
 }
 virtual void test(double){
  cout << "base::test(double)" << endl;
 }
 void test(int){
  cout << "base::test(int)" << endl;
 }
};
class derived : public base{
public:
 void test(){
  cout << "derived::test()" << endl;
 }
};

此时derived::test()虽然重写(override)了base::test(),可是同时也隐藏(hide)父类中的两个test重载函数(一个virtual函数base::test(double)和一个nonvirtual函数base::test(int))。如今,咱们为子类中加上一个using声明:
class derived : public base{
public:
 void test(){
  cout << "derived::test()" << endl;
 }
 using base::test;//此声明放在test前面和后面效果都同样
};

与上面的相似,此时derived::test()“仍然重写”了父类的base::test(),而且与父类中的base::test(double)和base::test(int)[在子类的域]中造成重载集合。

最后,留一个思考题目,以下:
class base{
public:
 virtual void test(){
  cout << "base::test()" << endl;
 }
 virtual void test(double){
  cout << "base::test(double)" << endl;
 }
 void test(int){
  cout << "base::test(int)" << endl;
 }
};
class derived : public base{
public:
 void test(){
  cout << "derived::test()" << endl;
 }
 //using base::test;
};
class A : public derived{
public:
 void test(double){
  cout << "A::test(double)" << endl;
 }
};
int main(int argc, char **argv){
 base *pb = new A;
 pb->test(2.4);
 return 0;
}

问题:derived中的using base::test加上与否,对程序的结果有什么影响? 答:没有影响。(关键点:名字解析是编译时期的事情,而virtual函数动态绑定是运行时期的事情。) (PS:可是将main函数改为“derived *pd = new A; pd->test(2.4);”,则有区别了:若是将using base::test去掉,则编译失败。)
相关文章
相关标签/搜索