C++类成员属性的一种简洁实现

                  C++类成员属性的一种简洁实现

    一般来说对于标准C++而言是不存在成员属性这个概念的,以前大家都是用GetXXX/SetXXX来访问或取得数据,好象也没有感觉到任何不便。但是当我们用过C#之类的语言之后,我们总觉得C++这个方式太老土了。于是我们想去实现“属性”这个C++语言缺乏的要素。事实上网络上有很多人已经做了这部分工作,实现的方法有很多种,一种是用模板,一种是根据特定语言来写的,如VC(指的是Microsoft实现的C++)。但是它们要么很复杂,要么很难记住它的准确用法,嗯我总是喜欢简单的东西,因为太复杂的东东会让我的头脑当机。废话少说,来看看如何实现。
在实现之前,我必需先探讨一下为什么需要“属性”这个东东。比如说下面雇员这个类:

    1. class CEmployee
    2. {
    3. public:
    4.   int Old; //年龄
    5. };
    6. CEmployee employee;
    7. employee.Old=22;
    8. int old =employee.Old;

    它有一个成员变量,我们可以直接对它们进行赋值或者读取,但是往往会缺少一个很重要的东东,就是不能对所赋值进行校验,这可是个大问题,比如我们给Old一个负值,比如-50,程序运行时不会有任何错误,但是的确这个成员变量的值在逻辑上是不正确的。于是我们会写上GetOld、SetOld。现在OK了,这个小问题解决了,但新问题来了。我们的类使用者,他们需要重新把他们的代码成写如下的样子,而不是上面的那样。
    1.   CEmployee employee;
    2.   employee.SetOld(22);
    3.   int old =employee.GetOld();

    你的伙伴一定会在写代码时诅咒你写了一个垃圾的类。所以你决定要改变这个现状。很幸运,你是MS的忠实用户,而且你对于MSDN看很仔细,所以你知道可以这样来写
    1. class CEmployee
    2. {
    3. private:
    4.   int m_old;
    5. public:
    6.  _declspec(property(get= GetOld,put=SetOld))int Old;
    7.  
    8.  int GetOld(void)
    9.  {
    10.   return m_old;
    11.  }
    12.  void SetOld(int value)
    13.  {
    14.   if( (value >0) && (value <60))
    15.   {
    16.   m_old = value;
    17.   }
    18.   else
    19.   {
    20.   m_old =20;
    21.   }
    22.  }
    23. };

    Very Good,上面的类完美地完成一个属性所要做的目标,不过还有一点小问题,象我这样比较笨的经常需要查找MSDN才会知道   
    1. _declspec(property(get= GetOld,put=SetOld))int Old;
    这句话的含义,而且我也经常忘记它的具体写法,比如put我常把它写成了set,这总是让我想起了使用C#的美好时光,它是可以写成这个样子的

    1. public class CEmployee
    2.  {
    3.   private int m_old;
    4.   public int Old
    5.   {
    6.   get
    7.   { return m_old; }
    8.   set
    9.   {
    10.     if(value >0 && value <60)
    11.     {
    12.         m_old = value;
    13.      }
    14.     else
    15.     {
    16.         m_old =20;
    17.      }
    18.   }
    19.   }
    20.  }

所以我想到可以利用C/C++中强大的武器宏,我们来定义几个宏

    1. #define PROP(T,X) __declspec(property(get= __get##X,put= __put##X))T X;
    2. #define GETPROP(T,X) __declspec(property(get= __get##X))T X; //只读属性
    3. #define SETPROP(T,X) __declspec(property(put= __put##X))T X; //只写属必
    4. #define GET(T,X) T __get##X(void) 
    5. #define SET(T,X) void __put##X(T value)

    说明一下:T 代表属性的类型如int,double,CString,而X代表属性名称。如果你需要一个只读属性可以使用GETPROP,只写属性则可以使用SETPORP,然后对应使用一个GET或SET,当然如果你用PROP,而只用了一个GET或SET,也没有错,只是在编译时会告诉你没有一个__getXXX或__putXXX的方法。然后我们就可以这样来写我们的类。

    1. class CEmployee
    2. {
    3. private:
    4.   int m_old;
    5. public:
    6.  PROP(int ,Old)
    7.  GET(int,Old)
    8.  {
    9.   return m_old;
    10.  }
    11.  SET(int,Old)
    12.  {
    13.   if( (value >0) && (value <60)) //这里的value你可把它和C#一样当做关键字
    14.   {
    15.   m_old = value;
    16.   }
    17.   else
    18.   {
    19.   m_old =20;
    20.   }
    21.  }
    22. };

    好了,我们要做的工作已经做完了。当然这种方法还是有很多问题,比如不能使用C#中常用的索引属性,静态属性等等。但是毕竟我们是C++程序员么,呵呵!最后,这种方法只是在VC下有用。


相关推荐
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页