Android C++系列:C++最佳实践1虚函数

简介: C++多态的核心技术基础就是虚函数,虚函数允许我们使用同样的基类指针调用同一个方法的不同实现版本。我们Android使用Java开发过程中,方法重写技术自动实现了多态,C++角度可能更繁琐一些,本文从Java程序员思维角度来阐述C++虚函数及开发过程一些准则。

image.png


1. 背景


C++多态的核心技术基础就是虚函数,虚函数允许我们使用同样的基类指针调用同一个方法的不同实现版本。我们Android使用Java开发过程中,方法重写技术自动实现了多态,C++角度可能更繁琐一些,本文从Java程序员思维角度来阐述C++虚函数及开发过程一些准则。


2. 什么是虚函数


在Java中我们实现继承结构的两个类:


class Base{
  public void action(){
    System.out.pritln("in Base");
  }
}
class Sub extend Base{
  public void action(){
    System.out.println("in Sub");
  }
}
public static void main(String[] args){
  Base b = new Base();
  b.action();
  b = new Sub();
  b.action();
}


Java直接覆盖后就会自动的调用到实际类的对应方法,但是C++中不行。


class Base{
public:
  void action();
}
void Base::action(){
    std::cout>>("in Base");
}
class Sub:public Base{
public:
  void action();
}
void Sub::action(){
    std::cout>>("in Sub");
  }
static void main(){
  Base *b = new Base();
  b->action();
  b = new Sub();
  b->action();
}


打印的结果是:


in Base
in Base


就算是b指向了Sub对象,但是打印的还是Base的方法,没有自动绑定,写惯了Java后马上慌了,怎么破。这里就需要用到我们今天的主人公”虚函数“了,在方法声明之前加上virtual这个函数就变成虚函数。我们在Base的action方法声明前加上virtual打印结果就变成了:


in Base
in Sub


当前仅当对通过指针或引用掉用虚函数时,才会在运行时解析该调用,也只有这种情况下对象的动态类型才有可能与静态类型不同。


3. 虚函数注意点


  1. 基类通过在其成员函数的声明语句之前加上关键字virtual使得该函数执行动态绑定。如果基类把一个函数声明成虚函数,则该函数在派生类中隐式地也是虚函数。派生类可以在它覆盖的函数前使用virtual关键字,但是不是强制的。
  2. 如果我们想将某个类作基类,则该类必须已经定义而非仅仅声明。因为派生类中包含并且可以使用它从基类继承而来的成员,为了使用这些成员,派生类当然要知道它们是什么。一个类不能派生它本身。
  3. 如果在基类中含有一个或多个虚函数,我们可以使用dynamic_cast请求一个类型装换,这个转换的安全检查在运行时执行。同样,如果我们已知某个基类向派生类的转换是安全的,我们可以使用static_cast来强制覆盖掉编译器的检查工作。
  4. 如果我们不使用某个函数,则无须为该函数提供定义,但是我们必须为每一个虚函数都提供定义,而不管它是否被用到,因为连编译器也无法确定到底会使用哪个虚函数。
  5. 派生类中虚函数的返回类型必须与基类函数匹配。例外的情况是,当类的虚函数返回类型是类本身的指针或引用时,不需要遵循这个规则。如果Impl由Base派生,则基类的虚函数可以返回Base*,而派生类的对应函数可以返回Impl*,这样的返回类型要求从Impl到Base的转换是可访问的。
  6. 我们可以把某个函数指定为final,如果已经把函数指定为final,则之后任何尝试覆盖该函数的操作都将引发错误。
  7. 如果虚函数使用默认实参,则积累和派生类中定义的默认实参最好一致。
  8. 如果我们不想对虚函数的调用执行动态绑定,可以使用作用域运算法强迫执行虚函数的某个特定版本。通常情况,只有成员函数(或友元)中的代码才需要使用作用域运算法来回避虚函数的机制。在Java中子类调用父类被重写的方法直接super.xxx()即可,在C++中要强制使用某个版本的虚函数,必须使用该版本对应的类加::作用域,如Base::action()。我们Java中如果想调用父类的父类的被覆盖的方法就有点困难了,这一点C++反而更有优势了。
  9. 如果一个派生类虚函数需要调用它的基类版本,但是没有使用作用域运算符,则在运行时该调用将被解析为对派生类版本自身的调用,从而导致无限递归。


4. 总结


作为Android方向C++系列文章,不仅介绍C++相关的知识,Android主要开发语言是Java,文章中尽量会对比Java和C++实现的一些不同,以及一些优劣对比。本文介绍了C++多态的基础虚函数,对比了和Java版本实现多态的差异。

目录
相关文章
|
10天前
|
存储 移动开发 数据库
构建高效Android应用:探究LiveData和ViewModel的最佳实践
【4月更文挑战第20天】 在动态演化的移动开发领域,构建一个既响应迅速又能够在用户界面保持稳定的Android应用是至关重要的。近年来,随着Android架构组件的推出,特别是LiveData和ViewModel的引入,开发者得以更有效地管理应用状态并优化用户界面的响应性。本文将深入探讨LiveData和ViewModel的实现机制,并通过案例分析展示如何结合它们来构建一个高效且健壮的Android应用架构。我们将重点讨论如何通过这些组件简化数据绑定过程、提高代码的可维护性和测试性,同时确保用户界面的流畅性。
|
8天前
|
Linux Android开发
测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android
测试程序之提供ioctl函数应用操作GPIO适用于Linux/Android
12 0
|
2天前
|
NoSQL API Redis
最佳实践|如何使用c++开发redis module
本文将试着总结Tair用c++开发redis module中遇到的一些问题并沉淀为最佳实践,希望对redis module的使用者和开发者带来一些帮助(部分最佳实践也适用于c和其他语言)。
|
2天前
|
编译器 C++
【C++进阶】引用 & 函数提高
【C++进阶】引用 & 函数提高
|
5天前
|
安全 vr&ar C++
C++:编程语言的演变、应用与最佳实践
C++:编程语言的演变、应用与最佳实践
|
6天前
|
C++
C++虚函数学习笔记
C++虚函数学习笔记
10 0
|
7天前
|
C++
C++从入门到精通:2.1.2函数和类——深入学习面向对象的编程基础
C++从入门到精通:2.1.2函数和类——深入学习面向对象的编程基础
|
7天前
|
存储 C++
C++从入门到精通:2.1.1函数和类
C++从入门到精通:2.1.1函数和类
|
8天前
|
Android开发
Android构建系统:Android.mk(2)函数详解
Android构建系统:Android.mk(2)函数详解
15 1
|
12天前
|
数据库 Android开发 开发者
实现高效安卓应用:探究LiveData和ViewModel的最佳实践
【4月更文挑战第19天】 在构建响应式的Android应用程序时,LiveData和ViewModel是两个核心组件。它们不仅提供了数据持有和界面更新的机制,还促进了组件间的解耦。本文将深入探讨如何通过结合LiveData和ViewModel来优化应用架构,提升用户体验,并确保数据的一致性和生存期管理。我们将透过实际案例分析,揭示这些技术如何协同工作以应对复杂的UI场景,并展示如何在实际项目中实施这些最佳实践。