C++函数中那些不可以被声明为虚函数的函数

本文解析了C++中无法将普通函数、构造函数、内联成员函数等声明为虚函数的原因,并阐述了这些限制背后的逻辑和技术考量。


出处:http://blog.youkuaiyun.com/hackbuteer1/article/details/6878255#comments


常见的不不能声明为虚函数的有:普通函数(非成员函数);静态成员函数;内联成员函数;构造函数;友元函数。


1、为什么C++不支持普通函数为虚函数?

      普通函数(非成员函数)只能被overload,不能被override,声明为虚函数也没有什么意思,因此编译器会在编译时邦定函数。

2、为什么C++不支持构造函数为虚函数?

      这个原因很简单,主要是从语义上考虑,所以不支持。因为构造函数本来就是为了明确初始化对象成员才产生的,然而virtual function主要是为了再不完全了解细节的情况下也能正确处理对象。另外,virtual函数是在不同类型的对象产生不同的动作,现在对象还没有产生,如何使用virtual函数来完成你想完成的动作。(这不就是典型的悖论)

3、为什么C++不支持内联成员函数为虚函数?

      其实很简单,那内联函数就是为了在代码中直接展开,减少函数调用花费的代价,虚函数是为了在继承后对象能够准确的执行自己的动作,这是不可能统一的。(再说了,inline函数在编译时被展开,虚函数在运行时才能动态的邦定函数)

4、为什么C++不支持静态成员函数为虚函数?

      这也很简单,静态成员函数对于每个类来说只有一份代码,所有的对象都共享这一份代码,他也没有要动态邦定的必要性。

5、为什么C++不支持友元函数为虚函数?

      因为C++不支持友元函数的继承,对于没有继承特性的函数没有虚函数的说法。
*********************************************************************

1、顶层函数多态的运行期行为体现在虚函数上,虚函数通过继承方式来体现出多态作用,顶层函数不属于成员函数,是不能被继承的。

2、构造函数:(1)构造函数不能被继承,因而不能声明为virtual函数。

(2)构造函数一般是用来初始化对象,只有在一个对象生成之后,才能发挥多态的作用,如果将构造函数声明为virtual函数,则表现为在对象还没有生成的情况下就使用了多态机制,因而是行不通的,如下例:

[cpp]  view plain  copy
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. class B  
  5. {  
  6. public:  
  7.     B() {}  
  8.     virtual void show()  
  9.     {  
  10.         cout<<"***"<<endl;  
  11.     }  
  12. };  
  13. class D:public B  
  14. {  
  15. public:  
  16.     D() {}  
  17.     void show()  
  18.     {  
  19.         cout<<"==="<<endl;  
  20.     }  
  21. };  
  22.   
  23. int main(void)  
  24. {  
  25.     B *pb;  
  26.     D d;        //先生成对象  
  27.     pb=&d;  
  28.     pb->show(); //再体现多态  
  29.   
  30.     pb=new D(); //先调用构造函数  
  31.     pb->show(); //再多态  
  32.     delete pb;  
  33.     return 0;  
  34. }  

3、static函数:不能被继承,只属于该类。

4、友元函数:友元函数不属于类的成员函数,不能被继承。

5、inline函数:inline函数和virtual函数有着本质的区别,inline函数是在程序被编译时就展开,在函数调用处用整个函数体去替换,而virtual函数是在运行期才能够确定如何去调用的,因而inline函数体现的是一种编译期机制,virtual函数体现的是一种运行期机制。此外,一切virtual函数都不可能是inline函数。

<think>首先,用户的问题是:“我想查询C++中静态函数是否可以声明虚函数 请问C++静态函数能否声明虚函数”。这直接询问静态函数是否能被声明虚函数。 从系统级指令中,我需要: - 使用$...$格式表示行内数学表达式,但这里可能涉及数学表达式。 - 独立公式用$$...$$格式并单独成段。 - LaTeX语法正确。 - 使用中文回答。 - 生成相关问题。 - 在回答中引用的段落末尾自然地添加引用标识,如[^1]。 - 回答结构清晰,帮助用户逐步解决问题,保证真实可靠。 现在,参考提供的引用: - 引用[1]:提到C++哪些函数能被声明虚函数,包括构造函数、静态成员函数和友元函数。 - 引用[2]:讨论为什么模板成员函数和静态函数能是虚函数。 - 引用[3]:明确说明静态成员函数能被声明为virtual函数,并给出代码示例。 - 引用[4]:解释静态函数虚函数的区别,强调静态函数没有this指针,地址固定,编译时确定。 这些引用都一致表明:在C++中,静态成员函数能被声明虚函数。 关键点总结: - 静态成员函数能被声明虚函数。 - 原因包括:静态函数没有this指针,调用时依赖对象实例,地址在编译时固定。 - 虚函数需要依赖于对象的虚表(vtable),这要求对象实例存在。 - 代码示例会编译失败。 我的回答结构: 1. **直接回答问题**:清晰、简洁地回应用户的问题。 2. **解释原因**:详细说明为什么静态函数能是虚函数。 3. **提供代码示例**:根据引用中的代码,展示尝试声明静态虚函数会编译失败。 4. **添加引用标识**:在相关解释后自然地添加引用,如[^3]。 5. **生成相关问题**:在最后添加相关问题部分。 6. **使用中文**:整个回答用中文。 步骤: - 开头直接回答问题:能。 - 然后解释原因:静态函数没有this指针,编译时确定;虚函数需要运行时多态。 - 展示代码示例:类如`virtual static void fun() { }`会编译错误。 - 引用来源:如引用[^3]中提到。 - 最后生成相关问题。 相关问题(基于指令中的示例): - 类似于示例的快速排序相关问题,我应该生成与主题相关的问题。 - 例如: 1. 为什么C++的静态成员函数能是虚函数? 2. 虚函数的工作原理是什么? 3. 在C++中,哪些其他函数能被声明虚函数? 确保回答真实可靠:基于提供的引用和C++标准。 </think> ### C++中静态函数能否声明虚函数? 在C++中,**静态成员函数能被声明虚函数**。这是C++语言的明确限制,任何尝试将静态函数声明为`virtual`都会导致编译错误。下面我将逐步解释原因,并提供代码示例。 #### 原因分析 静态成员函数虚函数的机制存在根本冲突: - **静态函数特性**:静态成员函数属于类本身,而非特定对象。它没有`this`指针,调用时依赖于对象实例(例如通过`ClassName::functionName()`直接调用)。其地址在编译时固定,因此编译期间就能确定调用位置[^4]。 - **虚函数特性**:虚函数依赖于运行时多态。每个对象通过虚表(vtable)动态绑定函数调用,这要求函数具有`this`指针来访问对象状态[^2]。 由于静态函数缺乏`this`指针,无法实现虚函数所需的动态绑定机制,因此无法声明虚函数C++标准禁止这种组合,以保持语言的一致性和效率[^1][^3]。 #### 代码示例 以下代码展示了尝试将静态成员函数声明为`virtual`时出现的编译错误: ```cpp class Test { public: // 编译错误:static成员函数声明为virtual virtual static void fun() { } // 错误:'virtual'能与'static'同时使用 }; int main() { Test::fun(); // 静态函数调用需要对象 return 0; } ``` - **错误说明**:编译器会报错(如GCC中的`error: member 'fun' cannot be declared both 'virtual' and 'static'`),因为`virtual`和`static`是互斥的关键字。 - **正确替代**:如果需要类似功能,可以通过普通虚函数结合静态方法实现多态,但静态函数本身能是虚函数。 #### 结论 静态成员函数声明虚函数C++的设计选择,确保编译时确定性与运行时多态的分离。如果您在代码中遇到相关问题,建议检查函数声明是否误用了`virtual static`组合,并考虑使用接口或工厂模式作为替代方案[^2]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值