C++学习总结 复习篇2
延续上一小节内容:下面继续讲解虚函数和多态
- 虚函数和多态
- 基类指针可以指向任何派生类的对象,但是不能调用派生类对象的成员。
- 但是,基类可以调用覆盖了虚函数的函数。(现在调用将来,这有问题,说明现在影响了将来。)
- 基类可以被继承,如果虚函数没有被实现,可以继续被下一个类继承。
- 当派生类没有能够覆盖虚函数的时候,若派生类的对象访问这个函数,那么此时将使用基类定义的函数。(现在调用过去,没有问题,已经定义过)
举例:
// 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;
}
多态:
多态分为两种: 编译时的多态与运行时的多态。
而编译时的多态一般指早期绑定;运行时的多态指动态绑定。
早期绑定:函数的重载,函数的调用以及运算符重载。简单点说,就是在编译的时候就可以确定调用某一个函数了,这个函数是确定的。
动态绑定:一般指虚函数。即:运行时才能确定函数的调用。
-
预处理器先用编译器对源代码进行处理。
主要有以下几种: #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;
}
- C++运算符重载
运算符重载和函数重载有异曲同工之妙,不同之处在于运算符重载,载的是构造类型的数据,基本数据类型的数据不需要重载。重载的目的是为了让运算更加简洁、明白。
所谓的重载是对里面的运算符进行重新使用,而且只是针对这一种运算符,不是其他的。
语法格式:
返回类型 operator 单目运算符(参数列表)
{
//函数体
}
返回类型 operator 双目运算符(虚参1, 虚参2)
{
//函数体
}
注:对于双目运算符,两个虚参中至少得有一个是构造类型。
- 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的更多相关文章
- C++ 学习总结 复习篇
友元的使用 分为友元类和友元函数 //友元类与友元函数的共同点:都可以让某一个类作为另一个类或者函数的参数. //友元类:它让当前类成为另一个类的友元,然后,另一个类 ...
- NOIP复习篇
NOIP复习篇---枚举 --------------------------------------------------------------------------------------- ...
- Java工程师学习指南 初级篇
Java工程师学习指南 初级篇 最近有很多小伙伴来问我,Java小白如何入门,如何安排学习路线,每一步应该怎么走比较好.原本我以为之前的几篇文章已经可以解决大家的问题了,其实不然,因为我之前写的文章都 ...
- 数据库MySQL学习笔记高级篇
数据库MySQL学习笔记高级篇 写在前面 学习链接:数据库 MySQL 视频教程全集 1. mysql的架构介绍 mysql简介 概述 高级Mysql 完整的mysql优化需要很深的功底,大公司甚至有 ...
- 一步步学习javascript基础篇(0):开篇索引
索引: 一步步学习javascript基础篇(1):基本概念 一步步学习javascript基础篇(2):作用域和作用域链 一步步学习javascript基础篇(3):Object.Function等 ...
- 一步步学习javascript基础篇(3):Object、Function等引用类型
我们在<一步步学习javascript基础篇(1):基本概念>中简单的介绍了五种基本数据类型Undefined.Null.Boolean.Number和String.今天我们主要介绍下复杂 ...
- Python3学习(3)-高级篇
Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 文件读写 源文件test.txt line1 line2 line3 读取文件内容 f = ope ...
- Python3学习(2)-中级篇
Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 切片:取数组.元组中的部分元素 L=['Jack','Mick','Leon','Jane','A ...
- Python3学习(1)-基础篇
Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 安装(MAC) 直接运行: brew install python3 输入:python3 --v ...
随机推荐
- 牛客网暑期ACM多校训练营(第五场)F take(概率, 递推)
链接: https://www.nowcoder.com/discuss/84119 题意: 给定n个箱子, 每个箱子打开发现钻石的概率P(这里的P要除100), 每个钻石的重量, 有一个人只能持有一 ...
- JavaScript正则表达式-定义
通过RegExp()构造函数创建RegExp对象来定义正则表达式. var reg_pattern = new RegExt("a\d"); 通过字面语法直接定义正则表达式. va ...
- 算法学习记录-图——应用之拓扑排序(Topological Sort)
这一篇写有向无环图及其它的应用: 清楚概念: 有向无环图(DAG):一个无环的有向图.通俗的讲就是从一个点沿着有向边出发,无论怎么遍历都不会回到出发点上. 有向无环图是描述一项工程或者系统的进行过程的 ...
- BZOJ 2725: [Violet 6]故乡的梦
求出最短路径树,对于一个询问(x,y) 若不在树上S->T的链上,则答案不变,若在链上,考虑用一条非树边替换这条边,这条非树边必须跨越x->y这条边,线段树维护区间最小值 #include ...
- 【SaltStack】在Master上给Minion端安装zabbix
一.IP信息说明 [Master] IP: 192.168.236.100 [Minion] IP: 192.168.236.101 二.配置SaltStack 关于SaltStack Master和 ...
- NYOJ 104 最大和
最大和 时间限制:1000 ms | 内存限制:65535 KB 难度:5 描述 给定一个由整数组成二维矩阵(r*c),现在需要找出它的一个子矩阵,使得这个子矩阵内的所有元素之和最大,并把这个 ...
- TOJ 5021: Exchange Puzzle
5021: Exchange Puzzle Time Limit(Common/Java):1000MS/3000MS Memory Limit:65536KByteTotal Submit ...
- 微软.net一些类的源码
地址:http://referencesource.microsoft.com/#mscorlib/system/collections/generic/dictionary.cs 关键字:refer ...
- SPOJ - ADAFIELD ,Set+map,STL不会超时!
ADAFIELD - Ada and Field 这个题,如果用一个字来形容的话:-----------------------------------------------嗯! 题意:n*m的空白 ...
- Mountaineers
Mountaineers 时间限制: 3 Sec 内存限制: 128 MB 题目描述 The Chilean Andes have become increasingly popular as a ...