亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C++箴言:訪問模板化基類中名字

2019-11-17 05:09:06
字體:
來源:轉載
供稿:網友

  假設我們要寫一個應用程序,它可以把消息傳送到幾個不同的公司去。消息既可以以加密方式也可以以明文(不加密)的方式傳送。假如我們有足夠的信息在編譯期間確定哪個消息將要發送給哪個公司,我們就可以用一個 template-based(模板基)來解決問題:

  class CompanyA {
  public:
  ...
  void sendCleartext(const std::string& msg);
  void sendEncrypted(const std::string& msg);
  ...
  };

  class CompanyB {
  public:
  ...
  void sendCleartext(const std::string& msg);
  void sendEncrypted(const std::string& msg);
  ...
  };
  ... // classes for other companies

  class MsgInfo { ... }; // class for holding information
  // used to create a message
  template
  class MsgSender {
  public:
  ... // ctors, dtor, etc.

  void sendClear(const MsgInfo& info)
  {
  std::string msg;
  create msg from info;

  Company c;
  c.sendCleartext(msg);
  }
  void sendSecret(const MsgInfo& info) // similar to sendClear, except
  { ... } // calls c.sendEncrypted
  };

  這個能夠很好地工作,但是假設我們有時需要在每次發送消息的時候把一些信息記錄到日志中。通過一個 derived class(派生類)可以很簡單地增加這個功能,下面這個似乎是一個合理的方法:

  template
  class LoggingMsgSender: public MsgSender {
  public:
  ... // ctors, dtor, etc.
  void sendClearMsg(const MsgInfo& info)
  {
  write "before sending" info to the log;
  sendClear(info); // call base class function;
  // this code will not compile!
  write "after sending" info to the log;
  }
  ...
  };

  注重 derived class(派生類)中的 message-sending function(消息發送函數)的名字 (sendClearMsg) 與它的 base class(基類)中的那個(在那里,它被稱為 sendClear)不同。這是一個好的設計,因為它避開了 hiding inherited names(隱藏繼續來的名字)的問題(參見《C++箴言:避免覆蓋通過繼續得到的名字》)和重定義一個 inherited non-virtual function(繼續來的非虛擬函數)的與生俱來的問題(參見《C++箴言:絕不重定義繼續的非虛擬函數》)。但是上面的代碼不能通過編譯,至少在符合標準的編譯器上不能。這樣的編譯器會抱怨 sendClear 不存在。我們可以看見 sendClear 就在 base class(基類)中,但編譯器不會到那里去尋找它。我們有必要理解這是為什么。

  問題在于當編譯器碰到 class template(類模板)LoggingMsgSender 的 definition(定義)時,它們不知道它從哪個 class(類)繼續。當然,它是 MsgSender,但是 Company 是一個 template parameter(模板參數),這個直到更遲一些才能被確定(當 LoggingMsgSender 被實例化的時候)。不知道 Company 是什么,就沒有辦法知道 class(類)MsgSender 是什么樣子的。非凡是,沒有辦法知道它是否有一個 sendClear function(函數)。

  為了使問題具體化,假設我們有一個要求加密通訊的 class(類)CompanyZ:


  class CompanyZ { // this class offers no
  public: // sendCleartext function
  ...
  void sendEncrypted(const std::string& msg);
  ...
  };

  一般的 MsgSender template(模板)不適用于 CompanyZ,因為那個模板提供一個 sendClear function(函數)對于 CompanyZ objects(對象)沒有意義。為了糾正這個問題,我們可以創建一個 MsgSender 針對 CompanyZ 的特化版本:

  template<> // a total specialization of
  class MsgSender { // MsgSender; the same as the
  public: // general template, except
  ... // sendCleartext is omitted
  void sendSecret(const MsgInfo& info)
  { ... }
  };

  注重這個 class definition(類定義)開始處的 "template <>" 語法。它表示這既不是一個 template(模板),也不是一個 standalone class(獨立類)。正確的說法是,它是一個用于 template argument(模板參數)為 CompanyZ 時的 MsgSender template(模板)的 specialized version(特化版本)。這以 total template specialization(完全模板特化)聞名:template(模板)MsgSender 針對類型 CompanyZ 被特化,而且這個 specialization(特化)是 total(完全)的——只要 type parameter(類型參數)被定義成了 CompanyZ,就沒有剩下能被改變的其它 template's parameters(模板參數)。
  已知 MsgSender 針對 CompanyZ 被特化,再次考慮 derived class(派生類)LoggingMsgSender:

  template
  class LoggingMsgSender: public MsgSender {
  public:
  ...
  void sendClearMsg(const MsgInfo& info)

  {
  write "before sending" info to the log;
  sendClear(info); // if Company == CompanyZ,
  // this function doesn't exist!
  write "after sending" info to the log;
  }
  ...
  };

  就像注釋中寫的,當 base class(基類)是 MsgSender 時,這里的代碼是無意義的,因為那個類沒有提供 sendClear function(函數)。這就是為什么 C++ 拒絕這個調用:它認可 base class templates(基類模板)可以被特化,而這個特化不一定提供和 general template(通用模板)相同的 interface(接口)。結果,它通常會拒絕在 templatized base classes(模板化基類)中尋找 inherited names(繼續來的名字)。在某種意義上,當我們從 Object-oriented C++ 跨越到 Template C++,inheritance(繼續)會停止工作。

  為了重新啟動它,我們必須以某種方式使 C++ 的 "don't look in templatized base classes"(不在模板基類中尋找)行為失效。有三種方法可以做到這一點。首先,你可以在調用 base class functions(基類函數)的前面加上 "this->":

  template
  class LoggingMsgSender: public MsgSender {
  public:
  ...

  void sendClearMsg(const MsgInfo& info)
  {
  write "before sending" info to the log;
  this->sendClear(info); // okay, assumes that
  // sendClear will be inherited
  write "after sending" info to the log;
  }
  ...
  };

  第二,你可以使用一個 using declaration,假如你已經讀過《C++箴言:避免覆蓋通過繼續得到的名字》,這應該是你很熟悉的一種解決方案。該文解釋了 using declarations 如何將被隱藏的 base class names(基類名字)引入到一個 derived class(派生類)領域中。因此我們可以這樣寫 sendClearMsg:

  template
  class LoggingMsgSender: public MsgSender {
  public:
  using MsgSender::sendClear; // tell compilers to assume
  ... // that sendClear is in the
  // base class
  void sendClearMsg(const MsgInfo& info)
  {
  ...
  sendClear(info); // okay, assumes that
  ... // sendClear will be inherited
  }
  ...
  };

 ?。m然 using declaration 在這里和《C++箴言:避免覆蓋通過繼續得到的名字》中都可以工作,但要解決的問題是不同的。這里的情形不是 base class names(基類名字)被 derived class names(派生類名字)隱藏,而是假如我們不告訴它去做,編譯器就不會搜索 base class 領域。)

  最后一個讓你的代碼通過編譯的辦法是顯式指定被調用的函數是在 base class(基類)中的:

  template
  class LoggingMsgSender: public MsgSender {
  public:
  ...
  void sendClearMsg(const MsgInfo& info)
  {
  ...
  MsgSender::sendClear(info); // okay, assumes that
  ... // sendClear will be
  } // inherited

  ...
  };

  通常這是一個解決這個問題的最不合人心的方法,因為假如被調用函數是 virtual(虛擬)的,顯式限定會關閉 virtual binding(虛擬綁定)行為。

  從名字可見性的觀點來看,這里每一個方法都做了同樣的事情:它向編譯器保證任何后繼的 base class template(基類模板)的 specializations(特化)都將支持 general template(通用模板)提供的 interface(接口)。所有的編譯器在解析一個像 LoggingMsgSender 這樣的 derived class template(派生類模板)是,這樣一種保證都是必要的,但是假如保證被證實不成立,真相將在后繼的編譯過程中暴露。
例如,假如后面的源代碼中包含這些,

  LoggingMsgSender zMsgSender;
  MsgInfo msgData;
  ... // put info in msgData
  zMsgSender.sendClearMsg(msgData); // error! won't compile

  對 sendClearMsg 的調用將不能編譯,因為在此刻,編譯器知道 base class(基類)是 template specialization(模板特化)MsgSender,它們也知道那個 class(類)沒有提供 sendClearMsg 試圖調用的 sendClear function(函數)。

  從根本上說,問題就是編譯器是早些(當 derived class template definitions(派生類模板定義)被解析的時候)診斷對 base class members(基類成員)的非法引用,還是晚些時候(當那些 templates(模板)被特定的 template arguments(模板參數)實例化的時候)再進行。C++ 的方針是寧愿早診斷,而這就是為什么當那些 classes(類)被從 templates(模板)實例化的時候,它假裝不知道 base classes(基類)的內容。

  Things to Remember

  ·在 derived class templates(派生類模板)中,可以經由 "this->" 前綴引用 base class templates(基類模板)中的名字,經由 using declarations,或經由一個 eXPlicit base class qualification(顯式基類限定)。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美成人精品在线播放| 日韩欧美亚洲国产一区| 欧美性videos高清精品| 亚洲第一中文字幕| 成人国内精品久久久久一区| 成人自拍性视频| 热久久这里只有| 欧美精品久久久久| 午夜精品久久久久久久久久久久久| 亚洲人午夜精品| 午夜精品视频在线| 国产精品91在线观看| 国产亚洲精品va在线观看| 欧美性高潮床叫视频| 国产欧美在线播放| 中文字幕欧美国内| 欧美性生活大片免费观看网址| 亚洲成人精品视频| 国产一区二区欧美日韩| 在线不卡国产精品| 日韩精品在线观看一区二区| 国产精品极品美女在线观看免费| 欧美日韩国产va另类| 92看片淫黄大片欧美看国产片| 亚洲国产精品久久久久秋霞不卡| 国产人妖伪娘一区91| 亚洲欧美在线一区| 日韩欧美成人精品| 国产福利成人在线| 中文字幕精品www乱入免费视频| 国产日韩在线亚洲字幕中文| 欧美极品少妇xxxxⅹ免费视频| 高清欧美一区二区三区| 亚洲精品美女在线观看播放| 久久久欧美一区二区| 欧美香蕉大胸在线视频观看| 亚洲成人激情小说| 国产一区二区三区网站| 日韩的一区二区| 日韩一区二区三区xxxx| 欧美成人三级视频网站| 国产亚洲人成网站在线观看| 亚洲桃花岛网站| 欧美激情精品久久久久久久变态| 日韩精品亚洲视频| 欧美噜噜久久久xxx| 成人性教育视频在线观看| 国产日本欧美一区二区三区在线| 欧美人与性动交a欧美精品| 欧美性猛交丰臀xxxxx网站| 日韩免费在线播放| 按摩亚洲人久久| 亚洲色图五月天| 福利微拍一区二区| 欧美午夜丰满在线18影院| 国产91精品久久久久久久| 亚洲色图13p| 在线播放日韩精品| 日韩欧美高清视频| 欧美日韩在线看| 97av视频在线| 精品国产一区二区三区久久久| 国产在线观看一区二区三区| 中文字幕久热精品在线视频| 热久久99这里有精品| 欧美激情一区二区三区高清视频| 国产99在线|中文| 日日骚av一区| 国产精品手机播放| 成人网在线免费看| 久久久久久久久久久成人| 国产精品第一区| 91精品国产沙发| 亚洲а∨天堂久久精品9966| 91精品国产免费久久久久久| 国产九九精品视频| 5566日本婷婷色中文字幕97| 成人av.网址在线网站| 午夜精品一区二区三区视频免费看| 亚洲第一区第二区| 日韩美女视频免费看| 国产精品网站视频| 国产精品久久网| 91精品国产自产在线| 国产精品成人观看视频国产奇米| 亚洲免费视频在线观看| 国内免费精品永久在线视频| 2020国产精品视频| 91性高湖久久久久久久久_久久99| 国产精品久久国产精品99gif| 亚洲精品久久久久| 亚洲а∨天堂久久精品喷水| 亚洲人成人99网站| 亚洲精品国产拍免费91在线| 欧美激情综合亚洲一二区| 国产成人久久久| 亚洲欧洲免费视频| 日韩欧美在线字幕| 一级做a爰片久久毛片美女图片| 亚洲国产精久久久久久久| 久久精品国产视频| 2023亚洲男人天堂| 国产成人在线亚洲欧美| 欧美激情在线观看视频| 亚洲天堂网在线观看| 欧美成人免费全部| 午夜精品在线观看| 亚洲精品中文字幕有码专区| 久久999免费视频| 成人亚洲欧美一区二区三区| 久久久亚洲天堂| 国产亚洲激情在线| 亚洲天堂一区二区三区| 国产成人精品电影| 中文一区二区视频| 欧美乱大交xxxxx| 国产69精品久久久久久| 91美女高潮出水| 黄色成人在线播放| 日韩精品免费看| 亚洲aa在线观看| 精品视频在线播放色网色视频| 亚洲视频在线观看网站| 久久久亚洲欧洲日产国码aⅴ| 国产在线一区二区三区| 91精品国产91久久久久久不卡| 欧美日韩国产影院| 奇门遁甲1982国语版免费观看高清| 日韩免费在线播放| 国产69久久精品成人看| 国产精品mp4| 国产欧美日韩高清| 国产日韩欧美电影在线观看| 国产在线不卡精品| 久久精品99无色码中文字幕| 国产精品视频白浆免费视频| 久久天天躁狠狠躁夜夜躁| 色婷婷**av毛片一区| 亚洲美女精品久久| 国产精品美女久久久免费| 亚洲精品一区二区久| 午夜精品久久久久久久99黑人| 午夜精品一区二区三区在线视| 国产精品丝袜白浆摸在线| 亚洲国产另类 国产精品国产免费| 欧美视频裸体精品| 7m精品福利视频导航| 91在线视频一区| 国产视频精品免费播放| 欧美成人免费视频| 亚洲精品久久视频| 亚洲成avwww人| 欧美精品激情在线观看| 伊人av综合网| 4p变态网欧美系列| 欧美视频在线观看免费网址| 欧美极品在线播放| 精品成人国产在线观看男人呻吟| 中文字幕av日韩| 成人黄色av免费在线观看| 国产精彩精品视频| 久久久久国色av免费观看性色| 国产做受69高潮| 国产xxx69麻豆国语对白|