实验四
Task2:
1 Gradecalc.cpp: 2 #include <iostream> 3 #include <vector> 4 #include <string> 5 #include <algorithm> 6 #include <numeric> 7 #include <iomanip> 8 9 using std::vector; 10 using std::string; 11 using std::cin; 12 using std::cout; 13 using std::endl; 14 15 class GradeCalc: public vector<int> { 16 public: 17 GradeCalc(const string &cname, int size); 18 void input(); // 录入成绩 19 void output() const; // 输出成绩 20 void sort(bool ascending = false); // 排序 (默认降序) 21 int min() const; // 返回最低分 22 int max() const; // 返回最高分 23 float average() const; // 返回平均分 24 void info(); // 输出课程成绩信息 25 26 private: 27 void compute(); // 成绩统计 28 29 private: 30 string course_name; // 课程名 31 int n; // 课程人数 32 vector<int> counts = vector<int>(5, 0); // 保存各分数段人数([0, 60), [60, 70), [70, 80), [80, 90), [90, 100] 33 vector<double> rates = vector<double>(5, 0); // 保存各分数段比例 34 }; 35 36 GradeCalc::GradeCalc(const string &cname, int size): course_name{cname}, n{size} {} 37 38 void GradeCalc::input() { 39 int grade; 40 41 for(int i = 0; i < n; ++i) { 42 cin >> grade; 43 this->push_back(grade); 44 } 45 } 46 47 void GradeCalc::output() const { 48 for(auto ptr = this->begin(); ptr != this->end(); ++ptr) 49 cout << *ptr << " "; 50 cout << endl; 51 } 52 53 void GradeCalc::sort(bool ascending) { 54 if(ascending) 55 std::sort(this->begin(), this->end()); 56 else 57 std::sort(this->begin(), this->end(), std::greater<int>()); 58 } 59 60 int GradeCalc::min() const { 61 return *std::min_element(this->begin(), this->end()); 62 } 63 64 int GradeCalc::max() const { 65 return *std::max_element(this->begin(), this->end()); 66 } 67 68 float GradeCalc::average() const { 69 return std::accumulate(this->begin(), this->end(), 0) * 1.0 / n; 70 } 71 72 void GradeCalc::compute() { 73 for(int grade: *this) { 74 if(grade < 60) 75 counts.at(0)++; 76 else if(grade >= 60 && grade < 70) 77 counts.at(1)++; 78 else if(grade >= 70 && grade < 80) 79 counts.at(2)++; 80 else if(grade >= 80 && grade < 90) 81 counts.at(3)++; 82 else if(grade >= 90) 83 counts.at(4)++; 84 } 85 86 for(int i = 0; i < rates.size(); ++i) 87 rates.at(i) = counts.at(i) * 1.0 / n; 88 } 89 90 void GradeCalc::info() { 91 cout << "课程名称:\t" << course_name << endl; 92 cout << "排序后成绩: \t"; 93 sort(); output(); 94 cout << "最高分:\t" << max() << endl; 95 cout << "最低分:\t" << min() << endl; 96 cout << "平均分:\t" << std::fixed << std::setprecision(2) << average() << endl; 97 98 compute(); // 统计各分数段人数、比例 99 100 vector<string> tmp{"[0, 60) ", "[60, 70)", "[70, 80)","[80, 90)", "[90, 100]"}; 101 for(int i = tmp.size()-1; i >= 0; --i) 102 cout << tmp[i] << "\t: " << counts[i] << "人\t" 103 << std::fixed << std::setprecision(2) << rates[i]*100 << "%" << endl; 104 } 105 task2.hpp: 106 #include "GradeCalc.hpp" 107 #include <iomanip> 108 109 void test() { 110 int n; 111 cout << "输入班级人数: "; 112 cin >> n; 113 114 GradeCalc c1("OOP", n); 115 116 cout << "录入成绩: " << endl;; 117 c1.input(); 118 cout << "输出成绩: " << endl; 119 c1.output(); 120 121 cout << string(20, '*') + "课程成绩信息" + string(20, '*') << endl; 122 c1.info(); 123 } 124 125 int main() { 126 test(); 127 }

问题1:
成绩存储在从vector<int>继承而来的容器中;sort方法通过begin()和end()函数就能访问到存储成绩的容器中的每个元素;min方法使用this->begin()和this->end()作为迭代器范围传递给std::min_element函数,以此来遍历存储成绩的容器中的每个元素;max方法与min方法类似;average方法通过this->begin()和this->end()指定要累加的元素范围,也就是存储成绩的容器中的所有成绩元素,然后再除以学生人数n得到平均分;output方法通过一个循环遍历成绩容器。
问题2:
在代码的line68处,分母n的功能是用于计算各分数段人数占总人数的比例;
如果去掉乘以1.0的代码,重新编译和运行,结果会有影响。在原来的代码中,counts.at(i)是int类型,n也是int类型,当进行除法运算counts.at(i) / n时,如果不进行类型转换(乘以1.0就是将其中一个操作数转换为double类型),结果会按照整数除法规则进行计算,即只保留整数部分,小数部分会被截断。
乘以1.0后,会将其中一个操作数隐式转换为double类型,使得除法运算按照浮点数除法规则进行,能得到正确的比例值,保留小数部分,符合统计各分数段比例的需求。
问题3:
在input方法中,没有对用户输入的成绩进行有效性验证。
目前GradeCalc类只考虑了单一课程的成绩统计。在实际应用场景中,可能需要同时处理多门课程的成绩情况。
程序运行结束后,所有输入的成绩数据和计算得到的统计信息都会丢失。
Task3:
1 复制代码 2 #include <iostream> 3 #include <vector> 4 #include <string> 5 #include <algorithm> 6 #include <numeric> 7 #include <iomanip> 8 9 using std::vector; 10 using std::string; 11 using std::cin; 12 using std::cout; 13 using std::endl; 14 15 class GradeCalc { 16 public: 17 GradeCalc(const string &cname, int size); 18 void input(); // 录入成绩 19 void output() const; // 输出成绩 20 void sort(bool ascending = false); // 排序 (默认降序) 21 int min() const; // 返回最低分 22 int max() const; // 返回最高分 23 float average() const; // 返回平均分 24 void info(); // 输出课程成绩信息 25 26 private: 27 void compute(); // 成绩统计 28 29 private: 30 string course_name; // 课程名 31 int n; // 课程人数 32 vector<int> grades; // 课程成绩 33 vector<int> counts = vector<int>(5, 0); // 保存各分数段人数([0, 60), [60, 70), [70, 80), [80, 90), [90, 100] 34 vector<double> rates = vector<double>(5, 0); // 保存各分数段比例 35 }; 36 37 GradeCalc::GradeCalc(const string &cname, int size): course_name{cname}, n{size} {} 38 39 void GradeCalc::input() { 40 int grade; 41 42 for(int i = 0; i < n; ++i) { 43 cin >> grade; 44 grades.push_back(grade); 45 } 46 } 47 48 void GradeCalc::output() const { 49 for(int grade: grades) 50 cout << grade << " "; 51 cout << endl; 52 } 53 54 void GradeCalc::sort(bool ascending) { 55 if(ascending) 56 std::sort(grades.begin(), grades.end()); 57 else 58 std::sort(grades.begin(), grades.end(), std::greater<int>()); 59 60 } 61 62 int GradeCalc::min() const { 63 return *std::min_element(grades.begin(), grades.end()); 64 } 65 66 int GradeCalc::max() const { 67 return *std::max_element(grades.begin(), grades.end()); 68 } 69 70 float GradeCalc::average() const { 71 return std::accumulate(grades.begin(), grades.end(), 0) * 1.0 / n; 72 } 73 74 void GradeCalc::compute() { 75 for(int grade: grades) { 76 if(grade < 60) 77 counts.at(0)++; 78 else if(grade >= 60 && grade < 70) 79 counts.at(1)++; 80 else if(grade >= 70 && grade < 80) 81 counts.at(2)++; 82 else if(grade >= 80 && grade < 90) 83 counts.at(3)++; 84 else if(grade >= 90) 85 counts.at(4)++; 86 } 87 88 for(int i = 0; i < rates.size(); ++i) 89 rates.at(i) = counts.at(i) *1.0 / n; 90 } 91 92 void GradeCalc::info() { 93 cout << "课程名称:\t" << course_name << endl; 94 cout << "排序后成绩: \t"; 95 sort(); output(); 96 cout << "最高分:\t" << max() << endl; 97 cout << "最低分:\t" << min() << endl; 98 cout << "平均分:\t" << std::fixed << std::setprecision(2) << average() << endl; 99 100 compute(); // 统计各分数段人数、比例 101 102 vector<string> tmp{"[0, 60) ", "[60, 70)", "[70, 80)","[80, 90)", "[90, 100]"}; 103 for(int i = tmp.size()-1; i >= 0; --i) 104 cout << tmp[i] << "\t: " << counts[i] << "人\t" 105 << std::fixed << std::setprecision(2) << rates[i]*100 << "%" << endl; 106 }
1 #include "GradeCalc.hpp" 2 #include <iomanip> 3 4 void test() { 5 int n; 6 cout << "输入班级人数: "; 7 cin >> n; 8 9 GradeCalc c1("OOP", n); 10 11 cout << "录入成绩: " << endl;; 12 c1.input(); 13 cout << "输出成绩: " << endl; 14 c1.output(); 15 16 cout << string(20, '*') + "课程成绩信息" + string(20, '*') << endl; 17 c1.info(); 18 } 19 20 int main() { 21 test(); 22 }

问题1:成绩存储在std::vector<int>这个容器中。min、max、sort、average、output是通过std::vector<int>提供的迭代器接口访问每个成绩。
问题2:类的私有成员变量(如grades、counts、rates等)被隐藏在类的内部,外部代码无法直接访问和修改这些数据,只能通过类提供的公有方法来间接操作。这样可以保证数据的安全性和一致性,防止外部代码无意或恶意地篡改数据,使得程序的行为更加可预测和稳定。
Task4.1:
1 #include <iostream> 2 #include <string> 3 #include <limits> 4 5 using namespace std; 6 7 void test1() { 8 string s1, s2; 9 cin >> s1 >> s2; // cin: 从输入流读取字符串, 碰到空白符(空格/回车/Tab)即结束 10 cout << "s1: " << s1 << endl; 11 cout << "s2: " << s2 << endl; 12 } 13 14 void test2() { 15 string s1, s2; 16 getline(cin, s1); // getline(): 从输入流中提取字符串,直到遇到换行符 17 getline(cin, s2); 18 cout << "s1: " << s1 << endl; 19 cout << "s2: " << s2 << endl; 20 } 21 22 void test3() { 23 string s1, s2; 24 getline(cin, s1, ' '); //从输入流中提取字符串,直到遇到指定分隔符 25 getline(cin, s2); 26 cout << "s1: " << s1 << endl; 27 cout << "s2: " << s2 << endl; 28 } 29 30 int main() { 31 cout << "测试1: 使用标准输入流对象cin输入字符串" << endl; 32 test1(); 33 cout << endl; 34 35 cin.ignore(numeric_limits<streamsize>::max(), '\n'); 36 37 cout << "测试2: 使用函数getline()输入字符串" << endl; 38 test2(); 39 cout << endl; 40 41 cout << "测试3: 使用函数getline()输入字符串, 指定字符串分隔符" << endl; 42 test3(); 43 }

问题1:







浙公网安备 33010602011771号