为什么c++中,有时可以用类名直接访问非静态成员函数?

正规的C++语言标准目前(截止到C++14)应该还不支持这种调用方法。目前微软似乎在它的VC++中推行一种叫做C++/CLI的标准,有可能会支持这种调用,如果一定要用这种调用方法的话,还应该用VS2013尝试编译运行一下。

实际上,C++语言中类的静态成员函数本身应该是所有这一类对象的集体所具有的行为,就是说,不是某一个对象能够具有或者说实现的;而非静态成员函数应该是某一个对象自己的动作行为,跟本类其他对象乃至整个类关系不大,是对象依靠自己的数据以及函数参数就可以完成的行为。根据以上的讨论,我们可以看出,很难存在一种需求,使得一个成员函数不需要引用本对象的非静态成员,同时又必须是一个对象自己的行为(即声明为非静态成员函数)。如果真的存在不引用非静态成员的成员函数,那还是直接声明为静态成员函数为好,这样就可以万无一失地通过编译,也避免了移植性问题。

说句题外话,非静态成员函数总会有一个隐含的参数,就是this指针。通过反汇编分析也可以发现,非静态成员函数的调用属于特殊的thiscall,就是说总会传入一个this指针。而静态成员函数和类外的函数一样,经过编译后都是普通的调用,不会得到this指针,因此也不可能访问非静态成员(因为非静态成员的引用总是通过this指针完成的)。因此一个函数能否通过类名调用,主要还是要看它是否需要编译器传入this指针(要看编译后的代码,源代码级别上的调用是看不到传入的this指针的)。

如果真的希望在没有实例的前提下,调用一个非静态成员函数,可以使用下面的方法(前提是必须符合您提出的那个条件,即不访问任何非静态成员,如果它访问了非静态成员,则可能导致内存读写异常):

// 假设要引用的类类型为 TargetType, 成员函数为 void TargetType::TargetFunc();
// C++11 版:
static_cast<TargetType *>(nullptr)->TargetFunc();
// C++98/03 版:
reinterpret_cast<TargetType *>(0)->TargetFunc();
// 如果使用 C 风格的类型转换操作符:
((TargetType *) 0)->TargetFunc();

转自:https://segmentfault.com/q/1010000005650649/a-1020000005651548
时间: 2024-08-07 04:14:56

为什么c++中,有时可以用类名直接访问非静态成员函数?的相关文章

回调函数中调用类中的非静态成员变量或非静态成员函数

有关这方面的问题,首先说一点: 回调函数必须是静态成员函数或者全局函数来实现回调函数,大概原因是普通的C++成员函数都隐含了一个函数参数,即this指针,C++通过传递this指针给成员函数从而实现函数可以访问类的特定对象的数据成员.由于this指针的原因,使得一个普通成员函数作为回调函数时就会因为隐含的this指针问题使得函数参数个数不匹配,从而导致回调函数编译失败. 基于上面的理论,如何在类中封装回调函数呢? 回调函数只能是全局函数或者静态成员函数,但是由于全局函数会破坏封装性,所以只能用静

为什么在静态方法中 不能访问非静态成员

程序最终都将在内存中执行,变量只有在内存中占有一席之地时才能被访问.类的静态成员(变量和方法)属于类本身,在类加载的时候就会分配内存,可以通过类名直接去访问:非静态成员(变量和方法)属于类的对象,所以只有在类的对象产生(创建类的实例)时才会分配内存,然后通过类的对象(实例)去访问.在一个类的静态成员中去访问其非静态成员之所以会出错是因为在类的非静态成员不存在的时候类的静态成员就已经存在了,访问一个内存中不存在的东西当然会出错:class CA{private:int a; //非静态成员,创建类

在c++中,静态数据成员可以被非静态成员函数调用吗?如果可以调用的话那为什么还要定义静态成员函数呢

静态数据成员可以被非静态成员函数访问.但静态成员函数只能访问静态数据成员.静态数据成员从属于某一个类,而不是某一个类的 对象.同理,静态成员函数也是一样. 追问 定义静态成员函数的作用有何在呢 回答 静态成员函数隶属于类,不用实例化对象,就可以直接调用静态成员函数来操作静态数据成员 提问者评价 太给力了,你的回答完美解决了我的问题!

C++类中的静态成员变量与静态成员函数

最近一直看c++相关的项目,但总是会被c++类中的静态成员变量与静态成员函数的理解感觉很是模糊,不明白为什么类中要是用静态成员变量.于是在网上搜集了一些资料,自己再稍微总结下. 静态成员的概念: 静态类中的成员加入static修饰符,即是静态成员.可以直接使用类名+静态成员名访问此静态成员,因为静态成员先于类的声明而存在于内存,也可以根据类声明的对象来访问.而非静态成员必须实例化之后才会分配内存. 非静态成员的概念: 所有没有加static的成员都是非静态成员.而类被实例化后,可以通过实例化的类

C++中静态成员变量和静态成员函数(面向过程&amp;&amp;面向对象)

数据成员可以分静态变量.非静态变量两种. 静态成员:静态类中的成员加入static修饰符,即是静态成员.可以直接使用类名+静态成员名访问此静态成员,因为静态          成员存在于内存,非静态成员需要实例化才会分配内存,所以静态成员不能访问非静态的成员..因为静态成员  存在于内存,所以非静态成员可以直接访问类中静态的成员. 非成静态员:所有没有加Static的成员都是非静态成员,当类被实例化之后,可以通过实例化的类名进行访问..非静态成             员的生存期决定于该类的生存

静态成员函数为什么不能访问本类中的非静态成员?

和静态数据成员一样,静态成员函数是类的一部分,而不是对象的一部分.如果要在类外调用公用的静态成员函数,要用类名和域运算符"∷".如Box∷volume( );实际上也允许通过对象名调用静态成员函数,如a.volume( );但这并不意味着此函数是属于对象a的,而只是用a的类型而已.静态成员函数的作用是为了能处理静态数据成员.可以说,静态成员函数与非静态成员函数的根本区别是:非静态成员函数有this指针,静态成员函数并不属于某一对象,它与任何对象都无关,静态成员函数没有this指针.由此

单实例中对静态成员函数及变量的回顾

程序最终都将在内存中执行,变量只有在内存中占有一席之地时才有被访问的可能,除此之外,还要有相应的权限. 静态成员函数和非静态成员函数的一个重大区别是,静态成员函数不接受隐含的this指针,它是属于类的,是公有的,因此,静态成员是类的所有对象中共享的成员,而不是某个对象的成员:非静态成员函数带有隐含的this指针,它是属于对象的. 类的静态成员(变量和方法)属于类本身,在类加载的时候就会分配内存,存储在静态存储区,可以通过类名直接去访问:非静态成员(变量和方法)属于类的对象,所以只有在类的对象产生

使用反射创建Bean、Spring中是如何根据类名配置创建Bean实例、Java提供了Class类获取类别的字段和方法,包括构造方法

Java提供了Class类,可以通过编程方式获取类别的字段和方法,包括构造方法 获取Class类实例的方法: 类名.class 实例名.getClass() Class.forName(className) public class RefTest { @Test public void testRef(){ //Class cls = RefTest.class; //Class.forName("com.jboa.service.RefTest"); //new RefTest()

linux下设置eclipse中的项目和类名的字体大小

由于刚装的eclipse中的项目和类名字体太小,并且windows中没有可以更改的方法, 所以参考了网上一些文章,终于修改成功,所以反馈进行分享,望国内开源风气如stackoverflow一样. 步骤:(本人的本机环境是win10,ubuntu版本16.04.1) 1.首先找到eclipse中的安装目录,eclipse.在eclipse/plugins/org.eclipse.ui.themes/css中,有许多的css文件,由于没有linux后缀的css文件,所以我 更改了e4_default