update dir

This commit is contained in:
light-city 2019-07-24 22:13:01 +08:00
parent 96ea146c60
commit 16cade9752
14 changed files with 575 additions and 0 deletions

View File

@ -13,6 +13,7 @@
- [x] [函数指针那些事](./func_pointer)
- [x] [纯虚函数和抽象类那些事](./abstract)
- [x] [vptr_vtable那些事](./vptr_vtable)
- [x] [virtual那些事](./virtual)
## 关于作者:

27
virtual/README.md Normal file
View File

@ -0,0 +1,27 @@
# virtual那些事
## 关于作者:
个人公众号:
![](../img/wechat.jpg)
## 1.虚函数与运行多态
对应的代码:[emp.cpp](./set1/emp.cpp)
**虚函数的调用取决于指向或者引用的对象的类型,而不是指针或者引用自身的类型。**
## 2.vptr与vtable
见[vptr_vtable那些事](../vptr_vtable)
## 3.虚函数中默认参数
对应的代码:[default_arg.cpp](./set2/default_arg.cpp)
**默认参数是静态绑定的,虚函数是动态绑定的。 默认参数的使用需要看指针或者引用本身的类型,而不是对象的类型**。
## 4.可以不可以
1 **静态函数可以声明为虚函数吗?**

View File

@ -0,0 +1,39 @@
/**
* @file first_example.cpp
* @brief
* 使
* @author
* @version v1
* @date 2019-07-24
*/
#include <iostream>
using namespace std;
class Base
{
public:
virtual void fun ( int x = 10 )
{
cout << "Base::fun(), x = " << x << endl;
}
};
class Derived : public Base
{
public:
virtual void fun ( int x=20 )
{
cout << "Derived::fun(), x = " << x << endl;
}
};
int main()
{
Derived d1;
Base *bp = &d1;
bp->fun(); // 10
return 0;
}

View File

@ -0,0 +1,41 @@
#include <iostream>
using namespace std;
class Base
{
public:
};
class Derived : public Base
{
public:
Derived()
{
cout << "Derived created" << endl;
}
Derived(const Derived &rhs)
{
cout << "Derived created by deep copy" << endl;
}
~Derived()
{
cout << "Derived destroyed" << endl;
}
};
int main()
{
Derived s1;
Derived s2 = s1; // Compiler invokes "copy constructor"
// Type of s1 and s2 are concrete to compiler
// How can we create Derived1 or Derived2 object
// from pointer (reference) to Base class pointing Derived object?
return 0;
}

View File

@ -0,0 +1,39 @@
/**
* @file full_virde.cpp
* @brief
*
* Constructing base
* Constructing derived
* Destructing derived
* Destructing base
* @author
* @version v1
* @date 2019-07-24
*/
#include<iostream>
using namespace std;
class base {
public:
base()
{ cout<<"Constructing base \n"; }
virtual ~base()
{ cout<<"Destructing base \n"; }
};
class derived: public base {
public:
derived()
{ cout<<"Constructing derived \n"; }
~derived()
{ cout<<"Destructing derived \n"; }
};
int main(void)
{
derived *d = new derived();
base *b = d;
delete b;
return 0;
}

View File

@ -0,0 +1,13 @@
/**
* @file static_error.cpp
* @brief const和volatile关键字修饰!
*
* static成员函数不属于任何类对象或类实例使virutal也是没有任何意义
* vptr和vtable来处理vptr是一个指针this指针来访问它this指针访vptr
* @author
* @version v1
* @date 2019-07-24
*/
virtual static void fun() { }
static void fun() const { }

206
virtual/set3/vir_con.cpp Normal file
View File

@ -0,0 +1,206 @@
/**
* @file vir_con.cpp
* @brief inline之外使
*
*
*
* vtable是在编译阶段就已经建立的vptr是在运行阶段实例化对象时才产生的 vptr vptr来访问vtablevptr还没产生
* 使
*
* Virtual Copy Constructor
*
* @author
* @version v1
* @date 2019-07-24
*/
#include <iostream>
using namespace std;
//// LIBRARY SRART
class Base
{
public:
Base() { }
virtual // Ensures to invoke actual object destructor
~Base() { }
virtual void ChangeAttributes() = 0;
// The "Virtual Constructor"
static Base *Create(int id);
// The "Virtual Copy Constructor"
virtual Base *Clone() = 0;
};
class Derived1 : public Base
{
public:
Derived1()
{
cout << "Derived1 created" << endl;
}
Derived1(const Derived1& rhs)
{
cout << "Derived1 created by deep copy" << endl;
}
~Derived1()
{
cout << "~Derived1 destroyed" << endl;
}
void ChangeAttributes()
{
cout << "Derived1 Attributes Changed" << endl;
}
Base *Clone()
{
return new Derived1(*this);
}
};
class Derived2 : public Base
{
public:
Derived2()
{
cout << "Derived2 created" << endl;
}
Derived2(const Derived2& rhs)
{
cout << "Derived2 created by deep copy" << endl;
}
~Derived2()
{
cout << "~Derived2 destroyed" << endl;
}
void ChangeAttributes()
{
cout << "Derived2 Attributes Changed" << endl;
}
Base *Clone()
{
return new Derived2(*this);
}
};
class Derived3 : public Base
{
public:
Derived3()
{
cout << "Derived3 created" << endl;
}
Derived3(const Derived3& rhs)
{
cout << "Derived3 created by deep copy" << endl;
}
~Derived3()
{
cout << "~Derived3 destroyed" << endl;
}
void ChangeAttributes()
{
cout << "Derived3 Attributes Changed" << endl;
}
Base *Clone()
{
return new Derived3(*this);
}
};
// We can also declare "Create" outside Base.
// But is more relevant to limit it's scope to Base
Base *Base::Create(int id)
{
// Just expand the if-else ladder, if new Derived class is created
// User need not be recompiled to create newly added class objects
if( id == 1 )
{
return new Derived1;
}
else if( id == 2 )
{
return new Derived2;
}
else
{
return new Derived3;
}
}
//// LIBRARY END
//// UTILITY SRART
class User
{
public:
User() : pBase(0)
{
// Creates any object of Base heirarchey at runtime
int input;
cout << "Enter ID (1, 2 or 3): ";
cin >> input;
while( (input != 1) && (input != 2) && (input != 3) )
{
cout << "Enter ID (1, 2 or 3 only): ";
cin >> input;
}
// Create objects via the "Virtual Constructor"
pBase = Base::Create(input);
}
~User()
{
if( pBase )
{
delete pBase;
pBase = 0;
}
}
void Action()
{
// Duplicate current object
Base *pNewBase = pBase->Clone();
// Change its attributes
pNewBase->ChangeAttributes();
// Dispose the created object
delete pNewBase;
}
private:
Base *pBase;
};
//// UTILITY END
//// Consumer of User (UTILITY) class
int main()
{
User *user = new User();
user->Action();
delete user;
}

41
virtual/set3/vir_de.cpp Normal file
View File

@ -0,0 +1,41 @@
/**
* @file vir_de.cpp
* @brief !
*
* Constructing base
* Constructing derived
* Destructing base
* @author
* @version v1
* @date 2019-07-24
*/
// CPP program without virtual destructor
// causing undefined behavior
#include<iostream>
using namespace std;
class base {
public:
base()
{ cout<<"Constructing base \n"; }
~base()
{ cout<<"Destructing base \n"; }
};
class derived: public base {
public:
derived()
{ cout<<"Constructing derived \n"; }
~derived()
{ cout<<"Destructing derived \n"; }
};
int main(void)
{
derived *d = new derived();
base *b = d;
delete b;
return 0;
}

View File

@ -0,0 +1,33 @@
/**
* @file virtual_function.cpp
* @brief
*
* int main()Base类的友元 ptr无法访问私有函数
* public private---->
* @author
* @version v1
* @date 2019-07-24
*/
#include<iostream>
using namespace std;
class Derived;
class Base {
private:
virtual void fun() { cout << "Base Fun"; }
friend int main();
};
class Derived: public Base {
public:
void fun() { cout << "Derived Fun"; }
};
int main()
{
Base *ptr = new Derived;
ptr->fun();
return 0;
}

View File

@ -0,0 +1,22 @@
#include<iostream>
using namespace std;
class Derived;
class Base {
public:
virtual void fun() { cout << "Base Fun"; }
// friend int main();
};
class Derived: public Base {
private:
void fun() { cout << "Derived Fun"; }
};
int main()
{
Base *ptr = new Derived;
ptr->fun();
return 0;
}

View File

@ -0,0 +1,44 @@
/**
* @file virtual_inline.cpp
* @brief
*
*
* @author
* @version v1
* @date 2019-07-24
*/
#include <iostream>
using namespace std;
class Base
{
public:
virtual void who()
{
cout << "I am Base\n";
}
};
class Derived: public Base
{
public:
void who()
{
cout << "I am Derived\n";
}
};
int main()
{
// note here virtual function who() is called through
// object of the class (it will be resolved at compile
// time) so it can be inlined.
Base b;
b.who();
// Here virtual function is called through pointer,
// so it cannot be inlined
Base *ptr = new Derived();
ptr->who();
return 0;
}

BIN
virtual/set4/rtti Executable file

Binary file not shown.

34
virtual/set4/rtti.cpp Normal file
View File

@ -0,0 +1,34 @@
/**
* @file rtti.cpp
* @brief 使Java的instanceofC#asis运算符类似C++dynamic_cast函数用于动态转型C风格的强制类型转换和C++ reinterpret_castdynamic_cast提供了类型安全检查(Capability Query)dynamic_cast
* @author
* @version v1
* @date 2019-07-24
*/
// CPP program to illustrate
// // Run Time Type Identification
#include<iostream>
#include<typeinfo>
using namespace std;
class B { virtual void fun() {} };
class D: public B { };
int main()
{
B *b = new D; // 向上转型
B &obj = *b;
D *d = dynamic_cast<D*>(b); // 向下转型
if(d != NULL)
cout << "works"<<endl;
else
cout << "cannot cast B* to D*";
try {
D& dobj = dynamic_cast<D&>(obj);
cout << "works"<<endl;
} catch (bad_cast bc) { // ERROR
cout<<bc.what()<<endl;
}
return 0;
}

View File

@ -0,0 +1,35 @@
// 在使用时需要注意被转换对象obj的类型T1必须是多态类型即T1必须公有继承自其它类或者T1拥有虚函数继承或自定义。若T1为非多态类型使用dynamic_cast会报编译错误。
// A为非多态类型
class A{
};
//B为多态类型
class B{
public: virtual ~B(){}
};
//D为多态类型
class D: public A{
};
//E为非多态类型
class E : private A{
};
//F为多态类型
class F : private B{
}