C++类的构建--complex(复数)类的设计过程与思考总结

项目

1.设计complex(复数类)

​ 需求:1)实部,虚部re,im

​ 2)重写+=符号,调用friend_doapl函数,对两个复数进行相加 其中+=函数调用一个全局函数_doapl(complex*,complext&)进行处理。这里只用完成两个复数的相加

​ 3)重写+符号,这里只用完成两个复数,复数和实数的相加

设计完成后的代码

#ifndef __MYCOMPLEX__
#define __MYCOMPLEX__

class complex;
complex&
__doapl(complex* ths, const complex& r);
complex&

__doami(complex* ths, const complex& r);
complex&
__doaml(complex* ths, const complex& r);


class complex
{
public:
	complex(double r = 0, double i = 0) : re(r), im(i) { }
	complex& operator += (const complex&);
	double real() const { return re; }
	double imag() const { return im; }
private:
	double re, im;

	friend complex& __doapl(complex*, const complex&);

};


inline complex&
__doapl(complex* ths, const complex& r)
{
	ths->re += r.re;
	ths->im += r.im;
	return *ths;
}

inline complex&
complex::operator += (const complex& r)
{
	return __doapl(this, r);
}


inline double
imag(const complex& x)
{
	return x.imag();
}

inline double
real(const complex& x)
{
	return x.real();
}

inline complex
operator + (const complex& x, const complex& y)
{
	return complex(real(x) + real(y), imag(x) + imag(y));
}

inline complex
operator + (const complex& x, double y)
{
	return complex(real(x) + y, imag(x));
}

inline complex
operator + (double x, const complex& y)
{
	return complex(x + real(y), imag(y));
}

endif //MYCOMPLEX

设计:

1)确定成员变量以及get方法 思考函数const,real()和img需要加,因为不需要修改值。

2)确定重载符号operator +=方法。

complex& operator +=(const complex&);

​ 思考返回值是什么类型。我们在复数类计算时,经常 x+=y,等价于x = x+y. 因此,确定返回的是原来的x的值,也就是this。我们可以得出,直接传回引用,因为我们传回去得并不是local variable。但为什么我们不设计是void类型呢?因为当我们计算x+=y的时候,我们可以直接通过引用改变x的值呀?

就比如:

complex& operator +=(const complex& c){
	this.re = this.re+c.real();
}

原因就在于,在c++中可以连续复制,如 x1+=x2+=x3; 当发生这种情况的时候,引用必须返回以供作为右值以供下一次的运算。

思考形式参数。能用引用得就用引用。且原值能不能修改呢? x = x+y this指的是x的数据,形式参数指的是y的数据。x的数据发生了改变,但是y的没有。所以我们得出,形式参数要加上const。

3)全局函数_doapl(complex&,complext&)

​ 可能是因为其他地方也要用到,所以就把它写成了全局函数而不是成员函数。为了实现 operator+=调用它,我们得将它声明为一个友元。

​ 其成员变量两个一个是complex*而非complex&的原因是,要传入this,与operator+=发生联动。

4)设计 operator+重载

​ 思考返回值类型,假设有 x3 = x1+x2。x3必然是在函数内新建一个变量并返回,所以不能返回一个局部变量的引用。只能使用值传递。

​ 此外,我们可以注意到,operator+定义在了类外,因为它不需要this这个变量,且operator+二目运算符只允许有两个参数,要想不加入this(类内函数默认添加this形式参数),只能写在类外作为成员函数。

​ 那还有几个问题,复数相加有很多种可能,以下是应该考虑的可能性,我们必须根据这些可能性设计函数的重载:

/**
有三种情况
1.复数加复数
2.复数+double
3.double+复数
**/
inline complex
operator + (const complex& x, const complex& y)
{
  return complex (real (x) + real (y), imag (x) + imag (y));
}

inline complex
operator + (const complex& x, double y)
{
  return complex (real (x) + y, imag (x));
}

inline complex
operator + (double x, const complex& y)
{
  return complex (x + real (y), imag (y));
}
从complex得到的经验

1)若方法不需要修改数据,一定加上const,因为const对象只能调用const方法。若方法不加const,const对象功能不全。

  1. 形式参数都应该传递引用,如果不需要修改,则一定加上const,否则原值可能被修改。若直接传递了值,则发生复制,会很慢。当然,char这种一个字节的参数可以传值。

3)返回值也尽量用引用传递,快。但是一定不能返回局部变量的引用,因为局部变量存在栈中,方法执行完之后内存就被回收了。

4)简单的函数要加inline即使编译器会自动选择。

5).h文件的划分

其中,类声明里的函数不用写具体的形式参数,只用写形式参数的类型即可。简单的函数可以在类声明部分就定义如 real()和Img()。复杂的应该在函数体外进行定义。最后,类内的函数均会自动加上inline,在类外定义的函数若比较简单,应该手动建议编译器使用inlien.

6)操作符重载时,若是双目运算符 如 + 则应该定义为全局函数。

如z = x+y。使用+号时,不需要this的值。

与之相对的是单目运算符,如==或+=

如判断 z==y 的时候,我们需要用到z的值,所以必须定义在类内,所以就没有意义了

但简单的如real()可以写完函数体。复杂的函数应该在类定义区域完成函数体的编写。

posted @ 2020-12-10 17:37  lsxkugou  阅读(908)  评论(0编辑  收藏  举报