2010-12-05 163 views
0

我有兩個類Person和Student。我目前正試圖從Person類派生Student類。鏈接錯誤C++繼承

但是,我不斷收到鏈接錯誤。

錯誤:

[Linker error] undefined reference to `Person::Person()' 

我的代碼:

#include <iostream> 
#include <iomanip> 
using namespace std; 

class Person { 

    string Name ; 

public: 
    Person(void); 

    void set(){ 

     cout << "Name:" << endl ; 
     cin >> Name ; 
     } 

    string get_Name(){ 
     return Name ; 
     } 


}; 

class Student:Person { 
    int x, Lab1, Lab2, Lab3, Lab4, Lab5, Lab6, LabPoints, Midterm, Final ; 
    float LabAvg, ExamAvg, Prcnt ; 
    string StuName ; 

public: 
    Student(void); 

    void set(int i){ 
     x = i; 

     cout << "Student " << x << endl << endl ; 
     cout << "Name:" << endl ; 
     cin >> StuName ; 
     cout << "Lab 1 score (1-10): " << endl ; 
     cin >> Lab1 ; 
     cout << "Lab 2 score (1-10): " << endl ; 
     cin >> Lab2 ; 
     cout << "Lab 3 score (1-10): " << endl ; 
     cin >> Lab3 ; 
     cout << "Lab 4 score (1-10): " << endl ; 
     cin >> Lab4 ; 
     cout << "Lab 5 score (1-10): " << endl ; 
     cin >> Lab5 ; 
     cout << "Lab 6 score (1-10): " << endl ; 
     cin >> Lab6 ; 
     cout << "Midterm score (1-100): " << endl ; 
     cin >> Midterm ; 
     cout << "Final score (1-100): " << endl ; 
     cin >> Final ; 

     LabAvg = (Lab1 + Lab2)/2.0 ; 
     LabPoints = (Lab1 + Lab2) ; 
     ExamAvg = (Midterm + Final)/2.0 ; 
     Prcnt = (((LabPoints/60.0) * 0.6) + ((Midterm/100.0) * 0.2) + 
     ((Final/100.0) * 0.2)) * 100 ; 

     } 

    string get_StuName(){ 
     return StuName ; 
     } 

    int get_Lab1(){ 
     return Lab1 ; 
     } 

    int get_Lab2(){ 
     return Lab2 ; 
     } 

    int get_Lab3(){ 
     return Lab3 ; 
     } 

    int get_Lab4(){ 
     return Lab4 ; 
     } 

    int get_Lab5(){ 
     return Lab5 ; 
     } 

    int get_Lab6(){ 
     return Lab6 ; 
     } 

    float get_LabAvg(){ 
     return LabAvg ; 
     } 

    int get_LabPoints(){ 
     return LabPoints ; 
     } 

    float get_ExamAvg(){ 
     return ExamAvg ; 
     } 

    float get_Prcnt(){ 
     return Prcnt ; 
     } 
}; 

Student::Student(void){ 
    x = 0, Lab1 = 0, Lab2 = 0, Lab3 = 0, Lab4 = 0, Lab5 = 0, Lab6 = 0, 
    LabPoints = 0, Midterm = 0, Final = 0 ; 
    LabAvg = 0.0, ExamAvg = 0.0, Prcnt = 0.0 ; 
    StuName = "" ; 
} 

int main(){ 
    int MaxNumStu = 10, NumOfRep , i ; 
    float FPrcnt ; 
    string LetGrd ; 

    cout << "Number of Students:" << endl ; 
    cin >> NumOfRep ; 
    cout << endl << endl ; 

    Student obs[MaxNumStu] ; 

    NumOfRep = ++NumOfRep ; 
    for(i=1 ; i < NumOfRep ; i++) 
    obs[i].set(i) ; 

    cout << endl << "---------------------------------" << endl << endl ; 

    for(i=1; i < NumOfRep; i++){ 
    cout << obs[i].get_StuName() << endl << endl; 
    cout << "Lab 1 Score: " << obs[i].get_Lab1() << endl ; 
    cout << "Lab 2 Score: " << obs[i].get_Lab2() << endl ; 
    cout << "Lab 3 Score: " << obs[i].get_Lab3() << endl ; 
    cout << "Lab 4 Score: " << obs[i].get_Lab4() << endl ; 
    cout << "Lab 5 Score: " << obs[i].get_Lab5() << endl ; 
    cout << "Lab 6 Score: " << obs[i].get_Lab6() << endl ; 
    cout << endl << "Average Lab Score: " << setprecision(4) << 
    obs[i].get_LabAvg() << endl ; 
    cout << "Total Lab Points: " << obs[i].get_LabPoints() << endl ; 
    cout << endl << "Average Exam Score: " << setprecision(4) << 
    obs[i].get_ExamAvg() << endl ; 

    FPrcnt = obs[i].get_Prcnt() ; 

    if (FPrcnt >= 90) 
     LetGrd = "% A" ; 
    if (FPrcnt >= 80) 
     if (FPrcnt < 90) 
      LetGrd = "% B" ; 
    if (FPrcnt >= 70) 
     if (FPrcnt < 80) 
      LetGrd = "% C" ; 
    if (FPrcnt >= 60) 
     if (FPrcnt < 70) 
      LetGrd = "% D" ; 
    if (FPrcnt < 60) 
     LetGrd = "% F"; 

    cout << endl << endl << "Overall Grade: " << setprecision(3) << FPrcnt 
    << LetGrd << endl; 
    cout << endl << endl ; 
    } 
    system("pause") ; 
} 

回答

4

你已經申報了默認構造Person

public: 
    Person(void); 

但從未定義它。

在這種情況下,您不需要擁有自己的默認構造函數 - 刪除聲明並讓編譯器爲您生成默認值。 或者,你可以定義一個什麼都不做:

Person::Person() 
{ 
} 
1

您在第10行申報人::人(),但你永遠不定義它。

3

您已經聲明Person::Person(void);沒有定義它。

如果你刪除該行,你會好起來的。

1

這個問題是關係到C++兩個基本概念:構造派生類,編譯器生成的成員函數

對於派生類的構造函數,首先需要調用基類的構造函數。在這種情況下,

Student::Student(void){ 
    x = 0, Lab1 = 0, Lab2 = 0, Lab3 = 0, Lab4 = 0, Lab5 = 0, Lab6 = 0, 
    LabPoints = 0, Midterm = 0, Final = 0 ; 
    LabAvg = 0.0, ExamAvg = 0.0, Prcnt = 0.0 ; 
    StuName = "" ; 
} 

它沒有指定要使用哪個類Person的構造函數。所以編譯器會自動插入Person的默認構造函數Person:Person()的調用。

對於編譯器生成的函數,編譯器會生成用於每個類別的那些下面的成員函數,如果他們是申報

  • 默認構造
  • 複製構造
  • 賦值運算
  • 析構函數

然而,類Person宣佈構造Person:Person(void)所以沒有默認的構造函數會被編譯器生成。當涉及鏈接類Student的Person:Person()時,編譯器會引發鏈接錯誤。

鏈接錯誤有兩種解決方案。

  1. 擦除Person::Person(void);的減速度,因此會生成默認構造函數。
  2. 添加Person::Person(void);的定義。鏈接器將使用我們自己的默認構造函數。

爲了擴展,讓我們假設在person類中定義的構造函數是Person:Person(string name)。 然後解決方案2也需要修改類Student的構造函數。

Student::Student(void):Person("any string"){ 
    x = 0, Lab1 = 0, Lab2 = 0, Lab3 = 0, Lab4 = 0, Lab5 = 0, Lab6 = 0, 
    LabPoints = 0, Midterm = 0, Final = 0 ; 
    LabAvg = 0.0, ExamAvg = 0.0, Prcnt = 0.0 ; 
    StuName = "" ; 
}