C++類的詳解

  • 2020 年 3 月 15 日
  • 筆記

超女選秀的例子我們玩了很久,為了學習的需要,暫時離開美眉們,我將採用實際項目開發的例子來講解類的更多知識。

在C語言基礎知識中已學習過文件操作,在實際開發中,為了提高效率,我會把文件操作封裝成一個類,類的聲明如下:

// 文件操作類聲明  class CFile  {  private:    FILE *m_fp;        // 文件指針    bool  m_bEnBuffer; // 是否啟用緩衝區,true-啟用;false-不啟用    public:    CFile();   // 類的構造函數    CFile(bool bEnBuffer);   // 類的構造函數     ~CFile();   // 類的析構函數      void EnBuffer(bool bEnBuffer=true);  // 啟、禁用緩衝區      // 打開文件,參數與fopen相同,打開成功true,失敗返回false    bool Open(const char *filename,const char *openmode);      // 調用fprintf向文件寫入數據    void Fprintf(const char *fmt, ... );      // 調用fgets從文件中讀取一行    bool Fgets(char *strBuffer,const int ReadSize);      // 關閉文件指針    void Close();  };

一、類成員的訪問許可權

C++通過 public、protected、private三個關鍵字來控制成員變數和成員函數的訪問許可權,它們分別表示公有的、受保護的、私有的,被稱為成員訪問限定符。所謂訪問許可權,就是類外面的程式碼訪問該類中成員許可權。

在類的內部,即類的成員函數中,無論成員被聲明為 public、protected 還是private,都是可以互相訪問的,沒有訪問許可權的限制。

在類的外部(定義類的程式碼之外),只能通過對象訪問public的成員,不能訪問 private、protected屬性的成員。

本節重點介紹 public 和 private,protected 將在以後介紹。

private 後面的成員都是私有的,如m_fp和m_bEnBuffer,直到有 public出現才會變成共有的;public 之後再無其他限定符,所以 public後面的成員都是共有的。

private關鍵字的作用在於更好地隱藏類的內部實現,該向外暴露的介面(能通過對象訪問的成員)都聲明為public,不希望外部知道、或者只在類內部使用的、或者對外部沒有影響的成員,都建議聲明為private。

聲明為 private 的成員和聲明為 public 的成員的次序任意,既可以先出現 private部分,也可以先出現 public 部分。如果既不寫 private 也不寫 public,就默認為private。

在一個類體中,private 和 public可以分別出現多次。每個部分的有效範圍到出現另一個訪問限定符或類體結束時(最後一個右花括弧)為止。

您可能會說,將成員變數全部設置為 public 省事,確實,這樣做 99.9%的情況下都不是一種錯誤,我也不認為這樣做有什麼不妥;但是,將成員變數設置為private 是一種軟體設計規範,尤其是在大中型項目中,還是請大家盡量遵守這一原則。

二、成員變數的命名

成員變數大都以m_開頭,這是約定成俗的寫法,不是語法規定的內容。以m_開頭既可以一眼看出這是成員變數,又可以和成員函數中的參數名字區分開。

例如成員函數EnBuffer的函數體如下:

// 啟、禁用緩衝區  void CFile::EnBuffer(bool bEnBuffer)  {    m_bEnBuffer=bEnBuffer;  }

三、構造函數

在CFile類的聲明中,有一些特殊的成員函數CFile(),它就是構造函數(constructor)。

  CFile();   // 類的構造函數    CFile(bool bEnBuffer);   // 類的構造函數    

構造函數的名字和類名相同,沒有返回值,不能被顯式的調用,而是在創建對象時自動執行。

構造函數具備以下特點:

1)構造函數必須是 public 屬性。

2)構造函數沒有返回值,因為沒有變數來接收返回值,即使有也毫無用處,不管是聲明還是定義,函數名前面都不能出現返回值類型,即使是void 也不允許。

3)構造函數可以有參數,允許重載。一個類可以有多個重載的構造函數,創建對象時根據傳遞的參數來判斷調用哪一個構造函數。

4)構造函數在實際開發中會大量使用,它往往用來做一些初始化工作,對成員變數進行初始化等,注意,不能用memset對整下類進行初始化。

示例

CFile::CFile()   // 類的構造函數  {    m_fp=0;    m_bEnBuffer=true;  }    CFile::CFile(bool bEnBuffer)   // 類的構造函數  {    m_fp=0;    m_bEnBuffer=bEnBuffer;  }

四、析構函數

在CFile類的聲明中,還有一個特殊的成員函數~CFile(),它就是析構函數(destructor)。

~CFile();   // 類的析構函數

析構函數的名字在類的名字前加~,沒有返回值,但可以被顯式的調用,在對象銷毀時自動執行,用於進行清理工作,例如釋放分配的記憶體、關閉打開的文件等,這個用途非常重要,可以防止程式設計師犯錯。

析構函數具備以下特點:

1)構造函數必須是 public 屬性的。

2)構造函數沒有返回值,因為沒有變數來接收返回值,即使有也毫無用處,不管是聲明還是定義,函數名前面都不能出現返回值類型,即使是void 也不允許。

3)析構函數不允許重載的。一個類只能有一個析構函數。

CFile::~CFile()   // 類的析構函數  {    Close();  // 調用Close釋放資源  }

五、C++程式也很優雅

很多人說C/C++程式很煩鎖,python程式很優雅,說這話人的很荒謬,那是因為他C/C++並不了解,只要我們願意,可以寫出和python一樣優雅簡潔的程式碼,在book210.cpp中,main函數的程式碼極為精簡。

示例(book210.cpp)

/*   * 程式名:book210.cpp,此程式演示用C++類的更多知識。   * 作者:C語言技術網(www.freecplus.net) 日期:20190525  */  #include <stdio.h>  #include <string.h>  #include <stdarg.h>    // 文件操作類聲明  class CFile  {  private:    FILE *m_fp;        // 文件指針    bool  m_bEnBuffer; // 是否啟用緩衝區,true-啟用;false-不啟用    public:    CFile();   // 類的構造函數    CFile(bool bEnBuffer);   // 類的構造函數     ~CFile();   // 類的析構函數      void EnBuffer(bool bEnBuffer=true);  // 啟、禁用緩衝區      // 打開文件,參數與fopen相同,打開成功true,失敗返回false    bool Open(const char *filename,const char *openmode);      // 調用fprintf向文件寫入數據    void Fprintf(const char *fmt,... );      // 調用fgets從文件中讀取一行    bool Fgets(char *strBuffer,const int ReadSize);      // 關閉文件指針    void Close();  };    int main(int argc,char *argv[])  {    if (argc !=2) { printf("請輸入待打開的文件名。n"); return -1; }      CFile File;      if (File.Open(argv[1],"r")==false) { printf("File.Open(%s)失敗。n",argv[1]); return -1; }      char strLine[301];      while (true)    { // 從文件中讀取每一行      if (File.Fgets(strLine,300)==false) break;        printf("%s",strLine);   // 把從文件中讀到的內容顯示到螢幕    }  }    CFile::CFile()   // 類的構造函數  {    m_fp=0;    m_bEnBuffer=true;  }    CFile::CFile(bool bEnBuffer)   // 類的構造函數  {    m_fp=0;    m_bEnBuffer=bEnBuffer;  }    // 關閉文件指針  void CFile::Close()  {    if (m_fp!=0) fclose(m_fp);  // 關閉文件指針    m_fp=0;  }    CFile::~CFile()   // 類的析構函數  {    Close();  // 調用Close釋放資源  }    // 啟、禁用緩衝區  void CFile::EnBuffer(bool bEnBuffer)  {    m_bEnBuffer=bEnBuffer;  }    // 打開文件,參數與fopen相同,打開成功true,失敗返回false  bool CFile::Open(const char *filename,const char *openmode)  {    Close();  // 打開新的文件之前,如果已經打開了文件,關閉它。      if ( (m_fp=fopen(filename,openmode)) == 0 ) return false;      return true;  }    // 調用fprintf向文件寫入數據  void CFile::Fprintf(const char *fmt,...)  {    if ( m_fp == 0 ) return;      va_list ap;    va_start(arg,ap);    vfprintf(m_fp,fmt,ap);    va_end(ap);      if ( m_bEnBuffer == false ) fflush(m_fp);  }    // 調用fgets從文件中讀取一行  bool CFile::Fgets(char *strBuffer,const int ReadSize)  {    if ( m_fp == 0 ) return false;      memset(strBuffer,0,ReadSize);      if (fgets(strBuffer,ReadSize,m_fp) == 0) return false;      return true;  }

book210運行的效果就是把文件的內容一行一行的顯示出來,類型linux系統的cat命令。

在這裡插入圖片描述

六、類的其它知識

關於類的其它知識,包括this指針、static靜態成員、友元等內容,意義不大,我不介紹了,時間太寶貴,有太多重要的知識要學習,沒必要把時間浪費在這些不痛不癢又沒什麼實用價值的知識點上,大家以後有時間了再看也行。

七、可變參數

我們已經介紹過printf、fprintf、sprintf、snprintf函數,它們是一組功能相似的函數,並且有一個共同點,就是函數的參數列表是可以變化的。

函數的聲明如下:

int printf(const char *format, ...);        // 格式化輸出到螢幕  int fprintf(FILE *stream, const char *format, ...);  // 格式化輸出到文件  int sprintf(char *str, const char *format, ...);     // 格式化輸出到字元串  int snprintf(char *str, size_t size, const char *format, ...); // 格式化輸出指定長度的內容到字元串

在實際開發中,我們的自定義函數也會用到可變參數,實現類似上述函數的功能,例如CFile類的Fprintf成員函數。

C語言採用va_start宏、va_end宏和一系列函數來實現可變參數功能。

void CFile::Fprintf(const char *fmt,...)  {    if ( m_fp == 0 ) return;      va_list ap;    va_start(arg,ap);    vfprintf(m_fp,fmt,ap);    va_end(ap);      if ( m_bEnBuffer == false ) fflush(m_fp);  }

以CFile類的Fprintf成員函數為例。

void CFile::Fprintf(const char *fmt,...);     // 可變參數自定義函數的聲明方法

va_list指針、va_start宏、va_end宏用於分析參數,難以理解,大家會用就行,我不詳細介紹。

  va_list ap;    va_start(ap,fmt);    vfprintf(m_fp,fmt,ap);    va_end(ap);

vfprintf函數把宏分析的結果輸出到文件,還有一系列功能相似的函數,聲明如下:

// 輸出的螢幕  int vprintf(const char *format, va_list ap);  // 輸出到文件  int vfprintf(FILE *stream, const char *format, va_list ap);  // 輸出到字元串  int vsprintf(char *str, const char *format, va_list ap);  // 輸出到字元串,第二個參數指定了輸出結果的長度,類似snprintf函數。  int vsnprintf(char *str, size_t size, const char *format, va_list ap);

八、課後作業

1)編寫示常式序,測試類的類成員的訪問許可權。

2)編寫示常式序,測試類的構造函數和它的重載,採用gdb跟蹤構造函數的執行過程。

3)編寫示常式序,測試類的析構函數,採用gdb跟蹤析構造的執行過程。

4)編寫示常式序,實現printf、sprintf和snprintf函數的功能,函數的聲明如下:

int myprintf(const char *format, ...);  int mysprintf(const char *format, ...);  int mysnprintf(const char *format, ...);

5)類定義包括成員變數和成員函數的聲明以及成員函數的定義,在實際開發中,我們通常將公共類的聲明放在頭文件中(如_public.h),成員函數的定義放在程式文件中(如_public.cpp),請按這種方式修改book210.cpp程式,增加_public.h和_public.cpp程式,修改makefile。

九、版權聲明

C語言技術網原創文章,轉載請說明文章的來源、作者和原文的鏈接。
來源:C語言技術網(www.freecplus.net)
作者:碼農有道

如果文章有錯別字,或者內容有錯誤,或其他的建議和意見,請您留言指正,非常感謝!!!