C++基礎教程面向對象學習筆記及心得感悟[圖]

C++基礎教程面向對象學習筆記及心得感悟[圖]
使用友元函數重載算術運算符:
C ++中一些最經常使用的運算符是算術運算符 - 即加號運算符(+),減運算符( - ),乘法運算符(*)和除法運算符(/)。請注意,全部算術運算符都是二元運算符 - 這意味着它們須要兩個操做數 - 運算符的每一側都有一個操做數。全部這四個運算符都以徹底相同的方式過載。
事實證實,有三種不一樣的方法來重載運算符:成員函數方式,友元函數方式和正常函數方式。在本課中,咱們將介紹友元函數的方式(由於它對大多數二元運算符更直觀)。下一課,咱們將討論正常的函數方式。最後,在本章後面的課程中,咱們將介紹成員函數的方式。固然,咱們還將總結什麼時候更詳細地使用每個。html

C++基礎教程面向對象學習筆記及心得感悟[圖]
使用友元函數重載運算符:
考慮如下的類:
class Cents
{
private:
int m_cents;
public:
Cents(int cents) { m_cents = cents; }
int getCents() const { return m_cents; }
};
如下示例顯示如何重載operator +(+)以便將兩個「Cents」對象一塊兒添加:
#include <iostream>
class Cents
{
private:
int m_cents;
public:
Cents(int cents) { m_cents = cents; }
// 用友元函數添加Cents + Cents
friend Cents operator+(const Cents &c1, const Cents &c2);
int getCents() const { return m_cents; }
};
// 注意: 這個函數不是成員函數
Cents operator+(const Cents &c1, const Cents &c2)
{
//用Cents構造函數和operator +(int,int)
// 咱們能夠直接訪問m_cents由於這是一個友元函數
return Cents(c1.m_cents + c2.m_cents);
}
int main()
{
Cents cents1(6);
Cents cents2(8);
Cents centsSum = cents1 + cents2;
std::cout << "I have " << centsSum.getCents() << " cents." << std::endl;
return 0;
}
這會產生結果:
I have14cents。
重載加號運算符(+)就像聲明一個名爲operator +的函數同樣簡單,爲它提供兩個咱們想要添加的操做數類型的參數,選擇一個合適的返回類型,而後編寫該函數。
對於咱們的Cents對象,實現咱們的operator +()函數很是簡單。首先,參數類型:在這個版本的operator +中,咱們將兩個Cents對象一塊兒添加,所以咱們的函數將採用兩個Cents類型的對象。第二,返回類型:咱們的運算符+將返回類型爲Cents的結果,所以這是咱們的返回類型。
最後,實現:要將兩個Cents對象一塊兒添加,咱們確實須要從每一個Cents對象添加m_cents成員。由於咱們重載的operator +()函數是類的友元函數,因此咱們能夠直接訪問參數的m_cents成員。安妮日記讀後感(http://m.simayi.net/duhougan/1188.html)此外,由於m_cents是一個整數,而且C ++知道如何使用與整數操做數一塊兒使用的plus運算符的內置版本將整數添加到一塊兒,因此咱們能夠簡單地使用+運算符來進行添加。
重載減法運算符( - )也很簡單:
#include <iostream>
class Cents
{
private:
int m_cents;
public:
Cents(int cents) { m_cents = cents; }
// 用友元函數添加Cents + Cents
friend Cents operator+(const Cents &c1, const Cents &c2);
// 用友元函數實現減法 Cents - Cents
friend Cents operator-(const Cents &c1, const Cents &c2);
int getCents() const { return m_cents; }
};
// 注意: 這個函數不是成員函數
Cents operator+(const Cents &c1, const Cents &c2)
{
//用Cents構造函數和operator +(int,int)
// 咱們能夠直接訪問m_cents由於這是一個友元函數
return Cents(c1.m_cents + c2.m_cents);
}
// 注意: 這個函數不是成員函數
Cents operator-(const Cents &c1, const Cents &c2)
{
//用Cents構造函數和operator-(int,int)
// 咱們能夠直接訪問m_cents由於這是一個友元函數
return Cents(c1.m_cents - c2.m_cents);
}
int main()
{
Cents cents1(6);
Cents cents2(2);
Cents centsSum = cents1 - cents2;
std::cout << "I have " << centsSum.getCents() << " cents." << std::endl;
return 0;
}
重載乘法運算符(*)和除法運算符(/)就像定義operator *和operator /的函數同樣簡單。
友元函數能夠在類中定義
即便友元函數不是類的成員,若是須要,它們仍然能夠在類中定義:
#include <iostream>
class Cents
{
private:
int m_cents;
public:
Cents(int cents) { m_cents = cents; }
// 用友元函數添加Cents + Cents
// 即便定義在類中,該函數也不被視爲類的成員
friend Cents operator+(const Cents &c1, const Cents &c2)
{
//用Cents構造函數和operator +(int,int)
// 咱們能夠直接訪問m_cents由於這是一個友元函數
return Cents(c1.m_cents + c2.m_cents);
}
int getCents() const { return m_cents; }
};
int main()
{
Cents cents1(6);
Cents cents2(8);
Cents centsSum = cents1 + cents2;
std::cout << "I have " << centsSum.getCents() << " cents." << std::endl;
return 0;
}
咱們一般不建議這樣作,由於非通常的函數定義最好保存在類定義以外的單獨的.cpp文件中。可是,咱們將在之後的教程中使用此模式來保持示例簡潔。
爲不一樣類型的操做數重載運算符
一般狀況下,您但願重載的運算符使用不一樣類型的操做數。例如,若是咱們有Cents(4),咱們可能想要將整數6加到此處以產生結果Cents(10)。
當C ++計算表達式時x + y,x成爲第一個參數,y成爲第二個參數。當x和y具備相同的類型時,若是添加x + y或y + x可有可無 - 不管哪一種方式,都會調用相同版本的operator +。可是,當操做數具備不一樣的類型時,x + y不會調用與y + x相同的函數。
例如,Cents(4) + 6將調用operator +(Cents,int),並6 + Cents(4)調用operator +(int,Cents)。所以,每當咱們爲不一樣類型的操做數重載二元運算符時,咱們實際上須要編寫兩個函數 - 每種狀況一個。這是一個例子:
#include <iostream>
class Cents
{
private:
int m_cents;
public:
Cents(int cents) { m_cents = cents; }
// 用友元函數添加Cents + int
friend Cents operator+(const Cents &c1, int value);
// 用友元函數添加 int + Cents
friend Cents operator+(int value, const Cents &c1);
int getCents() { return m_cents; }
};
//注意: 這個函數不是成員函數
Cents operator+(const Cents &c1, int value)
{
// 用友元函數添加Cents + Cents
// 即便定義在類中,該函數也不被視爲類的成員
return Cents(c1.m_cents + value);
}
// 注意: 這個函數不是成員函數
Cents operator+(int value, const Cents &c1)
{
// 用友元函數添加Cents + Cents
// 即便定義在類中,該函數也不被視爲類的成員
return Cents(c1.m_cents + value);
}
int main()
{
Cents c1 = Cents(4) + 6;
Cents c2 = 6 + Cents(4);
std::cout << "I have " << c1.getCents() << " cents." << std::endl;
std::cout << "I have " << c2.getCents() << " cents." << std::endl;
return 0;
}
請注意,兩個重載函數都具備相同的實現 - 這是由於它們執行相同的操做,它們只是以不一樣的順序獲取它們的參數。
另外一個例子
咱們來看看另外一個例子:
class MinMax
{
private:
int m_min; // 存放最小值
int m_max; // 存放最大值
public:
MinMax(int min, int max)
{
m_min = min;
m_max = max;
}
int getMin() { return m_min; }
int getMax() { return m_max; }
friend MinMax operator+(const MinMax &m1, const MinMax &m2);
friend MinMax operator+(const MinMax &m, int value);
friend MinMax operator+(int value, const MinMax &m);
};
MinMax operator+(const MinMax &m1, const MinMax &m2)
{
// 在m1和m2中獲取最小值
int min = m1.m_min < m2.m_min ? m1.m_min : m2.m_min;
// 在m1和m2中獲取最大值
int max = m1.m_max > m2.m_max ? m1.m_max : m2.m_max;
return MinMax(min, max);
}
MinMax operator+(const MinMax &m, int value)
{
// 在m1和value中獲取最小值 
int min = m.m_min < value ? m.m_min : value;
// 在m1和value中獲取最大值 
int max = m.m_max > value ? m.m_max : value;
return MinMax(min, max);
}
MinMax operator+(int value, const MinMax &m)
{
// 調用operator+(MinMax, int)
return m + value;
}
int main()
{
MinMax m1(10, 15);
MinMax m2(8, 11);
MinMax m3(3, 12);
MinMax mFinal = m1 + m2 + 5 + 8 + m3 + 16;
std::cout << "Result: (" << mFinal.getMin() << ", " <<
mFinal.getMax() << ")\n";
return 0;
}
MinMax類跟蹤它到目前爲止所見的最小值和最大值。咱們已經重載了+運算符3次,所以咱們能夠將兩個MinMax對象一塊兒添加,或者將整數添加到MinMax對象。
這個例子產生告終果:
Result:(3,16)
您將注意到咱們添加到mFinal的最小值和最大值。
讓咱們再談談「MinMax mFinal = m1 + m2 + 5 + 8 + m3 + 16」的調用方式。請記住,operator +的優先級高於operator =,operator +從左到右進行求值,所以m1 + m2首先求值。這成爲對運算符+(m1,m2)的調用,它產生返回值MinMax(8,15)。而後MinMax(8,15)+ 5接下來調用。這成爲對operator +(MinMax(8,15),5)的調用,它產生返回值MinMax(5,15)。而後MinMax(5,15)+ 8以相同的方式調用以產生MinMax(5,15)。而後MinMax(5,15)+ m3調用產生MinMax(3,15)。最後,MinMax(3,15)+ 16調用爲MinMax(3,16)。而後將最終結果分配給mFinal。
換句話說,此表達式的計算結果爲「MinMax mFinal =((((((m1 + m2)+ 5)+ 8)+ m3)+ 16)」,對於以後的對象,每次連續操做返回一個成爲左操做數的MinMax對象。
使用其餘運算符實現運算符
在上面的例子中,請注意咱們經過調用operator +(MinMax,int)來定義operator +(int,MinMax)(它產生相同的結果)。這容許咱們將operator +(MinMax,int)的實現減小到一行,經過最小化冗餘並使函數更易於理解,使咱們的代碼更易於維護。
一般能夠經過調用其餘重載運算符來定義重載運算符。若是這樣作會產生更簡單的代碼,您應該這樣作。在實現很簡單的狀況下(例如,單行),一般不值得這樣作,由於附加函數調用比直接實現函數更復雜。
Quiz Time:
1a)編寫一個名爲Fraction的類,它具備整數分子和分母成員。編寫print()函數打印出分數。
如下代碼應該編譯:
#include <iostream>
int main()
{
Fraction f1(1, 4);
f1.print();
Fraction f2(1, 2);
f2.print();
}
這應該打印:
1/4
1/2
解決方案:
#include <iostream>
class Fraction
{
private:
int m_numerator = 0;
int m_denominator = 1;
public:
Fraction(int numerator=0, int denominator=1):
m_numerator(numerator), m_denominator(denominator)
{
}
void print()
{
std::cout << m_numerator << "/" << m_denominator << "\n";
}
};
int main()
{
Fraction f1(1, 4);
f1.print();
Fraction f2(1, 2);
f2.print();
return 0;
}
1b)添加劇載乘法運算符以處理分數和整數之間以及兩個分數之間的乘法。使用友元函數方法。
提示:要將兩個分數相乘,首先將兩個分子相乘,而後將兩個分母相乘。要將分數和整數相乘,請將分數的分子乘以整數,並使分母單獨使用。
如下代碼應該編譯:
#include <iostream>
int main()
{
Fraction f1(2, 5);
f1.print();
Fraction f2(3, 8);
f2.print();
Fraction f3 = f1 * f2;
f3.print();
Fraction f4 = f1 * 2;
f4.print();
Fraction f5 = 2 * f2;
f5.print();
Fraction f6 = Fraction(1, 2) * Fraction(2, 3) * Fraction(3, 4);
f6.print();
}
這應該打印:
2/5
3/8
6/40
4/5
6/8
6/24
解決方案
1c)額外信用:分數2/4與1/2相同,但2/4不減小到最低項。咱們能夠經過找到分子和分母之間的最大公約數(GCD),而後將分子和分母除以GCD,將任何給定分數減小到最低項。
如下是查找GCD的功能:
int gcd(int a, int b)
{
return (b == 0) ? (a > 0 ? a : -a) : gcd(b, a % b);
}
將此函數添加到您的類中,並編寫一個名爲reduce()的成員函數來減小您的分數。確保全部餾分都適當減小。
如下應編譯:
#include <iostream>
int main()
{
Fraction f1(2, 5);
f1.print();
Fraction f2(3, 8);
f2.print();
Fraction f3 = f1 * f2;
f3.print();
Fraction f4 = f1 * 2;
f4.print();
Fraction f5 = 2 * f2;
f5.print();
Fraction f6 = Fraction(1, 2) * Fraction(2, 3) * Fraction(3, 4);
f6.print();
return 0;
}
併產生結果:
2/5
3/8
3/20
4/5
3/4
1/4
解決方案
#include <iostream>
class Fraction
{
private:
int m_numerator;
int m_denominator;
public:
Fraction(int numerator=0, int denominator=1):
m_numerator(numerator), m_denominator(denominator)
{
// 咱們在構造函數中放置了reduce()以確保咱們不會獲得假分數
// 因爲全部重載運算符都會建立新的Fractions,所以咱們能夠保證在此處調用它
reduce();
}
// 咱們將使gcd靜態,以便它能夠成爲類Fraction的一部分,而不須要使用類型爲Fraction的對象
static int gcd(int a, int b)
{
return (b == 0) ? (a > 0 ? a : -a) : gcd(b, a % b);
}
void reduce()
{
int gcd = Fraction::gcd(m_numerator, m_denominator);
m_numerator /= gcd;
m_denominator /= gcd;
}
friend Fraction operator*(const Fraction &f1, const Fraction &f2);
friend Fraction operator*(const Fraction &f1, int value);
friend Fraction operator*(int value, const Fraction &f1);
void print()
{
std::cout << m_numerator << "/" << m_denominator << "\n";
}
};
Fraction operator*(const Fraction &f1, const Fraction &f2)
{
return Fraction(f1.m_numerator * f2.m_numerator, f1.m_denominator * f2.m_denominator);
}
Fraction operator*(const Fraction &f1, int value)
{
return Fraction(f1.m_numerator * value, f1.m_denominator);
}
Fraction operator*(int value, const Fraction &f1)
{
return Fraction(f1.m_numerator * value, f1.m_denominator);
}
int main()
{
Fraction f1(2, 5);
f1.print();
Fraction f2(3, 8);
f2.print();
Fraction f3 = f1 * f2;
f3.print();
Fraction f4 = f1 * 2;
f4.print();
Fraction f5 = 2 * f2;
f5.print();
Fraction f6 = Fraction(1, 2) * Fraction(2, 3) * Fraction(3, 4);
f6.print();
return 0;
}ios

相關文章
相關標籤/搜索