免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2054 | 回复: 6
打印 上一主题 下一主题

问一个重载的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-09-28 15:50 |只看该作者 |倒序浏览
对于函数的重载能够理解,但是关于运算符的重载却一直不能够很好的明白。  
  我想知道为什么要重载运算符;重载后的运算符和没有重载的有什么区别;什么时候对运算符重载要合适些等。希望能够得到大家详细而具体的解答

论坛徽章:
0
2 [报告]
发表于 2007-09-28 16:10 |只看该作者
重载运算符后能支持自定义类型的运算操作。

论坛徽章:
0
3 [报告]
发表于 2007-09-28 16:11 |只看该作者
其实关于运算符为什么要重载以及如何重载,在很多入门级的书都有介绍的,楼主认真看一下书撒。

论坛徽章:
0
4 [报告]
发表于 2007-09-28 16:45 |只看该作者
推荐你看postgresql关于运算符重载的帮助文档. postgresql的运算符重载很猛的, 对你帮助也很大. 再看看运算符的选择算法. :wink:

其实你考虑下datetime - interval = datetime
datetime - datetime = interval
datetime + interval = datetime
datetime + datetime = ??无意义

可以理解运算符重载的思想了嘛?

论坛徽章:
0
5 [报告]
发表于 2007-09-28 17:16 |只看该作者
类.可以运算吗?类不是值当然不行..那类又想运算怎么办呢?

论坛徽章:
0
6 [报告]
发表于 2007-09-28 17:18 |只看该作者

回复 #5 依赛特小子 的帖子

明白了,谢谢

论坛徽章:
0
7 [报告]
发表于 2007-09-28 17:19 |只看该作者
找到一个比较详细的说法:

构造函数、析构函数与赋值函数是每个类最基本的函数。每个类只有一个析构函数,但可以有多个构造函数(包含一个拷贝构造函数,其它的称为普通构造函数)和多个赋值函数(除了同类的赋值以外,还有其他的赋值方法)。对于任意一个类A,如果不想编写上述函数,C++编译器将自动为A产生四个缺省的函数,如
A(void);                    // 缺省的无参数构造函数
A(const A &a);              // 缺省的拷贝构造函数
~A(void);                   // 缺省的析构函数
A & operate =(const A &a);  // 缺省的赋值函数

有几个需要注意的内容:
@ 构造函数与析构函数的另一个特别之处是没有返回值类型
@ 构造从类层次的最顶层的基类开始,在每一层中,首先调用基类的构造函数,然后调用成员对象的构造函数。析构则严格按照与构造相反的次序执行,在析构的时候,最低层的派生类的析构函数最开始被调用,然后调用每个基类的析构函数。
@ “缺省的拷贝构造函数”和“缺省的赋值函数”均采用“位拷贝”而非“值拷贝”的方式来实现,倘若类中含有指针变量,这两个函数注定将出错

下面通过例子进一步说明,


1.构造函数的初始化表
设存在两个类:
class A
{
    …
    A(void);                // 无参数构造函数
    A(const A &other);      // 拷贝构造函数
    A & operate =( const A &other);  // 赋值函数
    virtual ~A(void);        //析构函数
};
class B
{
public:
    B(const A &a);    // B的构造函数

private:   
    A  m_a;            // 成员对象
};


下面面是B的构造函数的2个实现,其中第一个的类B的构造函数在其初始化表里调用了类A的拷贝构造函数,从而将成员对象m_a初始化;而第二个的B的构造函数在函数体内用赋值的方式将成员对象m_a初始化。我们看到的只是一条赋值语句,但实际上B的构造函数干了两件事:先暗地里创建m_a对象(调用了A的无参数构造函数),再调用类A的赋值函数,将参数a赋给m_a。
B::B(const A &a)
: m_a(a)
{
   …
}

B::B(const A &a)
{
    m_a = a;
    …
}



2.拷贝函数和构造函数的区别
拷贝构造函数是在对象被创建时调用的,而赋值函数只能被已经存在了的对象调用。
String  a(“hello”);
String  b(“world”);
String  c = a;  // 调用了拷贝构造函数,最好写成 c(a);
c = b;             // 调用了赋值函数
本例中第三个语句的风格较差,宜改写成String c(a) 以区别于第四个语句。

如果我们实在不想编写拷贝构造函数和赋值函数,又不允许别人使用编译器生成的缺省函数,可以将拷贝构造函数和赋值函数声明为私有函数,不用编写代码。


3.析构函数与虚析构函数
基类的构造函数、析构函数、赋值函数都不能被派生类继承。如果类之间存在继承关系,在编写上述基本函数时应注意以下事项:
@ 派生类的构造函数应在其初始化表里调用基类的构造函数
@ 基类与派生类的析构函数应该为虚(即加virtual关键字)
#include <iostream>
class Base
{
public:
    virtual ~Base() { cout<< "~Base" << endl ; }
};
class Derived : public Base
{
public:
    virtual ~Derived() { cout<< "~Derived" << endl ; }
};

void main(void)
{
    Base * pB = new Derived;  // upcast
   delete pB;
}


输出结果为:
       ~Derived
       ~Base
如果析构函数不为虚,那么输出结果为
       ~Base
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP