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

旅途

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

如何組織編寫模板程序 轉

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

[轉]
前言
常遇到詢問使用模板到底是否容易的問題,我的回答是:“模板的使用是容易的,但組織編寫卻不容易”。看看我們幾乎每天都能遇到的模板類吧,如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>
            99re热精品| 欧美午夜激情小视频| 亚洲电影在线免费观看| 亚洲一区久久久| 亚洲在线播放| 欧美亚洲综合在线| 久久精品99国产精品日本| 久久久国产精品一区二区中文| 久久蜜臀精品av| 欧美成人精品一区二区三区| 亚洲黄色性网站| 国产精品嫩草影院av蜜臀| 亚洲调教视频在线观看| 亚洲一区尤物| 久久国产精品亚洲77777| 久久久久国色av免费看影院| 免费中文日韩| 国产精品一区二区久激情瑜伽| 韩国av一区二区三区在线观看 | 欧美—级高清免费播放| 欧美国产激情二区三区| 欧美婷婷久久| 在线日韩一区二区| 亚洲一区二区少妇| 欧美成人69| 亚洲综合电影| 欧美韩日一区二区| 国产日本欧美在线观看| 亚洲精品视频在线观看网站| 欧美一区二区成人| 亚洲卡通欧美制服中文| 欧美一区日本一区韩国一区| 欧美麻豆久久久久久中文| 国产一区二区剧情av在线| 一区二区三区不卡视频在线观看| 久久婷婷久久| 亚洲欧美日韩天堂| 欧美女激情福利| 亚洲高清123| 久久国产精品久久久久久| 亚洲美女在线观看| 免费视频亚洲| 激情久久婷婷| 久久久999| 亚洲男人的天堂在线aⅴ视频| 欧美激情精品久久久久| 亚洲黑丝在线| 欧美激情第1页| 久久综合999| 在线观看视频欧美| 久久久亚洲成人| 欧美一区二区黄| 国产欧美亚洲日本| 午夜精品久久久久久久久久久| 亚洲精品小视频| 欧美精品一区视频| 亚洲精品免费在线观看| 欧美国产日产韩国视频| 免费在线观看一区二区| 亚洲第一页在线| 欧美黄色小视频| 免费一区二区三区| 亚洲精品久久久久| 亚洲人成亚洲人成在线观看图片 | 亚洲欧美一区在线| 国产精品久久久久久久久婷婷| 一本色道精品久久一区二区三区 | 欧美成人自拍| 在线观看精品视频| 美女被久久久| 免费观看一级特黄欧美大片| 亚洲人成网站在线观看播放| 亚洲电影自拍| 欧美日韩成人在线播放| 亚洲无玛一区| 欧美亚洲自偷自偷| 伊人久久成人| 亚洲高清久久网| 欧美日韩精品系列| 欧美亚洲日本国产| 久久久xxx| 一片黄亚洲嫩模| 亚洲欧美日韩一区二区| 在线高清一区| 99精品国产热久久91蜜凸| 国产精品久久久久久久午夜片 | 国产九区一区在线| 久久久久国内| 免费中文字幕日韩欧美| 在线视频亚洲| 久久aⅴ国产欧美74aaa| 亚洲欧洲精品一区| 亚洲一区国产| 亚洲精品久久久久| 亚洲欧美激情视频在线观看一区二区三区 | 免费一区视频| 亚洲女同精品视频| 久久午夜精品一区二区| 亚洲一区二区三区三| 久久久久久久999精品视频| 99国产精品视频免费观看一公开 | 欧美三级日本三级少妇99| 久久国产精品亚洲va麻豆| 欧美成人一区在线| 欧美在线亚洲| 欧美日韩日本国产亚洲在线| 巨乳诱惑日韩免费av| 国产精品国产三级国产| 欧美激情按摩| 国产一二三精品| 中文一区在线| 亚洲色图制服丝袜| 美女性感视频久久久| 欧美主播一区二区三区美女 久久精品人| 久久久久久噜噜噜久久久精品| 亚洲婷婷综合久久一本伊一区| 久久亚洲精品欧美| 午夜一区二区三视频在线观看| 欧美激情久久久久| 欧美电影在线播放| 国模私拍视频一区| 欧美一级成年大片在线观看| 女同一区二区| 久久综合色婷婷| 国产亚洲精品久久久久久| 亚洲深夜激情| 亚洲一区二区三区四区五区午夜 | 欧美日韩一区二区三区在线视频| 欧美a级在线| 黄色日韩网站视频| 欧美中文字幕在线播放| 久久国产一二区| 国产欧美成人| 亚洲欧美一区二区视频| 亚洲男人av电影| 国产精品igao视频网网址不卡日韩| 亚洲国产精品久久久久秋霞蜜臀 | 欧美日韩精品一本二本三本| 亚洲日本成人| 99精品视频免费观看| 欧美激情五月| 91久久精品美女| 99re6热在线精品视频播放速度| 玖玖玖免费嫩草在线影院一区| 欧美粗暴jizz性欧美20| 亚洲精品久久视频| 欧美日韩成人一区| 中文日韩在线视频| 久久精品女人| 亚洲电影中文字幕| 欧美福利一区二区| 一本色道久久加勒比88综合| 欧美一区二区成人| 在线精品视频一区二区三四| 蜜桃av一区二区| 日韩一级成人av| 亚洲一区二区三区视频| 国产精品亚洲网站| 久久久水蜜桃av免费网站| 亚洲国产精品精华液2区45| 日韩小视频在线观看| 国产精品久久久久久久久免费樱桃| 香蕉免费一区二区三区在线观看| 久久全国免费视频| 亚洲日韩成人| 国产精品日韩在线| 久久久亚洲国产天美传媒修理工| 亚洲第一综合天堂另类专| 亚洲一区二区三区在线播放| 国产日韩一区二区三区在线播放| 久久久久久自在自线| 日韩午夜激情| 免费国产一区二区| 亚洲免费在线电影| 亚洲国产成人不卡| 国产精品视频久久久| 老司机午夜精品视频| 亚洲视频在线观看| 欧美成人午夜影院| 亚洲在线成人| 亚洲高清视频中文字幕| 欧美+亚洲+精品+三区| 亚洲精品视频一区| 久久久91精品国产一区二区精品| 亚洲第一精品影视| 久久xxxx| 正在播放日韩| 亚洲高清色综合| 国产精品一区二区在线观看网站| 裸体丰满少妇做受久久99精品| 亚洲性视频h| 亚洲精品视频免费| 免费不卡在线视频| 久久国产精品久久久久久| 中文日韩在线视频| 最近中文字幕mv在线一区二区三区四区| 国产精品卡一卡二卡三| 老司机精品导航| 久久xxxx精品视频| 亚洲淫性视频|