实验4 类与数组、指针
- Task1~4
浅复制:int a = 10; int b = a; 可用于静态内存的复制。对于简单的类,默认的复制构造函数已经够用了,但当类持有其他资源,如动态分配的内存、指针等,就需要用到深复制。
深复制:分配动态内存,再将原有对象的内存复制过来。原有对象和新对象所持有的动态内存是相互独立的。
- Task 5
vectorInt.hpp
1 #pragma once 2 #include<iostream> 3 #include<cassert> 4 5 using std::cout; 6 using std::cin; 7 8 class vectorInt { 9 public: 10 vectorInt(int n); //构造函数 11 vectorInt(int n,int value); //构造函数 12 vectorInt(const vectorInt& vi); //复制构造函数 13 ~vectorInt(); //析构函数 14 int& at(int index); 15 friend void output(vectorInt& vi); 16 private: 17 int size; 18 int* p; 19 }; 20 21 vectorInt::vectorInt(int n) { 22 p = new int[n]; 23 size = n; 24 cout << "constructor 1 called.\n"; 25 } 26 27 vectorInt::vectorInt(int n, int value) { 28 p = new int[n](); 29 size = n; 30 for (int i = 0; i < size; ++i) { 31 p[i] = value; 32 } 33 cout << "constructor 2 called. \n"; 34 } 35 36 vectorInt::vectorInt(const vectorInt& vi) { 37 size = vi.size; 38 //p = vi.p; //直接将vi的元素p的内存复制过来,应该是浅复制,运行main函数调用析构时出错 39 p = new int[vi.size]; 40 for (int i = 0; i < size; ++i) { 41 p[i] = vi.p[i]; 42 }//给p动态分配了内存空间,再复制内存,应该是深复制,运行正确 43 cout << "copy constructor called.\n"; 44 } 45 46 vectorInt::~vectorInt(){ 47 delete[] p; //要加上[] 48 cout << "destructor called. \n"; 49 } 50 51 int& vectorInt::at(int index) {//返回下标为index的元素引用 52 assert(index >= 0 && index < size); 53 return p[index]; 54 } 55 56 void output(vectorInt& vi) {//输出 57 for (int i = 0; i < vi.size; ++i) { 58 cout << vi.p[i] << ", "; 59 } 60 cout << "\b\b \n"; 61 }
task5.hpp
1 #include <iostream> 2 #include "vectorInt.hpp" 3 4 void test() { 5 using namespace std; 6 7 int n; 8 cin >> n; 9 10 vectorInt x1(n); 11 for (auto i = 0; i < n; ++i) 12 x1.at(i) = i * i; 13 14 output(x1); 15 16 vectorInt x2(n, 42); 17 vectorInt x3(x2); 18 19 output(x2); 20 output(x3); 21 22 x2.at(0) = 77; 23 24 output(x2); 25 output(x3); 26 } 27 28 int main() { 29 test(); 30 }
测试结果:
- Task 6
Matrix.hpp
1 #pragma once 2 3 #include <iostream> 4 #include<cassert> 5 6 using std::cout; 7 using std::endl; 8 9 class Matrix { 10 public: 11 Matrix(int n); // 构造函数,n*n矩阵 12 Matrix(int n, int m); // 构造函数,n*m矩阵 13 Matrix(const Matrix& X); // 复制构造函数 14 ~Matrix(); //析构函数 15 16 void set(const double* pvalue); // 用pvalue指向内存块数据按行为矩阵赋值 17 void set(int i, int j, int value); //设置矩阵第i行第j列元素值为value 18 double& at(int i, int j); //返回矩阵第i行第j列元素的引用 19 double at(int i, int j) const; // 返回矩阵第i行第j列元素的值 20 int get_lines() const; //返回矩阵行数 21 int get_cols() const; //返回矩阵列数 22 void print() const; //按行打印矩阵 23 24 private: 25 int lines; // 矩阵行数 26 int cols; // 矩阵列数 27 double* p; // 指向存放矩阵数据的内存块的首地址 28 }; 29 30 // 类Matrix的实现 31 // 构造函数,n*n矩阵 32 Matrix::Matrix(int n) { 33 p = new double[n * n]; 34 lines = n; 35 cols = n; 36 } 37 // 构造函数,n*m矩阵 38 Matrix::Matrix(int n, int m){ 39 p = new double[n * m]; 40 lines = n; 41 cols = m; 42 } 43 // 复制构造函数 44 Matrix::Matrix(const Matrix& X){ 45 p = new double[X.lines * X.cols]; 46 lines = X.lines; 47 cols = X.cols; 48 for (auto i = 0; i < X.lines * X.cols; ++i) { 49 p[i] = X.p[i]; 50 } 51 } 52 //析构函数 53 Matrix::~Matrix(){ 54 delete[]p; 55 } 56 57 // 用pvalue指向内存块数据按行为矩阵赋值 58 void Matrix::set(const double* pvalue){ 59 for (auto i = 0; i < sizeof(pvalue) && i < lines * cols; i++) {//退出循环的条件为i没到pvalue的长度或i小于矩阵的大小 60 p[i] = pvalue[i]; 61 } 62 } 63 //设置矩阵第i行第j列元素值为value 64 void Matrix::set(int i, int j, int value){ 65 p[i * lines + j] = value; 66 } 67 //返回矩阵第i行第j列元素的引用 68 double& Matrix::at(int i, int j){ 69 assert(i >= 0 && i < lines); 70 assert(j >= 0 && j < cols); 71 return p[i * lines + j]; 72 } 73 // 返回矩阵第i行第j列元素的值 74 double Matrix::at(int i, int j) const{ 75 return p[i * lines + j]; 76 } 77 //返回矩阵行数 78 int Matrix::get_lines() const{ 79 return lines; 80 } 81 //返回矩阵列数 82 int Matrix::get_cols() const{ 83 return cols; 84 } 85 //按行打印矩阵 86 void Matrix::print() const{ 87 for (auto i = 0,j = 0; i < lines * cols; i++) { 88 cout << p[i] << ", "; 89 j++; 90 if (j == cols){//若打完一行就换行 91 cout << "\b\b \n"; 92 j = 0; 93 } 94 } 95 }
task6.cpp
1 #include <iostream> 2 #include "matrix.hpp" 3 4 void test() { 5 using namespace std; 6 7 double x[] = { 1, 2, 3, 4, 5, 6 }; 8 9 Matrix m1(3, 2); 10 m1.set(x); 11 m1.print(); 12 cout << "the first line is: " << endl; 13 cout << m1.at(0, 0) << " " << m1.at(0, 1) << endl; 14 cout << endl; 15 16 Matrix m2(2, 3); 17 m2.set(x); 18 m2.print(); 19 cout << "the first line is: " << endl; 20 cout << m2.at(0, 0) << " " << m2.at(0, 1) << " " << m2.at(0, 2) << endl; 21 cout << endl; 22 23 Matrix m3(m2); 24 m3.set(0, 0, 999); 25 m3.print(); 26 } 27 28 int main() { 29 test(); 30 }
测试结果1:
测试结果2:
更换后的测试数据——
测试结果
别的收获:
在编写task6代码时测试一下代码,发现 cout 的获取输出的数据的顺序是从后往前的。