延续上一小节内容:下面继续讲解虚函数和多态

  1. 虚函数和多态
  1. 基类指针可以指向任何派生类的对象,但是不能调用派生类对象的成员。
  2. 但是,基类可以调用覆盖了虚函数的函数。(现在调用将来,这有问题,说明现在影响了将来。)
  3. 基类可以被继承,如果虚函数没有被实现,可以继续被下一个类继承。
  4. 当派生类没有能够覆盖虚函数的时候,若派生类的对象访问这个函数,那么此时将使用基类定义的函数。(现在调用过去,没有问题,已经定义过)

举例:

// BlankTest.cpp : 定义控制台应用程序的入口点。

//知识点:虚函数

//虚函数的强大之处在于能够预测未来。

//记住:基类指针可以指向任何派生类型的对象,但不能访问派生类型对象中定义的成员,但是可以调用虚函数(纯虚也可以)的实现函数。

#include "stdafx.h"

#include <iostream>

using namespace std;

 

class A

{

public:

    virtual void f()

    {

        cout << "我是虚函数 " << endl;

    }

protected:

private:

};

 

class B : public A

{

public:

    void f()

    {

        cout << "我派生自A " << endl;

    }

};

 

class C : public B

{

public:

    void f()

    {

        cout << "我派生自B"<< endl;

    }

};

 

class D :public A// 没有覆盖A中的虚函数,此时D的对象调用f这个函数的时候,将调用基类当中的函数。

{

public:

    void ff()

    {

        cout << "没有覆盖虚函数!" << endl;

    }

};

int _tmain(int argc, _TCHAR* argv[])

{

    A *p = new A();

    B b;

    C c;

    D d;

    p->f();

    p = &b;

    p->f();

    p = &c;

    p ->f();

    p = &d;

    p->f(); //基类指针可以调用虚函数的实现函数。即使它是派生类对象当中的成员。

    //p -> ff(); //基类指针没法调用派生类当中的成员。

    d.ff();

    return 0;

}

纯虚函数与抽象类

解释:含有一个或多个的纯虚函数的类叫抽象类。

特点:抽象类只能被继承不能被实例化。派生类必须覆盖这个纯虚函数,这个和虚函数有点不同,虚函数中,它的派生类不一定要必须覆盖。有时候,我们可以这样理解:

抽象类的派生类,才是我们需要的类,我们通过派生类,就可以利用基类的接口,调用基类的成员,而且可以基于派生类的不同而调用不同版本的虚函数的实现函数。

即:可以根据自己的需要,对纯虚函数重新定义,来满足自己的需要。

举例:

// BlankTest.cpp : 定义控制台应用程序的入口点。

//知识点:虚函数

//虚函数的强大之处在于能够预测未来。

//记住:基类指针可以指向任何派生类型的对象,但不能访问派生类型对象中定义的成员,但是可以调用虚函数(纯虚也可以)的实现函数。

#include "stdafx.h"

#include <iostream>

using namespace std;

 

class A

{

public:

    virtual void f() = 0 //纯虚函数

    {

        cout << "我是虚函数 " << endl;

    }

 

    void aa()

    {

        cout << "基函数的成员函数!" << endl;

    }

protected:

private:

};

 

class B : public A

{

public:

    void f()

    {

        cout << "我派生自A " << endl;

    }

};

 

class C : public B

{

public:

    void f()

    {

        cout << "我派生自B"<< endl;

    }

};

 

class D :public A// 没有覆盖A中的虚函数

{

public:

    void ff()

    {

        cout << "没有覆盖虚函数!" << endl;

    }

};

int _tmain(int argc, _TCHAR* argv[])

{

    B b;

    C c;

    //D d; //D没有实现纯虚函数,所以不能定义对象。

    b.f();

    b.aa();

    c.f();

    c.aa();

    return 0;

}

 

多态:

多态分为两种: 编译时的多态与运行时的多态。

而编译时的多态一般指早期绑定;运行时的多态指动态绑定。

早期绑定:函数的重载,函数的调用以及运算符重载。简单点说,就是在编译的时候就可以确定调用某一个函数了,这个函数是确定的。

动态绑定:一般指虚函数。即:运行时才能确定函数的调用。

 

  1. 预处理器先用编译器对源代码进行处理。

    主要有以下几种: #include #define #ifndef #if #else

    我们可以应用预处理器提高编程的效率。

    举例:

    // BlankTest.cpp : 定义控制台应用程序的入口点。

    #include "stdafx.h"

    #include <iostream>

    using namespace std;

     

    //#define DEBUG //通过这行可以决定 调试信息 的有无,这个比较好。赞一个。

    int _tmain(int argc, _TCHAR* argv[])

    {

    #ifdef DEBUG

        cout << "我在调试模式下!" << endl;

    #endif // _DEBUG

     

        cout << "我在运行模式下"<<endl;

        return 0;

    }

  2. C++运算符重载

    运算符重载和函数重载有异曲同工之妙,不同之处在于运算符重载,载的是构造类型的数据,基本数据类型的数据不需要重载。重载的目的是为了让运算更加简洁、明白。

    所谓的重载是对里面的运算符进行重新使用,而且只是针对这一种运算符,不是其他的。

    语法格式:

    返回类型 operator 单目运算符(参数列表)

    {

    //函数体

    }

    返回类型 operator 双目运算符(虚参1, 虚参2)

    {

    //函数体

    }

    注:对于双目运算符,两个虚参中至少得有一个是构造类型。

  3. C++命名空间

    所谓命名空间,你心里面想着文件夹的概念就OK了,主要用处就是为了避免冲突。

    比如,我在a文件夹下建立文件b,我也可以在c文件夹下建立文件b,但就是不能在 a 文件夹下,建立两个b。

    格式:

    namespace 命名空间名称

    {

    //函数体

    }

    而且,命名空间也要符合命名规范,所以对于命名空间名称我们也是可以起别名的,比如 namespace geography information system,我们可以简写为: namespace GIS = geography information system;

    以上只是一种使用别名的规范而已。

    Using 的作用:就是使当前文件夹下所有的函数名 暴露 在源程序中,这样我们就可以省略写 "命名空间名:: 使用到的名称或命令"。

    举例:

    // BlankTest.cpp : 定义控制台应用程序的入口点。

    #include "stdafx.h"

    #include <iostream>

    //using namespace std;

    using std::cout;

     

    //以下两个同名的函数:之所以不报错,是因为他们在不同的命名空间下。

    //#define DEBUG //通过这行可以决定 调试信息 的有无,这个比较好。赞一个。

    namespace zhu

    {

        void show()

        {

            cout << "我在zhu这个namespace下!" << std::endl;

        }

    }

    namespace xue

    {

        void show()

        {

            cout << "我在xue这个namespace下!" << std::endl;

        }

    }

    int _tmain(int argc, _TCHAR* argv[])

    {

    zhu::show();

         xue::show();

    #ifdef DEBUG

        cout << "我在调试模式下!" << endl;

    #endif // _DEBUG

     

        cout << "我在运行模式下"<< std::endl;

        return 0;

    }

命名空间和类联系在一起,珠联璧合,天下无敌。命名空间是类的一种有效的组织结构方式,我们用命名空间可以有效的将类进行分类管理,每个类有一个命名空间,这样我们在做大型程序开发时,可以为类库添加结构和层次的组织关系。

举例:

// BlankTest.cpp : 定义控制台应用程序的入口点。

#include "stdafx.h"

#include <iostream>

//using namespace std;

using std::cout;

 

//以下两个同名的函数:之所以不报错,是因为他们在不同的命名空间下。

//#define DEBUG //通过这行可以决定 调试信息 的有无,这个比较好。赞一个。

namespace zhu

{

    void show()

    {

        cout << "我在zhu这个namespace下!" << std::endl;

    }

}

namespace xue

{

    void show()

    {

        cout << "我在xue这个namespace下!" << std::endl;

    }

}

 

namespace kui

{

    class A

    {

    public:

        void show()

        {

            cout << "我是属于kui命名空间里面A类里面的函数!" << std::endl;

        }

    };

}

namespace kui1

{

    class A

    {

    public:

        void show()

        {

            cout << "我是属于kui1命名空间里面A类里面的函数!" << std::endl;

        }

    };

}

int _tmain(int argc, _TCHAR* argv[])

{

zhu::show();

     xue::show();

 

     kui::A a;

     a.show();

 

     kui1::A b;

     b.show();

#ifdef DEBUG

    cout << "我在调试模式下!" << endl;

#endif // _DEBUG

 

    cout << "我在运行模式下"<< std::endl;

    return 0;

}

C++学习总结 复习篇2的更多相关文章

  1. C++ 学习总结 复习篇

      友元的使用 分为友元类和友元函数     //友元类与友元函数的共同点:都可以让某一个类作为另一个类或者函数的参数.          //友元类:它让当前类成为另一个类的友元,然后,另一个类 ...

  2. NOIP复习篇

    NOIP复习篇---枚举 --------------------------------------------------------------------------------------- ...

  3. Java工程师学习指南 初级篇

    Java工程师学习指南 初级篇 最近有很多小伙伴来问我,Java小白如何入门,如何安排学习路线,每一步应该怎么走比较好.原本我以为之前的几篇文章已经可以解决大家的问题了,其实不然,因为我之前写的文章都 ...

  4. 数据库MySQL学习笔记高级篇

    数据库MySQL学习笔记高级篇 写在前面 学习链接:数据库 MySQL 视频教程全集 1. mysql的架构介绍 mysql简介 概述 高级Mysql 完整的mysql优化需要很深的功底,大公司甚至有 ...

  5. 一步步学习javascript基础篇(0):开篇索引

    索引: 一步步学习javascript基础篇(1):基本概念 一步步学习javascript基础篇(2):作用域和作用域链 一步步学习javascript基础篇(3):Object.Function等 ...

  6. 一步步学习javascript基础篇(3):Object、Function等引用类型

    我们在<一步步学习javascript基础篇(1):基本概念>中简单的介绍了五种基本数据类型Undefined.Null.Boolean.Number和String.今天我们主要介绍下复杂 ...

  7. Python3学习(3)-高级篇

    Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 文件读写 源文件test.txt line1 line2 line3 读取文件内容 f = ope ...

  8. Python3学习(2)-中级篇

    Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 切片:取数组.元组中的部分元素 L=['Jack','Mick','Leon','Jane','A ...

  9. Python3学习(1)-基础篇

    Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 安装(MAC) 直接运行: brew install python3 输入:python3 --v ...

随机推荐

  1. 牛客网暑期ACM多校训练营(第五场)F take(概率, 递推)

    链接: https://www.nowcoder.com/discuss/84119 题意: 给定n个箱子, 每个箱子打开发现钻石的概率P(这里的P要除100), 每个钻石的重量, 有一个人只能持有一 ...

  2. JavaScript正则表达式-定义

    通过RegExp()构造函数创建RegExp对象来定义正则表达式. var reg_pattern = new RegExt("a\d"); 通过字面语法直接定义正则表达式. va ...

  3. 算法学习记录-图——应用之拓扑排序(Topological Sort)

    这一篇写有向无环图及其它的应用: 清楚概念: 有向无环图(DAG):一个无环的有向图.通俗的讲就是从一个点沿着有向边出发,无论怎么遍历都不会回到出发点上. 有向无环图是描述一项工程或者系统的进行过程的 ...

  4. BZOJ 2725: [Violet 6]故乡的梦

    求出最短路径树,对于一个询问(x,y) 若不在树上S->T的链上,则答案不变,若在链上,考虑用一条非树边替换这条边,这条非树边必须跨越x->y这条边,线段树维护区间最小值 #include ...

  5. 【SaltStack】在Master上给Minion端安装zabbix

    一.IP信息说明 [Master] IP: 192.168.236.100 [Minion] IP: 192.168.236.101 二.配置SaltStack 关于SaltStack Master和 ...

  6. NYOJ 104 最大和

    最大和 时间限制:1000 ms  |  内存限制:65535 KB 难度:5   描述 给定一个由整数组成二维矩阵(r*c),现在需要找出它的一个子矩阵,使得这个子矩阵内的所有元素之和最大,并把这个 ...

  7. TOJ 5021: Exchange Puzzle

    5021: Exchange Puzzle  Time Limit(Common/Java):1000MS/3000MS     Memory Limit:65536KByteTotal Submit ...

  8. 微软.net一些类的源码

    地址:http://referencesource.microsoft.com/#mscorlib/system/collections/generic/dictionary.cs 关键字:refer ...

  9. SPOJ - ADAFIELD ,Set+map,STL不会超时!

    ADAFIELD - Ada and Field 这个题,如果用一个字来形容的话:-----------------------------------------------嗯! 题意:n*m的空白 ...

  10. Mountaineers

    Mountaineers 时间限制: 3 Sec  内存限制: 128 MB 题目描述 The Chilean Andes have become increasingly popular as a ...