C++中静态成员函数访问成员变量问题

本文探讨了C++中静态成员函数无法直接访问非静态成员的问题,并提出了四种解决方案,包括将所有成员声明为静态、传入对象引用、声明空指针及维护this指针列表等方法。
C++会区分两种类型的成员函数:静态成员函数和非静态成员函数。这两者之间的一个重大区别是,静态成员函数不接受隐含的this自变量。所以,它就无法访问自己类的非静态成员。

左例中分别声明了成员函数void add(),成员变量int a,和静态成员函数 static void staticfunc(),静态成员变量 static int sa.

在staticfunc()的定义中,由于没有this指针,所以无法访问普通成员变量和成员函数.

网络上有很多关于解决静态成员函数访问普通成员变量的方法,总结如下:

方法一:声明类的所有数据成员都是静态的。运用这种方式的话,静态的成员函数就能够直接地访问它们,例如:

此时,即可在staticfunc()中正常访问变量a了,如果还需访问add()函数,则将add()也冠以static即可.

该方法缺点是,很多情况下,你并不愿意将所有的变量都改成静态成员.或者有些变量不能是静态的.此时该方法失效.

方法二:传入对象参照信息:

静态成员函数staticfunc()会使用参照Obj来访问非静态成员a.

将一个参照或者指针作为静态成员函数的自变量传递,就是在模仿自动传递非静态成员函数里this自变量这一行为。

该方法很明确,给静态函数传入this指针.

一般来说静态成员函数的参数没有严格的要求时,采用这种方法不失为妙招.如果静态成员函数的传入参数有严格的限制,就是说不能传入实体引用时,就不能使用这种方法了.

方法三:申明空指针

这种方法在类只有一个实体时可行,因为传入的this指针是一个全局量,始终指向最后定义的那个实体,当定义了第n+1个实体候,第n个实体的this指针就被忘掉了.在这种方法的启发下,我采用了下面的方法,比较麻烦些,但确实可行.之所以可行,前提是静态成员函数是一个回调函数,它返回了一个实体标识,但有别于this指针.

方法四:定义全部量(可以是数组,也可是链表)保存每个this指针

定义结构体:


然后在每次生成实体时:

其中Insert()功能为将this指针和实体标识video_handle加入链表.相当于将每个实体标识与其对应的this指针保存起来了.所以在回调函数中使用时只需根据回调函数返回的实体表示video_handle来查找对应的this即可.

问题终于解决,麻烦的就是要增加对链表的操作.

### 静态成员函数访问限制 在 C++ 中,静态成员函数具有特定的访问限制,主要体现在其对成员变量成员函数访问能力上。静态成员函数的设计初衷是为了管理和操作静态成员变量,而不是非静态成员。 #### 静态成员函数只能访问静态成员变量 静态成员函数只能访问静态成员变量,这是因为静态成员函数没有 `this` 指针[^4]。`this` 指针是普通成员函数用来访问对象的非静态成员变量的关键。由于静态成员函数不与任何特定的对象实例绑定,因此无法通过它访问静态成员变量。 例如,在以下代码中,静态成员函数 `func()` 可以访问静态成员变量 `m_A`,但不能访问静态成员变量 `m_B`: ```cpp class Person { public: static void func() { m_A = 100; // 静态成员函数可以访问静态成员变量 // m_B = 200; // 编译错误:静态成员函数不能访问静态成员变量 } static int m_A; int m_B; }; int Person::m_A = 0; void test01() { Person p; p.func(); // 通过对象调用静态成员函数 Person::func(); // 通过类名调用静态成员函数 } ``` 上述代码中,`func()` 函数能够修改 `m_A` 的值,但尝试修改 `m_B` 会导致编译错误。这是因为 `m_B` 是一个非静态成员变量,只有在类的具体实例化对象存在时才能被访问。 #### 静态成员函数不能访问静态成员函数 除了不能访问静态成员变量外,静态成员函数也不能直接调用非静态成员函数。这是因为非静态成员函数依赖于具体的对象实例来执行[^3]。静态成员函数没有关联到任何对象实例,因此无法确定调用哪个对象的非静态成员函数。 例如: ```cpp class MyClass { public: static void staticFunc() { // nonStaticFunc(); // 编译错误:静态成员函数不能直接调用非静态成员函数 } void nonStaticFunc() { // ... } }; ``` 在上面的例子中,`staticFunc()` 尝试调用 `nonStaticFunc()` 会引发编译错误,因为 `nonStaticFunc()` 是一个非静态成员函数,需要一个对象实例来调用。 #### 访问权限的影响 静态成员函数同样受到访问控制符(如 `private`、`protected` 和 `public`)的限制。如果静态成员函数被声明为 `private`,则它只能在类内部访问,而不能从类外部调用[^1]。 例如: ```cpp class MyClass { private: static void privateFunc() { // ... } public: static void publicFunc() { privateFunc(); // 允许调用私有静态成员函数 } }; void test() { MyClass::publicFunc(); // 正确:通过公共静态成员函数间接调用私有静态成员函数 // MyClass::privateFunc(); // 错误:类外无法直接访问私有静态成员函数 } ``` 在这个例子中,`privateFunc()` 是私有的,因此不能从类外部直接调用。然而,可以通过公共的静态成员函数 `publicFunc()` 来间接调用 `privateFunc()`。 #### 静态成员函数的用途 静态成员函数的主要用途是管理静态成员变量,完成对静态数据成员的封装[^2]。它们可以在没有创建类实例的情况下被调用,这使得静态成员函数非常适合用于初始化或清理静态资源、提供工具方法等场景。 例如: ```cpp class Counter { public: static int getCount() { return count; } static void incrementCount() { ++count; } private: static int count; }; int Counter::count = 0; void testCounter() { Counter::incrementCount(); std::cout << "Current count: " << Counter::getCount() << std::endl; } ``` 在这个例子中,`getCount()` 和 `incrementCount()` 都是静态成员函数,它们用于管理和操作静态成员变量 `count`。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值