0

0

C++中的协变与逆变是什么?C++函数指针与返回类型详解【类型系统】

尼克

尼克

发布时间:2026-01-01 18:24:08

|

363人浏览过

|

来源于php中文网

原创

C++仅在虚函数返回类型(指针/引用)中支持协变,禁止逆变;函数指针和模板参数均严格不变,需手动包装实现语义适配。

c++中的协变与逆变是什么?c++函数指针与返回类型详解【类型系统】

协变(covariance)和逆变(contravariance)在 C++ 中**不直接作为语言关键字存在**,而是描述类型转换关系的术语,主要体现在继承体系中指针/引用的转换行为、模板参数的类型适配,以及 虚函数重写时返回类型和异常规范的放宽规则。C++ 对协变支持有限且明确,对逆变基本不支持(尤其在函数参数上)。函数指针的类型匹配则严格遵循“形参类型精确一致 + 返回类型精确一致”,没有自动协变或逆变转换。

协变:只允许出现在返回类型中(且仅限指针/引用)

当派生类重写基类虚函数时,如果返回的是类类型的指针或引用,C++ 允许返回更“具体”的类型——只要它是原返回类型的派生类。这叫返回类型协变

  • 必须是虚函数重写(override),且原函数返回的是基类的指针或引用(如 Base*Base&
  • 派生类函数可返回派生类的指针或引用(如 Derived*Derived&),编译器认可这种转换安全
  • 不能用于值类型(如 BaseDerived)、非指针/引用类型,也不能用于参数类型

例子:

class Base { virtual Base* clone() { return new Base; } };
class Derived : public Base {
  Derived* clone() override { return new Derived; } // ✅ 合法:Base* → Derived* 是协变
};

逆变:C++ 基本不支持,尤其禁止在函数参数中使用

逆变指“更通用的类型可替代更具体的类型”,典型如函数参数:若某处期待 Derived*,能否传入 Base*?答案是否定的——C++ 函数参数是不变的(invariant)

立即学习C++免费学习笔记(深入)”;

Kacha
Kacha

KaCha是一款革命性的AI写真工具,用AI技术将照片变成杰作!

下载
  • 派生类重写虚函数时,参数类型必须与基类完全一致;哪怕你把参数从 Derived* 改成 Base*,也不算重写,而是重载或编译错误
  • 函数指针、std::function、lambda 捕获等场景,参数类型不进行向上转型(即无逆变)
  • 这是为了保证类型安全:父类接口承诺“能处理任意 Derived*”,若子类只接受 Base*,就可能漏掉 Derived 特有行为,破坏 LSP(里氏替换原则)

函数指针的类型系统:严格不变(invariant)

C++ 中函数指针是完全类型化的:返回类型、每个参数的类型、const/volatile 限定符、调用约定(如 __cdecl)全部相同,才算同一类型。

  • int(*)(double)int(*)(float) 是不同类型,不可互转
  • void(*)()void(*)() const(成员函数)不兼容
  • 即使 Derived* 可隐式转为 Base*void(*)(Derived*)不能赋给 void(*)(Base*) —— 参数位置不协变也不逆变
  • 返回类型同样严格:Base* (*)() 不能赋给 Derived* (*)(),除非是虚函数重写场景(此时靠协变规则特许)

模板与 std::function 中的“伪协变”需手动适配

std::function 无法直接绑定 void(Derived*) 类型的函数,但可通过 lambda 包装实现语义等价:

void handle_base(Base* b) { /* ... */ }
void handle_derived(Derived* d) { /* ... */ }

std::function f1 = handle_base; // ✅ 直接赋值
std::function f2 = [](Base* b) {
  if (auto d = dynamic_cast(b)) handle_derived(d);
}; // ✅ 手动桥接,非语言级协变

这不是编译器自动做的协变,而是程序员用运行时检查+包装实现的逻辑适配。

基本上就这些。C++ 的类型系统偏保守:只在虚函数返回类型上开放协变这一处“安全缺口”,其余地方坚持不变性,以确保静态可验证的安全。理解这点,就能避开很多“为什么不能转”的困惑。

相关专题

更多
css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

553

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

95

2025.10.23

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

713

2023.08.22

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

519

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

312

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

522

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

190

2025.08.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

65

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 5.7万人学习

C 教程
C 教程

共75课时 | 3.8万人学习

C++教程
C++教程

共115课时 | 10.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号