记得上学时,阅读过这本经典之作。当时没有实际项目经验,全是为了看书而看书,读完后,一本经典之作竟然一点触动都没有。
这次偶尔翻到此书,闲暇时间,再次拜读几个章节,影响极深,对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);
}
本文回顾了经典著作《C++沉思录》的第10章,作者通过实例展示了面向对象编程的精髓,如动态绑定、类层次结构和引用计数等。代码示例在VC++6.0环境中成功运行,包含Picture_ad类的实现及其派生类,如String_Pic、Frame_Pic、Vcat_Pic和Hcat_Pic,提供了不同类型的图片操作。

408

被折叠的 条评论
为什么被折叠?



