[問題] 類別、建構子問題

作者: amamoimi (佛仔)   2023-09-18 17:16:13
程式新手
前幾天寫了一個練習題拿去問chatgpt
#include <iostream>
using namespace std;
class Animal {
public:
string name;
Animal(string, string, int);
void print1();
protected:
string type;
int weight;
};
Animal::Animal(string n, string t, int w) {
name = n;
type = t;
weight = w;
}
void Animal::print1() {
cout << "name:" << name << " type:" << type << " weight:" << weight;
}
class Cat : public Animal {
public:
Cat(string, string, int, int, int);
void print2();
private:
int body_length;
int tail_length;
};
Cat::Cat(string n, string t, int w, int b, int a)
: Animal(n, t, w), body_length(b), tail_length(a) {}
void Cat::print2() {
print1();
cout << " body length:" << body_length << " tail length:" << tail_length;
}
class Human : public Animal {
public:
Human(string, string, int, int, string);
void print3();
Cat pet;
protected:
int height;
};
Human::Human(string n, string t, int w, int h, string p)
: Animal(n, t, w), height(h) {
pet = Cat(p, "black", 7, 30, 20);
}
void Human::print3() {
print1();
cout << " height:" << height << " pet:" << pet.name << endl;
}
int main() {
Human Betty("Betty", "Asian", 46, 160, "Kitty");
Betty.print3();
Betty.pet.print2();
return 0;
}
結果竟然沒辦法執行xdd
後來回覆他後他又生了一個程式給我,基本上就是把黃字的部分改成
Human::Human(string n, string t, int w, int h, string p)
: Animal(n, t, w), height(h), pet(p, "black", 7, 30, 20) {}
這樣
但是這兩個有什麼不同啊@@他解釋給我聽但是我聽不懂....
但是這兩個有什麼不同啊@@他解釋給我聽但是我聽不懂....
請教板上高手!
作者: amamoimi (佛仔)   2023-09-18 17:29:00
藍字部分不知道為什麼會變藍 請忽略@@
作者: nh60211as   2023-09-18 17:52:00
Cat 沒有 default constructor,所以必須在member initialization list就初始化的樣子錯誤訊息 https://i.imgur.com/XmbhsYq.png
作者: amamoimi (佛仔)   2023-09-18 19:54:00
嗯嗯我有去run過 不過為什麼不能在human constructor的brace裡初始化呀?
作者: mihonisizumi (中原岬マジ天使)   2023-09-18 20:18:00
可參考 Initialization order 的部分https://tinyurl.com/36mvy7e8在第三步驟提到 data member 會在 constructor 的body 前被初始化https://pastebin.com/pDCBK8vU 這份code有加了一個沒參數的 cat constructor 就可以通過編譯 而且從輸出可以看到 constructor 執行的順序
作者: amamoimi (佛仔)   2023-09-18 20:43:00
謝謝大大我覺得我越來越不懂了...那這個code的pet用的是default constructor 還是我自己寫的建構子啊?@@
作者: jack7775kimo (阿龐)   2023-09-18 20:51:00
寫一些std::cout的指令在你自己的constructors中讓你知道compiler呼叫誰
作者: amamoimi (佛仔)   2023-09-18 21:30:00
m大給的那個code 我想說的是 如果assign不能算初始化那pet用的應該是default constructor 但是他之後又assign我設的建構子給pet 那最後pet到底是用什麼建構子..?
作者: mihonisizumi (中原岬マジ天使)   2023-09-18 21:33:00
可參考 explanation 中 effect 的第一點https://tinyurl.com/5n7yu67y由於data member在constructor body前就會被初始但由於沒有提供任何初始化參數 compiler會使用無參數的constructor 在你原本的 Cat 中因爲額外定義了constructor 所以編譯器不會自動幫你產生 defaultconstructor 可參考:https://tinyurl.com/bdh9jcb5在我後來貼的code中 明確定義無參數的constructor後在Human在初始時 會先呼叫無參數的 Cat constructor進到 body 後再呼叫有參數的 constructor可參考執行截圖:https://i.imgur.com/0ErYJcw.pnghttps://pastebin.com/sspJe3mv 稍微修改cout部分
作者: LPH66 (-6.2598534e+18f)   2023-09-18 22:00:00
補充一個: 進 body 後技術上是有參數的 ctor 建構暫時物件再使用複製指定 (或移動指定) 運算子把暫時物件放進 pet 裡在指定過去之後這個暫時物件會被銷毁, 這裡會呼叫解構子這其實跟一個普通的 Cat 變數被指定一個 Cat(...) 的過程是一模一樣的, 這並不是初始化而是一個新物件蓋過去寫在 : 那行裡的才是真正對 pet 變數代表的物件呼叫建構子這個基本上就是 C++ 語法這麼規定, 你寫在那裡才是建構子
作者: mihonisizumi (中原岬マジ天使)   2023-09-18 22:13:00
樓上大大講的內容執行起來大概會是這樣https://pastebin.com/FswErZhDhttps://i.imgur.com/nXSPfhR.png可以觀察記憶體位置瞭解技術上是怎麼運作
作者: amamoimi (佛仔)   2023-09-19 18:07:00
我看的書好像沒有講到rhs...我該換本書嗎==不過我應該懂我問的問題了...總之如果沒有在initialization list裡用我的建構子初始化pet,他就會自動使用default constructor對吧?問題是我沒有設定無引數的建構子所以程式不能執行..

Links booklink

Contact Us: admin [ a t ] ucptt.com