首页 > 代码库 > C++中的属性

C++中的属性

C++中的属性

首先我们来看一下C#中的属性

c#中的字段通常是 private,内部使用。
属性是 public 或 protected,对外公开,属性通过 get set 访问器对字段提供安全、有效范围等保护。
讲概念可能不好讲,下面来看一个例子吧!(最简单的)
class Program
{
private int field1; //这个field1就是字段
public int Field1 //这个Fileds1就是属性
{
get{return field1;}
set{field1=value;}
}
}//因为字段的访问修饰符为private外部不能访问它,但可以定义属性为public来访问。

 

http://blog.csdn.net/xieqidong/article/details/3265199
 C#中有属性(Property),而C++中却没有,其实属性对于C++而言也是非常重要的,请往下看。


 什么是属性

 属性就像那些可以存储数据的变量,但在从它们中读写数据时会引发事件,换句话来说,属性就是一个有互作用的变量,自我更新,并在读写其自身时产生不同的值。
 使用像C#这样包含属性的语言非常容易编写一个类,但对C++来说似乎就不太可能了,因为C++的编译器不支持C#那种形式的属性。正因为此,本文就是要介绍如何编写带有像C#中属性的C++类。


 为什么属性很重要

 如果你需要编写一个代表人物的对象,这个对象可能包含以下数据:全名、年龄、出生年月、性别。如果用C++来编写,代码如下:


class Person {
public:
   Person( ){}
   virtual ~Person( ){}

private:    //数据成员

   char m_fName[20];
   char m_lName[20];

   UINT m_YearOfBirth;
   bool m_bGender;
};


 注意:在大多数情况中,不能为了可直接使用,就定义一个数据成员为public,因为数据成员应由对象内实现的业务逻辑来维护。

 如果需要设置或读取m_bGender的值,就可能要实现以下的方法:


class Person {
public:
   Person( ){}
   virtual ~Person( ){}

   void SetGender(bool bGender) {m_bGender = bGender;}
   bool GetGender() {return m_bGender;}

private:    //数据成员

   char m_fName[20];
   char m_lName[20];

   UINT m_YearOfBirth;
   bool m_bGender;
};


 这种方法的弊端在于必须要事先知道其名称与用途,要是使用属性,就简单多了,所需了解的只是属性名而已,且单个属性能支持不同的数据类型,也就是说,在上例中,可让Gender接受字符串或布尔类型,如下所示:


Person.Gender = "Male";

或:

Person.Gender = true;


 属性声明

 现在,来看如何编写属性,就从Gender属性开始吧:


class Person {
public:
   Person( ){}
   virtual ~Person( ){}

   Begin_Property(char*,Gender)
      __get(char*,Gender)
      _set(char*);

      _get(bool);
      _set(bool);

      __release(Gender)
   End_Property(Gender)

private:    //数据成员

   char m_fName[20];
   char m_lName[20];

   UINT m_YearOfBirth;
   bool m_bGender;
};


 代码中使用Begin_Property宏开始定义一个属性,它接受两个参数:属性数据类型及属性名。因为Gender属性是一个字符串属性,所以它应为char *。之后,还需要声明事件get()与set(),用于引发事件,如下所示:


// 这会引发 _set(bool) 事件
Person.Gender = true;
// 这会引发 _get(bool) 事件
bool gender = Person.Gender ;
// 这会引发 _set(char*) 事件
Person.Gender = "Male";
// 这会引发 _get(char*) 事件
printf("Gender :%s/n",(char*)Person.Gender);


 _get与_set是两个宏,分别接受一个参数,而这个参数代表了属性可接受的数据类型,你可看到_get与_set事件的数据类型独立于属性的数据类型,换句话来说,尽管Gender属性的数据类型为char *,但它也可接受布尔类型。
 最后两个宏为:_release,其释放它分配的内存;End_Property,其结束属性声明,两个宏都接受属性名为参数。


 属性的实现

 声明属性后,需要实现set()与get()了,请看如下的代码:


……
Begin_Property(char*,Gender)
   __get(char*,Gender)
   _set(char*)
   {
      //某些代码
      return Gender;
   }

   _get(bool)
   {
      //某些代码
      return iValue;
   }
   _set(bool)
   {
      //某些代码
      return iValue;
   }

   __release(Gender)
End_Property(Gender)
……


 在此可用实现宏来实现它,在讲述如何使用这些宏之前,先解释一下两点问题:__get宏是什么及为什么set()要返回一个值。在__get(char *)中,仅是返回一个指针:“return Gender”,它是默认的getter属性,因为“Gender”是一个char *属性,所以应这样使用它:“__get(char *,Gender)”。
 来看第二个问题,为什么set()要像get()那样返回一个值呢?简单来说,在C++中,set()能扮演get()的角色,请看下面的代码:


bool bGender = Person.Gender = true;


 现在,可使用Imp_set及Imp_get两个宏来实现Gender属性了,两个宏都接受三个参数:数据类型、类名及属性名,如下:


Imp_set(char*,Person,Gender)
{
   PROPERTY_PROLOGUE(Person,Gender)

   if (!Gender) Gender = new char[7];

   if (strlen(iValue)<6 )
   {
      int result;
      if ((result=strcmp(iValue,"Male"))==0)
         pThis->m_bGender = true;
      else
      {
         if ((result=strcmp(iValue,"Female"))==0)
         pThis->m_bGender = false;
      }

      if(result==0) strcpy(Gender,iValue);
   }
   return Gender;
}

Imp_set(bool,Person,Gender)
{
   PROPERTY_PROLOGUE(Person,Gender)

   if (!Gender) Gender = new char[7];

   if (pThis->m_bGender = iValue)
      strcpy(Gender,"Male");
   else
      strcpy(Gender,"Female");

   return (bool)iValue;
}

Imp_get(bool,Person,Gender)
{
   PROPERTY_PROLOGUE(Person,Gender)
   return pThis->m_bGender;
}


 因为set()与get()不能直接访问类成员,所以要使用PROPERTY_PROLOGUE宏,它定义了一个指向属性类的指针:“pThis”,如上所示。
 同时,在类的结尾处,需要释放内存及所有分配的资源,这由__release宏完成。__release是默认的释放宏,它的代码如下:


if (Gender)
{
   delete Gender;
   Gender = NULL;
}


 另外,也可使用_release及Imp_release来实现释放事件,如下所示:


Imp_release(Person,Gender)
{
   //在此释放所有分配的资源
}