1. 程式人生 > >c++ 友元函式,模板函式

c++ 友元函式,模板函式

友元函式.

類的友元函式是定義在類外部,但有權訪問類的所有私有(private)成員和保護(protected)成員。儘管友元函式的原型有在類的定義中出現過,但是友元函式並不是成員函式。

友元可以是一個函式,該函式被稱為友元函式;友元也可以是一個類,該類被稱為友元類,在這種情況下,整個類及其所有成員都是友元。

如果要宣告函式為一個類的友元,需要在類定義中該函式原型前使用關鍵字 friend

下面是友元函式的例子:

#include <iostream>

using namespace std;

class Box
{
   double width;
public
: friend void printWidth( Box box ); void setWidth( double wid ); }; // 成員函式定義 void Box::setWidth( double wid ) { width = wid; } // 請注意:printWidth() 不是任何類的成員函式 void printWidth( Box box ) { /* 因為 printWidth() 是 Box 的友元,它可以直接訪問該類的任何成員 */ cout << "Width of box : " << box.width <<endl; } // 程式的主函式
int main( ) { Box box; // 使用成員函式設定寬度 box.setWidth(10.0); // 使用友元函式輸出寬度 printWidth( box ); return 0; }

列印結果:

Width of box : 10

詳細看這裡

模板函式

現在寫個簡單的程式碼 交換函式

#include <iostream>
#include <string>

using namespace std;

void myswap(int& a, int& b){
    int tmp = 0
; tmp = a; a = b; b = tmp; } void myswap(char& a, char& b){ char tmp = 0; tmp = a; a = b; b = tmp; } void main(){ int a = 111; int b = 222; myswap(a, b); cout << "a = " << a << " b = " << b << endl; char c = '1'; char d = '2'; myswap(c, d); cout << "c = " << c << " d = " << c << endl; system("pause"); }

列印結果:

a = 222 b = 111
c = 2 d = 2
請按任意鍵繼續. . .

如果現在又有兩個double 型別的值需要交換,或者float 型別的值交換,怎麼做. 繼續寫函式就不方便了.

在java中使用泛型來做類似的事情, 在c++中 使用模板函式.

使用模板函式

#include <iostream>
#include <string>

using namespace std;

//void myswap(int& a, int& b){
//  int tmp = 0;
//  tmp = a;
//  a = b;
//  b = tmp;
//
//}
//
//void myswap(char& a, char& b){
//  char tmp = 0;
//  tmp = a;
//  a = b;
//  b = tmp;
//}

template <typename T>
void myswap(T& a, T& b){
    T tmp = 0;
    tmp = a;
    a = b;
    b = tmp;
}

void main(){
    int a = 111;
    int b = 222;
    myswap(a, b);

    cout << "a = " << a << " b = " << b << endl;

    char c = '1';
    char d = '2';
    myswap(c, d);

    cout << "c = " << c << " d = " << c << endl;
    system("pause");
}

列印結果一樣:

a = 222 b = 111
c = 2 d = 2
請按任意鍵繼續. . .

和java裡面的泛型 類似

模板類

例項定義了類 Stack<>,並實現了泛型方法來對元素進行入棧出棧操作

模擬入棧出棧操作

#include <iostream>
#include <vector>
#include <cstdlib>
#include <string>
#include <stdexcept>

using namespace std;

template <class T>
class Stack { 
  private: 
    vector<T> elems;     // 元素 

  public: 
    void push(T const&);  // 入棧
    void pop();               // 出棧
    T top() const;            // 返回棧頂元素
    bool empty() const{       // 如果為空則返回真。
        return elems.empty(); 
    } 
}; 

template <class T>
void Stack<T>::push (T const& elem) 
{ 
    // 追加傳入元素的副本
    elems.push_back(elem);    
} 

template <class T>
void Stack<T>::pop () 
{ 
    if (elems.empty()) { 
        throw out_of_range("Stack<>::pop(): empty stack"); 
    }
    // 刪除最後一個元素
    elems.pop_back();         
} 

template <class T>
T Stack<T>::top () const 
{ 
    if (elems.empty()) { 
        throw out_of_range("Stack<>::top(): empty stack"); 
    }
    // 返回最後一個元素的副本 
    return elems.back();      
} 

int main() 
{ 
    try { 
        Stack<int>         intStack;  // int 型別的棧 
        Stack<string> stringStack;    // string 型別的棧 

        // 操作 int 型別的棧 
        intStack.push(7); 
        cout << intStack.top() <<endl; 

        // 操作 string 型別的棧 
        stringStack.push("hello"); 
        cout << stringStack.top() << std::endl; 
        stringStack.pop(); 
        stringStack.pop(); 
    } 
    catch (exception const& ex) { 
        cerr << "Exception: " << ex.what() <<endl; 
        return -1;
    } 
}

程式碼比較簡單 ,很好的解釋了模板類 既可以傳入int 又可以傳入stirng

結果:

7
hello
Exception: Stack<>::pop(): empty stack

C++ list 也原始碼裡也用了模板類 可以 list 轉到定義看一下

class 轉換 型別轉換.

  • static_cast 普通值型別轉換
  • const_cast 去常量
  • dynamic_cast 基類和派生類之間的轉換

- reinterpret_cast 不通用 不常用 函式指標的轉換 (一般在Void * 之間轉)

  • const_cast (expr): const_cast 運算子用於修改型別的 const / volatile 屬性。除了 const 或 volatile 屬性之外,目標型別必須與源型別相同。這種型別的轉換主要是用來操作所傳物件的 const 屬性,可以加上 const 屬性,也可以去掉 const 屬性。

  • dynamic_cast (expr): dynamic_cast 在執行時執行轉換,驗證轉換的有效性。如果轉換未執行,則轉換失敗,表示式 expr 被判定為 null。dynamic_cast 執行動態轉換時,type 必須是類的指標、類的引用或者 void*,如果 type 是類指標型別,那麼 expr 也必須是一個指標,如果 type 是一個引用,那個 expr 也必須是一個引用。

  • reinterpret_cast (expr): reinterpret_cast 運算子把某種指標改為其他型別的指標。它可以把一個指標轉換為一個整數,也可以把一個整數轉換為一個指標。

  • static_cast (expr): static_cast 運算子執行非動態轉換,沒有執行時類檢查來保證轉換的安全性。例如,它可以用來把一個基類指標轉換為派生類指標。

詳細看這裡

異常捕獲

和java 差不多