今天下午看书的时候发现一个很有趣的问题。
《C++ Primer Plus》
示例14.1(studentc.h)中,定义了两个非常奇怪的类成员函数
#ifndef STUDENTC_H_ #define STUDENTC_H_ #include <iostream> #include <string> #include <valarray> class Student { private: typedef std::valarray<double> ArrayDb; std::string name; ArrayDb scores; std::ostream & arr_out(std::ostream & os) const; public: Student() :name("Null Student"), scores() {} explicit Student(const std::string & s) :name(s), scores() {} explicit Student(int n) :name("Nully"), scores(n) {} Student(const std::string & s, int n) :name(s), scores(n) {} Student(const std::string & s, const ArrayDb & a) : name(s), scores(a) {} Student(const char * str, const double * pd, int n) :name(str), scores(pd,n) {} ~Student() {} double Average() const; const std::string & Name() const; //[]重载函数1,似乎能匹配所有的调用 double & operator[](int i); //[]重载函数2,在有上面的重载函数下,似乎永远匹配不到 double operator[](int i) const; friend std::istream & operator>>(std::istream & is, Student & stu); friend std::istream & getline(std::istream & is, Student & stu); friend std::ostream & operator<<(std::ostream & os,const Student & stu); }; #endif
在实现中分别添加两个重载函数的输出,查看调用的情况
#include "studentc.h" using std::ostream; using std::endl; using std::istream; using std::string; double Student::Average() const { if (scores.size() > 0) return scores.sum() / scores.size(); else return 0; } const string & Student::Name() const { return name; } double & Student::operator[](int i) { std::cout << "Use double & " << std::endl; return scores[i]; } double Student::operator[](int i) const { std::cout << "Use double " << std::endl; return scores[i]; } ostream & Student::arr_out(ostream & os) const { int i; int lim = scores.size(); if (lim > 0) { for(i = 0; i < lim; i++) { os << scores[i] << " "; if ( i % 5 == 4) os << endl; } if ( i % 5 != 0) os << endl; } else os << " empty array "; return os; } istream & operator>>(istream & is, Student & stu) { is >> stu.name; return is; } istream & getline(istream & is, Student & stu) { getline(is,stu.name); return is; } ostream & operator<<(ostream & os, const Student & stu) { os << "Scores for " << stu.name << ":\n"; stu.arr_out(os); return os; }
#include "studentc.h" #include <iostream> using std::cin; using std::cout; using std::endl; void set(Student & sa, int n); const int pupils = 3; const int quizzes = 5; int main(void) { Student ada[pupils] = { Student(quizzes), Student(quizzes), Student(quizzes) }; int i; for (i = 0; i < pupils; ++i) set(ada[i], quizzes); cout << "\nStudent List:\n"; for (i = 0; i < pupils; ++i) cout << ada[i].Name() << endl; cout << "\nResults:"; for (i = 0; i < pupils; ++i) { cout << endl << ada[i]; cout << "average: " << ada[i].Average() << endl; } cout << "Done.\n"; return 0; } void set(Student & sa, int n) { cout << "Please enter the student's name: "; getline(cin,sa); cout << "Please enter " << n << " quiz scores:\n"; for (int i = 0; i < n; i++) cin >> sa[i]; while (cin.get() != '\n') continue; } [code title="输出结果"] Please enter the student's name: Gil Bayts Please enter 5 quiz scores: Use double & 92 94 96 93 95 Use double & Use double & Use double & Use double & Please enter the student's name: Pat Roone Please enter 5 quiz scores: Use double & 83 89 72 78 95 Use double & Use double & Use double & Use double & Please enter the student's name: Fleur O'Day Please enter 5 quiz scores: Use double & 92 89 96 74 64 Use double & Use double & Use double & Use double & Student List: Gil Bayts Pat Roone Fleur O'Day Results: Scores for Gil Bayts: 92 94 96 93 95 average: 94 Scores for Pat Roone: 83 89 72 78 95 average: 83.4 Scores for Fleur O'Day: 92 89 96 74 64 average: 83 Done.
可以看到,const版本的operator[]完全没有匹配到。