您的当前位置:首页正文

c++矩阵类的实现

2020-06-10 来源:好走旅游网
哈尔滨工业大学(威海)软件学院

C程序设计 实验报告

编号: 姓名 孟令鑫 院系 任课教师 丁建睿 实验地点 研究院中507-508 实验名称 同 组 人 预习报告(对实验主要内容的认识) 学会如何进行基本的运算符重载 了解运算符重载的注意事项 学会两种运算符重载的方式 学会利用运算符重载解决实际问题 矩阵操作 软件学院 学号 131110412 指导教师 实验时间 2014年 4月18日 类的深入剖析 无 得分 实验内容(问题,思路,程序,结果) 得分 实验一——实现矩阵类的操作 1、设计思路:通过类中运算符的重载功能实现对矩阵的运算操作具体如下: (1)运算符重载定义: 可以将已有运算符用于用户自定义数据类型(重载) 何时用:当利用运算符重载比完成相同工作的函数调用使程序更清晰时用 意义:C++通过重新定义运算符,使它能够用于特定类的对象执行特定功能 (2)运算符重载的几个注意事项:运算符必须被显式重载 重载运算符只对用户自定义的类对象起作用,不 影响基本数据类型的操作。 通常分为成员函数重载和友元函数重载 重载方式; (friend) <函数类型> operator<运算符>(<参数表>) { <函数体> } 单目运算符最好重载为类的成员函数; 双目运算符则最好重载为类的友元函数。 特殊的,以下一些双目运算符不能重载为类的友元函数: =、()、[]、-> (3)矩阵类的定义:在private私有成员中定义row和column以及一个一维数组pt存放矩阵的行数、列数以数据成员。 (4)+、-重载:首先定义一个中间类的对象(不分配内存),再进行两个矩阵是否能进行相加(减)的判断,既行数和列数是否相等,如果能进行相加(减)的运算则分配内存,利用for循环,将两个矩阵中对应位置上的元素相加(减)得到的结果存到中间类中,然后作为返回值返回。如果不能进行相加(减)的,输出提示语句:这两个矩阵不能相加或者相减,然后返回中间类对象,由于此时的中间类的对象没有分配内存,调用的时候是使用默认构造函数进行构造的,所以为0行0列的空矩阵。 (5)*重载:首先定义一个中间类的对象(不分配内存),再进行两个矩阵是否能进行相乘的判断,既第一矩阵的列数和第二个矩阵的行数是否相等,如果能进行相加(减)的运算则分配内存,分别定义两个用来计数行和列的变量以及一个用来定位新产生的矩阵的位置的变量count,先通过外层的for循环对行数进行++操作,内层通过while循环对列进行逐次的++操作,最内层的for循环是将当前所在的第一个矩阵的行元素和第二个矩阵的列元素进行相乘累加的操作,每一次最内层的for循环结束代表着第一个矩阵的第i行和第二个矩阵的第k列的元素逐一相乘累加完毕,此时count++新产生的矩阵进行移位,进入第二个位置。同时进入while循环,k++,当前行再和下一列进行相应的操作。当k循环结束全部的列数之后,返回最外层的for循环,行数+1,再重复上面的操作,进行相乘并累加求和。此时要格外注意的一点是,由于使用的一维数组,则在行数进行++操作的时候。相应的第一个矩阵的元素要跳过column个元素,对应代码中的j=i*column一句话,然后再依次向后移位,与此对应的右侧数组则是每次跳过列数个元素(每行的元素个数)对应temp1 += m.column一句。将数据存入对应位置,最后返回中间矩阵对象。如果不能进行相乘的,输出提示语句:这两个矩阵不能相加乘,然后返回中间类对象,由于此时的中间类的对象没有分配内存,调用的时候是使用默认构造函数进行构造的,所以为0行0列的空矩阵。 (6)<<重载:利用for循环输出每一元素,如果该元素位置%列数=0说明已经循环输出完一行则输出换行符,否则继续输出。返回地址。 (7)>>重载:此时定义一个全局的static count1变量。每一次使用重载的>>运算符则说明输入了一个矩阵,则count1++;先让用户输入行数和列数,分别赋值给右操作数的成员,将原有的数组释放,并为他重新申请内存,将用户输入的数据循环in到该对象中。返回地址。 (8)=重载:对应位置上的数据进行赋值。 (9)+=重载:类似于+重载,特别的就是将左右操作数进行加的操作之后付给左操作数。 (10)==和!=重载:由于返回值为真假,所以要定义为bool类型,先判断行和列是否相等,如果行和列都相等了,在循环比较对应位置上的数值,如果对应位置上的数值都相等则返回真否则返回假;如果两个矩阵的行或者列不相等则直接返回假 (11)Setcount函数,当用户要重复进行使用的时候,将计数器count1置零。 2、代码设置:《main函数》 #include #include \"Matrix.h\" using namespace std; int main() { system(\"COLOR 0a\"); char choice; do { cout<<\"****************************************************\"<> first; cout<<\"=====================================================\"<> second; cout<<\"=====================================================\"<>choice; }while(choice=='y'); return 0; } 《Matrix.cpp》 #include\"Matrix.h\" #include static int count1=0;//每调用一次构造函数计数器++ using namespace std; Matrix::Matrix()//默认构造函数 ,初始化行数列数为0.数组指针为NULL { row = 0; column = 0; pt = NULL; //count++; } Matrix::Matrix(int r, int c, double* p):row(r), column(c)// 带参构造函数,利用基类和派生类实现初始化 { pt = new double[r*c];//为矩阵申请(rXc)大小的内存空间。 for(int i=0; i>(istream& in, Matrix&m)//流输入运算符重载 { int Row,Column; count1++; cout<<\"请输入矩阵 M\"<>Row>>Column;//输入行数和列数 m.row = Row;//分别赋值给右操作数的成员 m.column = Column; delete [] m.pt;//将原有的数组释放 m.pt = new double[Row * Column];//重新申请内存 cout<<\"请按行输入\"<>m.pt[i]; return in;//返回地址 } Matrix& Matrix::operator=(const Matrix& m)//赋值运算符重载 { row = m.row; column = m.column; pt = new double[row*column]; for(int i=0; i #include #include #include #include #include #ifndef _MATRIX_H #define _MATRIX_H #include using namespace std; class Matrix { public: //构造函数和析构函数 Matrix(); Matrix(int, int, double*); //矩阵行数、列数、数组指针 Matrix(const Matrix&); //拷贝构造函数 ~Matrix();//析构函数 //运算符重载函数以及友元函数 Matrix operator+(const Matrix&); Matrix operator-(const Matrix&); Matrix operator*(const Matrix&); Matrix& operator=(const Matrix&); Matrix& operator+=(const Matrix&); friend ostream& operator<<(ostream&, const Matrix&); friend istream& operator>>(istream&, Matrix&); bool operator ==( Matrix& )const; bool operator !=( Matrix& right) const { return !(*this == right); } //其他操作函数 //int getcount(); void setcount(); int getrow(); int getcolumn(); private: int row; int column; double* pt; }; #endif // MATRIX_H_INCLUDED 3、遇到问题和解决方法:(1)==和!=返回值问题,由于这两个是判断是否为真或者为假,则定义的返回值类型应该为bool类型(2)count1计数器的问题,原来代码中我将count1计数器写在了构造函数中,每次调用构造函数的时候计数器++,这就导致了代码的复杂程度,这样使用中间对象的时候,就必须使用new的动态内存分配,使用之后必须马上delete释放,调用构造函数将count1--。经过讨论,比较方便的方法就是由于重载了>>流插入运算符,由于重载的运算符只能针对类的对象使用,如果重载的>>被调用了就证明新输入了一个矩阵此时直接++计数器就可以了(3)原来的代码运行的时候出现这样一个问题,当输入的矩阵不能相加的时候,就直接退出了程序导致剩下的部分不能执行。原来代码如下: if(row != m.row || column != m.column) //判断两个矩阵是否是同类型的矩阵(通过判断行数和列数是否相等) { cout<<\"这两个矩阵不是同类型矩阵不能相减!\"<

因篇幅问题不能全部显示,请点此查看更多更全内容