【小设计】基于宏实现的C++ 可复用setter 和getter设计

前言

  • 最近在开发unity游戏的时候,面对庞大复杂的不同类之间进行数据交换和调用,我们必须做好类数据的信息管理,往往我希望暴露给其他类越少越好,这时候我就利用了C#的一个语言特性
cs 复制代码
public PlayerStateMachine stateMachine{get;private set;}
  • 熟悉的朋友不应该陌生,在面向对象编程中,settergetter 是用于访问和修改私有成员变量的方法。它们通常用于封装(encapsulation),这是一种将数据和行为组合在一个单元(即类)中,并仅通过公共接口访问和操作这些成员的技术。
  • 然后在C++中,却没有提供这个内置的语言特性,本期我来分享一个可复用的C++ 可复用setter 和getter设计

C++ 普通getter和setter的实现

  • 在正常情况下,setter和getter在C++没有原生态支持的情况下,我们可以仿照其设计思路,完成下述代码。
cpp 复制代码
class Context {
private:
    int value;

public:
    void setValue(const int&v) { value = v; }
    int getValue() const { return value; }
};
  • 上述代码看起来没有任何问题,在private作用域下,用户需要访问value必须通过getValue(),如果我们只希望用户能得到数值而不允许用户修改,我们只需要保留getValue()函数即可
  • 然后上述代码存在一个潜在的问题,当一个类中存在多个像这样维护的变量,那我岂不是要定义一堆新的getter函数
cpp 复制代码
public:
    int getValue1() const { return value1; }
    int getValue2() const { return value2; }
    int getValue3() const { return value3; }
    int getValue4() const { return value4; }
    int getValue5() const { return value5; }

宏实现

  • 这里使用宏 PUBLIC_GETTER_PRIVATE_SETTER 来定义一个带有公共获取器和私有设置器的成员变量。
cpp 复制代码
#define PUBLIC_GETTER_PRIVATE_SETTER(memberType, memberName) \
private: \
    memberType memberName; \
public: \
    memberType get##memberName() const { return memberName; } \
    //void set##memberName(memberType val) { memberName = val; }
    
#include <iostream>
class Context {
public:
    Context() :value1(10) {}

	PUBLIC_GETTER_PRIVATE_SETTER(int, value1) ;
};

int main() {
    Context context;
    std::cout << context.getvalue1() << std::endl;
    return 0;
}
  • 指针测试,我们替换为其他类
cpp 复制代码
#define PUBLIC_GETTER_PRIVATE_SETTER(memberType, memberName) \
private: \
    memberType memberName; \
public: \
    memberType get##memberName() const { return memberName; } \
    //void set##memberName(memberType val) { memberName = val; }
class A {

PUBLIC_GETTER_PRIVATE_SETTER(int, value);
public:
    A() :value(10) {}
};
#include <iostream>
class Context {
public:
    Context() :a(new A()) {}

PUBLIC_GETTER_PRIVATE_SETTER(A*, a) ;
};

int main() {
    Context context;
    std::cout << context.geta()->getvalue()<< std::endl;
    //context.geta()->value = 20;//无法访问
    return 0;
}

小结

  • 题外话:C++适当用宏!适当用宏!适当用宏!
  • 如有错误,欢迎指出!
相关推荐
2301_8079973812 小时前
代码随想录-day54
数据结构·c++·算法
️公子12 小时前
无人直播系统-黑客主题
人工智能·c#·visual studio
curry____30312 小时前
study in pta + 豆包(求区间和)(前缀和算法)(如何处理zhan栈溢出和超出时间复杂度问题)(2025.12.2)
数据结构·c++·算法
BestOrNothing_201512 小时前
【C++基础】Day 6:前置++ VS 后置++(语法底层 + STL规范 + 面试高频)
c++·运算符重载·面试八股·前置++·后置++·stl迭代器
缘三水12 小时前
【C语言】10.操作符详解(下)
c语言·开发语言·c++·语法·基础定义
渡我白衣12 小时前
深入理解算法库的灵魂——彻底掌握 <algorithm> 的范式、迭代器约束、隐藏陷阱与性能真相
数据结构·c++·人工智能·网络协议·mysql·rpc·dubbo
报错小能手12 小时前
C++流类库 文件流操作
开发语言·c++
暗然而日章12 小时前
C++基础:Stanford CS106L学习笔记 3 流
c++·笔记·学习
獭.獭.12 小时前
C++ -- STL【list的使用】
c++·stl·list
Q741_14712 小时前
C++ 栈 模拟 1047. 删除字符串中的所有相邻重复项 题解 每日一题
c++·算法·leetcode·模拟·