第7周-任务3-复数模板类

简介: 【预备】阅读P314的例10.1。 #include <iostream>using namespace std;class Complex {public: Complex( ){real=0;imag=0;} Complex(double r,double i){real=r;imag=i;} Complex complex_add(Co

【预备】阅读P314的例10.1。

#include <iostream>
using namespace std;
class Complex   
{
public:
	Complex( ){real=0;imag=0;}     
	Complex(double r,double i){real=r;imag=i;} 
	Complex complex_add(Complex &c2); 
	void display( );   
private:
	double real; 
	double imag; 
};
Complex Complex::complex_add(Complex &c2)
{
	Complex c;
	c.real=real+c2.real;
	c.imag=imag+c2.imag;
	return c;
}   
void Complex::display( )   
{
	cout<<"("<<real<<","<<imag<<"i)"<<endl;
}
int main( )
{
	Complex c1(3,4),c2(5,-10),c3;  
	c3=c1.complex_add(c2);  
	cout<<"c1="; c1.display( );  
	cout<<"c2="; c2.display( ); 
	cout<<"c1+c2="; c3.display( );  
	return 0;
}

【题目】该例实现了一个复数类,但是美中不足的是,复数类的实部和虚部都固定只能是double型的。可以通过模板类的技术手段,设计Complex,使实部和虚部的类型为定义对象时用的实际类型。

(1)要求类成员函数在类外定义。

(2)在此基础上,再实现减法、乘法和除法

你可以使用的main()函数如下。

int main( )
{   Complex<int>c1(3,4),c2(5,-10),c3; 
    c3=c1.complex_add(c2); 
    cout<<"c1+c2=";
    c3.display( ); 
    Complex<double>c4(3.1,4.4),c5(5.34,-10.21),c6; 
    c6=c4.complex_add(c5); 
    cout<<"c4+c5=";
    c6.display( );
    system("pause");
    return 0;
}


【参考解答】复数运算规则见《 复数的运算规则》。

#include <iostream>
using namespace std;
template<class T>  //类声明前加模板的声明
class Complex  
{
public:
    Complex( ){real=0;imag=0;}    
    Complex(T r,T i){real=r;imag=i;}   //类声明中的每一个T,将被对象定义时提供的实际类型代替
    Complex complex_add(Complex &c2);
    Complex complex_minus(Complex &c2);
    Complex complex_multiply(Complex &c2);
    Complex complex_divide(Complex &c2);
    void display( );  
private:
    T real;   //数据成员的类型,也将被对象定义时提供的实际类型代替
    T imag;
};
 
//复数相加: (a+bi)+(c+di)=(a+c)+(b+d)i.
template<class T>  //每一个成员函数的定义前,必须要声明类模板
Complex<T> Complex<T>::complex_add(Complex<T> &c2)  //使用了模板的类,将不再独立使用,其类名的完整表示为“类模板名<虚拟类型参数>”
{
    Complex<T> c;   //凡用到类名处也用“类模板名<虚拟类型参数>”形式;本题中求两个复数的和,自然要产生一个新的复数对象
    c.real=real+c2.real;
    c.imag=imag+c2.imag;
    return c;
}


//复数相减:(a+bi)-(c+di)=(a-c)+(b-d)i.
template <class T>    
Complex<T> Complex<T>::complex_minus(Complex <T> &c2)    
{    
    Complex <T> c;    
    c.real=real-c2.real;    
    c.imag=imag-c2.imag;    
    return c;    
}    
  
//复数相乘:(a+bi)(c+di)=(ac-bd)+(bc+ad)i.
template <class T>    
Complex<T> Complex<T>::complex_multiply(Complex <T> &c2)     
{    
    Complex <T> c;    
    c.real=real*c2.real-imag*c2.imag;    
    c.imag=imag*c2.real+real*c2.imag;    
    return c;    
}    
  
//复数相除:(a+bi)/(c+di)=(ac+bd)/(c^2+d^2) +(bc-ad)/(c^2+d^2)i
template <class T>    
Complex<T> Complex<T>::complex_divide(Complex <T> &c2)     
{    
    Complex <T> c;    
    T d=c2.real*c2.real+c2.imag*c2.imag;
    c.real=(real*c2.real+imag*c2.imag)/d;     //此处有危险未排除:除法溢出
    c.imag=(imag*c2.real-real*c2.imag)/d;    
    return c;    
} 
 
template<class T>
void Complex<T>::display( )  
{
    cout<<"("<<real<<","<<imag<<"i)"<<endl;
}
 
int main( )
{
    Complex<int> c1(3,4),c2(5,-10),c3; //定义对象时,用“类模板名<实际类型名>”形式
    c3=c1.complex_add(c2); 
    cout<<"c1+c2=";
    c3.display( ); 
    Complex<double> c4(3.1,4.4),c5(5.34,-10.21),c6; //定义对象时,用“类模板名<实际类型名>”形式
    c6=c4.complex_add(c5); 
    cout<<"c4+c5=";
    c6.display( );


    c6=c4.complex_minus(c5); 
    cout<<"c4-c5=";
    c6.display( );


    >c6=c4.complex_multiply(c5); 
    cout<<"c4*c5=";
    c6.display( );


    c6=c4.complex_divide(c5); 
    cout<<"c4/c5=";
    c6.display( );


    system("pause");
    return 0;
}

【拓展】模板类中的友元函数

友元函数提供了一种非成员函数访问私有数据成员的途径。模板类中能不能使用友元函数呢?当然可以,从下面的解法中品味一二。

#include <iostream>
using namespace std;

template<class numtype>
class Complex   
{
public:
	Complex( ){real=0;imag=0;}     
	Complex(numtype r,numtype i){real=r;imag=i;} 
	Complex complex_add(const Complex &c2); 
	template<class numtype> friend Complex<numtype> add_complex(const Complex<numtype> &c1, const Complex<numtype> &c2);   //利用了模板的外部函数要作为友元函数,注意声明方式:类声明中也必须给出模板声明
	void display( );   
private:
	numtype real; 
	numtype imag; 
};

template<class numtype>
Complex<numtype> Complex<numtype>::complex_add(const Complex<numtype> &c2)
{
	Complex<numtype> c;
	c.real=real+c2.real;
	c.imag=imag+c2.imag;
	return c;
} 

//利用了模板的友元函数的定义
template<class numtype> 
Complex<numtype> add_complex(const Complex<numtype> &c1, const Complex<numtype> &c2) 
{
	Complex<numtype> c; 
	c.real=c1.real+c2.real; 
	c.imag=c1.imag+c2.imag; 
	return c; 
} 

template<class numtype> 
void Complex<numtype>::display( )   
{
	cout<<"("<<real<<","<<imag<<"i)"<<endl;
}

int main( )
{
	Complex<int> c1(3,4),c2(5,-10),c3;  
	c3=c1.complex_add(c2);  
	cout<<"c1+c2="; 
	c3.display( );  
	Complex<double> c4(3.1,4.4),c5(5.34,-10.21),c6;  
	c6=c4.complex_add(c5);  
	cout<<"c4+c5="; 
	c6.display( ); 
	Complex<int> c7=add_complex(c1,c2);  
	cout<<"c1+c2="; 
	c7.display( );
	system("pause");
	return 0;
}

【最后的一点讨论】本例中complex_add和add_complex两个函数中的参数用作为引用类型,这是对象作参数的一般处理方法;参数在函数中不允许改变,我们加上了限定词const。请体会这样处理的好处,并且在以后的工作中主动利用。




目录
相关文章
|
JavaScript 测试技术 索引
js数组方法汇总
js数组方法汇总
439 55
|
Web App开发 前端开发 Windows
CSS基础常用属性之字体属性(如果想知道CSS的字体属性知识点,那么只看这一篇就足够了!)
CSS基础常用属性之字体属性(如果想知道CSS的字体属性知识点,那么只看这一篇就足够了!)
|
数据采集
LabVIEW Actor架构特点与适用范围
LabVIEW Actor架构特点与适用范围
328 1
|
Web App开发 JavaScript 前端开发
Selenium + Chrome带配置项启动
Selenium + Chrome带配置项启动
|
Java 测试技术 开发工具
spring data elasticsearch:动态配置实体类索引名称indexName
最近接到一个需要,需要在spring data elasticsearch关联的实体类中动态的根据配置文件动态创建索引名称,比如开发环境下索引名称为user-dev,测试环境下为user-test,生产环境为user-prod 一开始接到这个需要觉得很怪,因为不同环境的区分直接搭建不同的es服务器环境不就行了吗,为什么要这么麻烦来处理呢?后来了解到是因为当前项目成本受限,需要隔离开发环境和测试环境用的同一台服务器,于是就产生了这个奇怪的需求。 其实想想,不排除某些场景下,就需要动态的创建索引名称。于是今天我们就来看看怎么实现这个需求
894 0
spring data elasticsearch:动态配置实体类索引名称indexName
|
存储 固态存储 大数据
你知道企业级SSD与消费级SSD的区别吗?
【引语】 前两天,一位从事大数据工作的好朋友问我: “企业级SSD与消费级SSD的区别是什么?” “大数据中心能否采用消费级SSD?” 我想,大部分朋友应该都知道SSD分为企业级和消费级,但是这两者之间的区别,也许并不十分清楚。
|
SQL 安全 网络安全
网络安全——SQL注入之安全狗bypass深度剖析
sql注入不会绕过WAF?关注我,让我带你由简入难实战各个WAF,今天先来看看web安全渗透必会的安全狗WAF,你会绕吗?看我带你将它拿下
656 0
网络安全——SQL注入之安全狗bypass深度剖析
|
SQL 安全 Oracle
5 款漏洞扫描工具:实用、强力、全面(含开源)(三)
5 款漏洞扫描工具:实用、强力、全面(含开源)
1812 0
5 款漏洞扫描工具:实用、强力、全面(含开源)(三)
|
存储 监控 算法
2022赛季RoboMaster 空中机器人六轴无人机云台程序框架开源
2022赛季RoboMaster 空中机器人六轴无人机云台程序框架开源
900 0
|
传感器 机器学习/深度学习 编解码
领域最全!多传感器融合方法综述!(Camera/Lidar/Radar等多源异构数据)(下)
自动驾驶正成为影响未来行业的关键技术,传感器是自动驾驶系统中感知外部世界的关键,其协作性能直接决定自动驾驶车辆的安全性。本文主要讨论了近年来自动驾驶中多传感器融合的不同策略。分析了常规传感器的性能和多传感器融合的必要性,包括radar、激光雷达、摄像机、超声波、GPS、IMU和V2X。
领域最全!多传感器融合方法综述!(Camera/Lidar/Radar等多源异构数据)(下)

热门文章

最新文章

下一篇
开通oss服务