C++ 沉思录源代码

本文回顾了经典著作《C++沉思录》的第10章,作者通过实例展示了面向对象编程的精髓,如动态绑定、类层次结构和引用计数等。代码示例在VC++6.0环境中成功运行,包含Picture_ad类的实现及其派生类,如String_Pic、Frame_Pic、Vcat_Pic和Hcat_Pic,提供了不同类型的图片操作。

         记得上学时,阅读过这本经典之作。当时没有实际项目经验,全是为了看书而看书,读完后,一本经典之作竟然一点触动都没有。

        这次偶尔翻到此书,闲暇时间,再次拜读几个章节,影响极深,对C++以及面向对象又有了新的认识,打动了本来已经沉寂好久的求学欲望。回想这几年工作,全部精力都投入到了项目中。很多时候都处于赶进度的状态,为了快速得到结果,忽略了详细的斟酌过程。结果还是代码出来后修修改改,没有采用合理的模式和应该的设计思想。

   以下是本人今天下午调试出来的第10章源代码,在VC++6.0中调试成功,本打算上传源代码,但是第一次写BLOG,不知怎么上传,暂时全拷贝上。

 

    本章本书作者让我们充分认识到面向对象的伟大和其运用技巧,C++中的动态绑定被充分运用。仔细参与编程,可以充分认识到其实现细节,值得玩味。

main函数:

void main()
{
 printf("hello world!/r/n");

/*
 Expr t = Expr((string)"*",Expr((string)"-",(Expr)5),Expr((string)"+",(Expr)3,(Expr)4));
 cout << t << endl;
 //(-5)*(3+4)
 cout<<t.eval()<<endl;

 t = Expr((string)"*",t,t);
 cout << t << endl;
 //(-5)*(3+4)*(-5)*(3+4)
 cout<<t.eval()<<endl;
*/

 char *str[] ={"Hello", "World", "Thinks"};
 char *warn[] = {"Yulong", "Conmunication", "Limited", "Corporation"};
// Picture p(str,3);
// cout<<p<<endl;
// 
// Picture *q = frame(p);
// cout<<(*q)<<endl;
//  delete q; 

 Picture_ad p(str,3);
 Picture_ad q(warn,4);
 cout<<p<<endl;

 Picture_ad r = frame(p);
 cout<<r<<endl;

 Picture_ad s = frame(q);
 cout<<s<<endl;

 Picture_ad t = r | s;
 cout<<t<<endl;

 Picture_ad o = frame(t);
 cout<<o<<endl;

 Picture_ad m = o & r;
 cout<<m<<endl;

 Picture_ad n = frame(m);
 cout<<n<<endl;

}

 

头文件:

/**********************************************
 *C++沉思录 Chapter 10
 *Creator: 
 *Date:    2008-10-30

 *Description:
 *1.利用句柄类隐藏细节
 *2.类层次保存结构信息
 *3.内存分配策略采用引用计数
 *********************************************/

#include <iostream>
using std::ostream;
using std::endl;


#define max(a,b) ((a>b)?a:b)

static void pad(ostream&, int, int);
class P_Node;

class Picture_ad
{
 friend ostream& operator<<(ostream&, const Picture_ad &);
 
 friend class String_Pic;
 friend class Frame_Pic;
 friend class Vcat_Pic;
 friend class Hcat_Pic;
 
public:
 Picture_ad();
 Picture_ad(const char* const*,int);
 Picture_ad(const Picture_ad&);
 Picture_ad(P_Node *);

 ~Picture_ad();
 
 Picture_ad& operator=(const Picture_ad&);
 
 
private:
 int height() const;
 int width() const;
 
 //单行显示策略
 void display(ostream&, int, int) const;

 P_Node *p;
};

class P_Node
{
 friend class Picture_ad;
protected:
 P_Node():use(1){};
 virtual ~P_Node();
 virtual int height() const = 0;
 virtual int width() const = 0;
 virtual void display(ostream&, int, int) const = 0;

public:
 int use;
 
};

class String_Pic:P_Node
{
public:
 String_Pic(const char* const*, int);
 ~String_Pic();

 int height() const;
 int width() const;
 void display(ostream&, int, int) const;

private:
 char **chdata;
 int w;
 int h;
};

class Frame_Pic:P_Node
{
 friend Picture_ad frame(const Picture_ad&);
public:
 Frame_Pic();
 Frame_Pic(const Picture_ad&);
 int height() const;
 int width() const;
 void display(ostream&, int, int) const;
 
private:
 Picture_ad p;
 
};

class Vcat_Pic:P_Node
{
 friend Picture_ad operator&(const Picture_ad&, const Picture_ad&);
 Vcat_Pic (const Picture_ad&,const Picture_ad&);

 int height() const;
 int width() const;
 void display(ostream&, int, int) const;

 Picture_ad top,bottom;
};


class Hcat_Pic:P_Node
{
 friend Picture_ad operator|(const Picture_ad&, const Picture_ad&);
 Hcat_Pic (const Picture_ad&,const Picture_ad&);

 int height() const;
 int width() const;
 void display(ostream&, int, int) const;

 Picture_ad right,left;
};

 

//CPP文件

#include "picture_ad.h"

static void pad(ostream& os, int x, int y)
{
 for(int i=x; i<y; i++)
 {
  os<<" ";
 }
}

ostream& operator<<( ostream& os, const Picture_ad& p)
{
 for(int i =0; i<p.height(); i++)
 {
  p.display(os,i,p.width());
  os<<endl;
 }
 return os;
}

/***********************************************************
 *Piture_ad class 
 ***********************************************************/
Picture_ad::Picture_ad(const char* const* str, int n):p(new String_Pic(str,n))
{
 //p指向一个 String_Pic*
}

Picture_ad::Picture_ad(const Picture_ad& orig)
{
 p = orig.p;
 p->use++;//引用增加
}

Picture_ad& Picture_ad::operator=(const Picture_ad& orig)
{
 orig.p->use++;

 if(--p->use == 0)
 {
  delete p;
 }

 p = orig.p;

 return *this;
}

Picture_ad::Picture_ad(P_Node* porig):p(porig)
{

}

Picture_ad::~Picture_ad()
{
 if(--p->use == 0)
 {
  delete p;
 }
 
}


Picture_ad::height() const
{
 return p->height();
}

Picture_ad::width() const
{
 return p->width();
}

//具体的显示策略封装在各自对象中
void Picture_ad::display(ostream& os, int row, int col) const
{
 p->display(os, row, col);
}


/************************************************************************/
/*     P_Node class                                                     */
/************************************************************************/
P_Node::~P_Node()
{
 
}

/************************************************************************/
/*         String_Pic class                                             */
/************************************************************************/

/********************************************************************
 *name:
 *pram:
 *retu:null
 *func:
 *desc:为成员变量赋值
 *******************************************************************/
String_Pic::String_Pic(const char* const* str, int n)
{
 int i=0;
 int m=0,j=0;

 w=0;
 for(i=0; i<n; i++)
 {
  w = max(w,strlen(str[i]));
 }
 h = n;

 chdata = new char*[n];
 for( i =0; i<n; i++)
 {
  chdata[i] = new char(strlen(str[i]));
  strcpy(chdata[i], str[i]);
 }

}


String_Pic::~String_Pic()
{
 delete[] chdata;
}

int String_Pic::height() const
{
 return h;
}

int String_Pic::width() const
{
 return w;
}

void String_Pic::display(ostream& os, int row, int col) const
{
 int start = 0;

 if(row>= 0 && row< height())
 {
  os<<chdata[row];
  start = strlen(chdata[row]);
 }
 pad(os, start, col);
}


/************************************************************************/
/*   Frame_Pic class                                                    */
/************************************************************************/
Frame_Pic::Frame_Pic(const Picture_ad& orig):p(orig)
{

}

int Frame_Pic::width() const
{
 return p.width()+2;
}

int Frame_Pic::height() const
{
 return p.height()+2;
}

void Frame_Pic::display(ostream& os, int row, int wd) const
{
 if(row < 0 || row >= height())
 {
  pad(os, 0, wd);
 }
 else
 {
  if(row == 0|| row == height()-1)
  {
   os<<"+";
   int i = p.width();
   while(--i >= 0)
   {
    os<<"-";
   }
   os<<"+";
  }
  else
  {
   os<<"|";
   p.display(os,row-1,p.width());
   os<<"|";
  }
  pad(os,width(),wd);
 }
}

Picture_ad frame(const Picture_ad& orig)
{
 return new Frame_Pic(orig);//需要添加Picture_ad(P_Node* porig)进行转化 
}


/************************************************************************/
/*           Vcat_Pic class                                             */
/************************************************************************/
Vcat_Pic::Vcat_Pic(const Picture_ad  &tp, const Picture_ad &bm):top(tp),bottom(bm)
{

}

int Vcat_Pic::width() const
{
 return max(top.width(), bottom.width());
}

int Vcat_Pic::height() const
{
 return top.height()+bottom.height();
}

void Vcat_Pic::display(ostream& os, int row, int wd) const
{
 if(row>=0 &&row <top.height())
 {
  top.display(os,row,wd);
 }
 else if(row<top.height()+bottom.height())
 {
  bottom.display(os,row-top.height(),wd);
 }
 else
  pad(os,0,wd);
}

Picture_ad operator&(const Picture_ad& t, const Picture_ad& b)
{
 return new Vcat_Pic(t,b);
}


/************************************************************************/
/*         Hcat_Pic class                                               */
/************************************************************************/
Hcat_Pic::Hcat_Pic(const Picture_ad  &r, const Picture_ad &l):right(r),left(l)
{
 
}

int Hcat_Pic::width() const
{
 return left.width()+right.width();
}

int Hcat_Pic::height() const
{
 return max(left.height(),right.height());
}

void Hcat_Pic::display(ostream& os, int row, int wd) const
{
 left.display(os,row,left.width());
 right.display(os,row,right.width());
 pad(os, width(),wd);
}

Picture_ad operator|(const Picture_ad& r, const Picture_ad& l)
{
 return new Hcat_Pic(r,l);
}

 


 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值