青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

旅途

如果想飛得高,就該把地平線忘掉

如何組織編寫模板程序 轉

前兩天遇到一個問題, 第一反應是用模板類來解決是最好的,所以就動手寫了一個類.
但寫完才發現會在編譯器鏈接的時候報錯,百思不得其解,上網一頓搜羅,發現下文,終于解決了我的問題.同時我也對模板類有了進一步的理解.

[轉]
前言
常遇到詢問使用模板到底是否容易的問題,我的回答是:“模板的使用是容易的,但組織編寫卻不容易”。看看我們幾乎每天都能遇到的模板類吧,如STL, ATL, WTL, 以及Boost的模板類,都能體會到這樣的滋味:接口簡單,操作復雜。

我在5年前開始使用模板,那時我看到了MFC的容器類。直到去年我還沒有必要自己編寫模板類。可是在我需要自己編寫模板類時,我首先遇到的事實卻是“傳統”編程方法(在*.h文件聲明,在*.cpp文件中定義)不能用于模板。于是我花費一些時間來了解問題所在及其解決方法。

本文對象是那些熟悉模板但還沒有很多編寫模板經驗的程序員。本文只涉及模板類,未涉及模板函數。但論述的原則對于二者是一樣的。

問題的產生
通過下例來說明問題。例如在array.h文件中有模板類array:
// array.h
template <typename T, int SIZE>
class array
{
T data_[SIZE];
array (const array& other);
const array& ōperator = (const array& other);
public:
array(){};
T& operator[](int i) {return data_[i];}
const T& get_elem (int i) const {return data_[i];}
void set_elem(int i, const T& value) {data_[i] = value;}
operator T*() {return data_;}
};
然后在main.cpp文件中的主函數中使用上述模板:
// main.cpp
#include "array.h"

int main(void)
{
array<int, 50> intArray;
intArray.set_elem(0, 2);
int firstElem = intArray.get_elem(0);
int* begin = intArray;
}
這時編譯和運行都是正常的。程序先創建一個含有50個整數的數組,然后設置數組的第一個元素值為2,再讀取第一個元素值,最后將指針指向數組起點。

但如果用傳統編程方式來編寫會發生什么事呢?我們來看看:

將array.h文件分裂成為array.h和array.cpp二個文件(main.cpp保持不變)
// array.h
template <typename T, int SIZE>
class array
{
T data_[SIZE];
array (const array& other);
const array& ōperator = (const array& other);
public:
array(){};
T& operator[](int i);
const T& get_elem (int i) const;
void set_elem(int i, const T& value);
operator T*();
};
// array.cpp
#include "array.h"

template<typename T, int SIZE> T& array<T, SIZE>::operator [](int i)
{
return data_[i];
}

template<typename T, int SIZE> const T& array<T, SIZE>::get_elem(int i) const
{
return data_[i];
}

template<typename T, int SIZE> void array<T, SIZE>::set_elem(int i, const T& value)
{
data_[i] = value;
}
template<typename T, int SIZE> array<T, SIZE>::operator T*()
{
return data_;
}
編譯時會出現3個錯誤。問題出來了:
為什么錯誤都出現在第一個地方?
為什么只有3個鏈接出錯?array.cpp中有4個成員函數。
要回答上面的問題,就要深入了解模板的實例化過程。

模板實例化
程序員在使用模板類時最常犯的錯誤是將模板類視為某種數據類型。所謂類型參量化(parameterized types)這樣的術語導致了這種誤解。模板當然不是數據類型,模板就是模板,恰如其名:

編譯器使用模板,通過更換模板參數來創建數據類型。這個過程就是模板實例化(Instantiation)。
從模板類創建得到的類型稱之為特例(specialization)。
模板實例化取決于編譯器能夠找到可用代碼來創建特例(稱之為實例化要素,
point of instantiation)。
要創建特例,編譯器不但要看到模板的聲明,還要看到模板的定義。
模板實例化過程是遲鈍的,即只能用函數的定義來實現實例化。


再回頭看上面的例子,可以知道array是一個模板,array<int, 50>是一個模板實例 - 一個類型。從array創建array<int, 50>的過程就是實例化過程。實例化要素體現在main.cpp文件中。如果按照傳統方式,編譯器在array.h文件中看到了模板的聲明,但沒有模板的定義,這樣編譯器就不能創建類型array<int, 50>。但這時并不出錯,因為編譯器認為模板定義在其它文件中,就把問題留給鏈接程序處理。

現在,編譯array.cpp時會發生什么問題呢?編譯器可以解析模板定義并檢查語法,但不能生成成員函數的代碼。它無法生成代碼,因為要生成代碼,需要知道模板參數,即需要一個類型,而不是模板本身。

這樣,鏈接程序在main.cpp 或 array.cpp中都找不到array<int, 50>的定義,于是報出無定義成員的錯誤。

至此,我們回答了第一個問題。但還有第二個問題,在array.cpp中有4個成員函數,鏈接器為什么只報了3個錯誤?回答是:實例化的惰性導致這種現象。在main.cpp中還沒有用上operator[],編譯器還沒有實例化它的定義。

解決方法
認識了問題,就能夠解決問題:
在實例化要素中讓編譯器看到模板定義。
用另外的文件來顯式地實例化類型,這樣鏈接器就能看到該類型。
使用export關鍵字。

前二種方法通常稱為包含模式,第三種方法則稱為分離模式。

第一種方法意味著在使用模板的轉換文件中不但要包含模板聲明文件,還要包含模板定義文件。在上例中,就是第一個示例,在array.h中用行內函數定義了所有的成員函數。或者在main.cpp文件中也包含進array.cpp文件。這樣編譯器就能看到模板的聲明和定義,并由此生成array<int, 50>實例。這樣做的缺點是編譯文件會變得很大,顯然要降低編譯和鏈接速度。

第二種方法,通過顯式的模板實例化得到類型。最好將所有的顯式實例化過程安放在另外的文件中。在本例中,可以創建一個新文件templateinstantiations.cpp:
// templateinstantiations.cpp
#include "array.cpp"

template class array <int, 50>; // 顯式實例化
array<int, 50>類型不是在main.cpp中產生,而是在templateinstantiations.cpp中產生。這樣鏈接器就能夠找到它的定義。用這種方法,不會產生巨大的頭文件,加快編譯速度。而且頭文件本身也顯得更加“干凈”和更具有可讀性。但這個方法不能得到惰性實例化的好處,即它將顯式地生成所有的成員函數。另外還要維護templateinstantiations.cpp文件。

第三種方法是在模板定義中使用export關鍵字,剩下的事就讓編譯器去自行處理了。當我在
Stroustrup的書中讀到export時,感到非常興奮。但很快就發現VC 6.0不支持它,后來又發現根本沒有編譯器能夠支持這個關鍵字(第一個支持它的編譯器要在2002年底才問世)。自那以后,我閱讀了不少關于export的文章,了解到它幾乎不能解決用包含模式能夠解決的問題。欲知更多的export關鍵字,建議讀讀Herb Sutter撰寫的文章。

結論
要開發模板庫,就要知道模板類不是所謂的"原始類型",要用其它的編程思路。本文目的不是要嚇唬那些想進行模板編程的程序員。恰恰相反,是要提醒他們避免犯下開始模板編程時都會出現的錯誤。

posted on 2007-10-06 00:42 旅途 閱讀(183) 評論(0)  編輯 收藏 引用 所屬分類: C/C++

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美成人dvd在线视频| 激情自拍一区| 久久爱www.| 久久九九免费| 免费成年人欧美视频| 麻豆精品91| 欧美日韩国产精品自在自线| 欧美日精品一区视频| 国产乱码精品1区2区3区| 国产一区观看| 久久漫画官网| 欧美成人免费一级人片100| 欧美国产91| 在线看欧美视频| 国产精品久久久久99| 国产免费观看久久| 一区二区视频免费在线观看| 激情综合久久| 一区二区三区精品久久久| 性色av一区二区三区红粉影视| 久久中文字幕一区| 亚洲免费观看视频| 久久精品国产一区二区电影| 欧美大片在线观看一区二区| 国产日韩一区欧美| 久久午夜电影| 久久综合网色—综合色88| 亚洲免费在线看| 欧美77777| 国产视频观看一区| 夜夜精品视频| 蜜桃久久av一区| 亚洲视频在线观看一区| 麻豆精品在线观看| 国产精品久久久久久五月尺| 亚洲国产高清一区| 久久国产精品99久久久久久老狼| 亚洲国产一区视频| 性欧美video另类hd性玩具| 欧美连裤袜在线视频| 亚洲国产免费看| 六月丁香综合| 午夜宅男久久久| 国产精品色婷婷久久58| 欧美精品一区二区三区四区| 一本色道88久久加勒比精品 | 久久国产精品网站| 欧美日韩免费一区二区三区| 樱桃视频在线观看一区| 午夜一区在线| 亚洲视频高清| 欧美视频专区一二在线观看| 一区二区三区国产在线观看| 亚洲欧洲精品一区二区三区不卡 | 国产精品免费看久久久香蕉| 国产精品久久久久久亚洲毛片| 亚洲精品永久免费| 欧美大片一区二区三区| 久久精品动漫| 中文精品99久久国产香蕉| 美女黄毛**国产精品啪啪| 亚洲欧美日韩成人| 国产美女一区二区| 久久精品二区| 久久国产精品久久久久久电车| 国产一区二区成人| 老色鬼久久亚洲一区二区| 久久蜜桃av一区精品变态类天堂| 樱花yy私人影院亚洲| 欧美成人国产一区二区| 欧美成人自拍视频| 中文欧美字幕免费| 黄色成人在线网站| 国产精品午夜春色av| 亚洲一品av免费观看| 99热免费精品在线观看| 欧美日韩免费一区二区三区| 亚洲欧美日韩精品久久久| 亚洲午夜一区二区三区| 国产日韩欧美电影在线观看| 老司机午夜精品视频| 蜜桃av一区二区| 中文日韩在线| 欧美亚洲一区| 91久久精品国产91久久性色tv| 亚洲三级国产| 国产精品丝袜xxxxxxx| 鲁大师成人一区二区三区| 欧美大香线蕉线伊人久久国产精品| 亚洲精品一区二区三区99| 一区二区高清在线| 国产中文一区二区三区| 亚洲福利视频一区| 国产精品久久久久久久午夜 | 久久国产色av| 久久久噜噜噜久久人人看| aa国产精品| 欧美亚洲免费电影| 亚洲毛片在线观看.| 亚洲性夜色噜噜噜7777| 亚洲人成7777| 欧美在线一区二区三区| 夜夜嗨av一区二区三区四区| 欧美制服丝袜第一页| 一区二区三区日韩在线观看| 欧美在线免费播放| 亚洲直播在线一区| 欧美成人中文| 免费国产一区二区| 国产午夜精品理论片a级大结局| 91久久国产精品91久久性色| 狠狠狠色丁香婷婷综合激情| 亚洲视频axxx| 一本色道久久综合亚洲91| 久久久国产精品亚洲一区| 亚洲欧美综合v| 欧美喷水视频| 亚洲福利视频网| 在线日本高清免费不卡| 欧美激情自拍| 美女网站在线免费欧美精品| 国产精品久久久久久久久 | 亚洲图片在线| 亚洲精品一区二区三区蜜桃久| 销魂美女一区二区三区视频在线| 在线视频中文亚洲| 欧美好骚综合网| 欧美成人r级一区二区三区| 国产亚洲欧美一区二区| 亚洲一区二区综合| 欧美一级免费视频| 国产精品视频免费在线观看| 99视频一区二区| 一区二区三区.www| 欧美久久影院| 亚洲作爱视频| 亚洲视频一起| 国产精品网红福利| 性欧美长视频| 久久综合狠狠| 最近看过的日韩成人| 欧美二区在线播放| 亚洲日韩欧美视频| 亚洲综合日韩| 国产午夜精品福利| 久久久蜜桃一区二区人| 欧美国产激情| 亚洲午夜免费视频| 国产伦精品一区二区三区视频孕妇| 亚洲综合首页| 久久嫩草精品久久久久| 尤物yw午夜国产精品视频| 裸体一区二区三区| 亚洲精选久久| 久久爱www久久做| 伊人夜夜躁av伊人久久| 欧美黑人在线观看| 亚洲性线免费观看视频成熟| 久久精品麻豆| 日韩视频亚洲视频| 国产精品视频专区| 美国三级日本三级久久99| 亚洲精品国精品久久99热| 亚洲一区二区伦理| 国产一区在线看| 欧美黄色一级视频| 亚洲欧美日韩在线| 欧美国产亚洲另类动漫| 亚洲影视在线| 亚洲成人在线网| 欧美午夜不卡| 老鸭窝91久久精品色噜噜导演| 亚洲乱码国产乱码精品精98午夜| 欧美一区二区视频在线观看| 亚洲国产精品成人综合| 国产精品婷婷午夜在线观看| 免费看亚洲片| 午夜精品久久久久久久久| 亚洲国产成人高清精品| 久久av老司机精品网站导航| 99国产精品久久久久久久久久| 国产一区二区三区久久| 欧美三级韩国三级日本三斤| 久久天天躁夜夜躁狠狠躁2022 | 久久一二三国产| 亚洲一二三区精品| 狠狠狠色丁香婷婷综合久久五月| 免费久久99精品国产自| 欧美日韩天堂| 亚洲精品中文字| 久久久7777| 亚洲图片欧美日产| 欧美激情网友自拍| 久久久精品2019中文字幕神马| 夜夜嗨网站十八久久| 亚洲国产精品ⅴa在线观看| 国产精品亚洲成人| 欧美日韩免费在线| 欧美电影免费观看网站| 久久一区国产|