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

首頁 > 編程 > C++ > 正文

C++實現 vector 的四則運算

2020-05-23 14:01:50
字體:
來源:轉載
供稿:網友

這里假設 vector 的運算定義為對操作數 vector 中相同位置的元素進行運算,最后得到一個新的 vector。具體來說就是,假如 vector<int> d1{1, 2, 3}, d2{4, 5, 6};則, v1 + v2 等于 {5, 7, 9}。實現這樣的運算看起來并不是很難,一個非常直觀的做法如下所示:

vector<int> operator+(const vector<int>& v1, const vector<int>& v2) {  // 假設 v1.size() == v2.size()  vector<int> r;  r.reserve(v1.size());  for (auto i = 0; i < v1.size(); ++i) {    r.push_back(v1[i] + v2[i]);  }  return r;}

// 同理,需要重載其它運算符
我們針對 vector 重載了每種運算符,這樣一來,vector 的運算就與一般簡單類型無異,實現也很直白明了,但顯然這個直白的做法有一個嚴重的問題:效率不高。效率不高的原因在于整個運算過程中,每一步的運算都產生了中間結果,而中間結果是個 vector,因此每次都要分配內存,如果參與運算的 vector 比較大,然后運算又比較長的話,效率會比較低,有沒有更好的做法呢?

既然每次運算產生中間結果會導致效率問題,那能不能優化掉中間結果?回過頭來看,這種 vector 的加減乘除與普通四則運算并無太大差異,在編譯原理中,對這類表達式進行求值通??梢酝ㄟ^先把表達式轉為一棵樹,然后通過遍歷這棵樹來得到最后的結果,結果的計算是一次性完成的,并不需要保存中間狀態,比如對于表達式:v1 + v2 * v3,我們通??梢韵葘⑵滢D化為如下樣子的樹:

c語言實現四則運算,大數四則運算的c,實現,c,vector

因此求值就變成一次簡單的中序遍歷,那么我們的 vector 運算是否也可以這樣做呢?

表達式模板

要把中間結果去掉,關鍵是要推遲對表達式的求值,但 c++ 不支持 lazy evaluation,因此需要想辦法把表達式的這些中間步驟以及狀態,用一個輕量的對象保存起來,具體來說,就是需要能夠將表達式的中間步驟的操作數以及操作類型封裝起來,以便在需要時能動態的執行這些運算得到結果,為此需要定義類似如下這樣一個類:

enum OpType {  OT_ADD,  OT_SUB,  OT_MUL,  OT_DIV,};class VecTmp {  int type_;  const vector<int>& op1_;  const vector<int>& op2_;public:  VecTmp(int type, const vector<int>& op1, const vector<int>& op2)    : type_(type), op1_(op1), op2_(op2) {}  int operator[](const int i) const {    switch(type_) {      case OT_ADD: return op1_[i] + op2_[i];      case OT_SUB: return op1_[i] - op2_[i];      case OT_MUL: return op1_[i] * op2_[i];      case OT_DIV: return op1_[i] / op2_[i];      default: throw "bad type";    }  }};

有了這個類,我們就可以把一個簡單的運算表達式的結果封裝到一個對象里面去了,當然,我們得先將加法操作符(以及其它操作符)重載一下:

VecTmp operator+(const vector<int>& op1, const vector<int>& op2) {  return VecTmp(OT_ADD, op1, op2);}

這樣一來,對于 v1 + v2,我們就得到了一個非常輕量的 VecTmp 對象,而該對象可以很輕松地轉化 v1 + v2 的結果(遍歷一遍 VecTmp 中的操作數)。但上面的做法還不能處理 v1 + v2 * v3 這樣的套嵌的復雜表達式:v2 * v3 得到一個 VecTmp,那 v1 + VecTmp 怎么搞呢?

同理,我們還是得把 v1 + VecTmp 放到一個輕量的對象里,因此最好我們的 VecTmp 中保存的操作數也能是 VecTmp 類型的,有點遞歸的味道。。。用模板就可以了,于是得到如下代碼:

#include <vector>#include <iostream>using namespace std;enum OpType {  OT_ADD,  OT_SUB,  OT_MUL,  OT_DIV,};template<class T1, class T2>class VecSum {    OpType type_;  const T1& op1_;  const T2& op2_; public:  VecSum(int type, const T1& op1, const T2& op2): type_(type), op1_(op1), op2_(op2) {}     int operator[](const int i) const {      switch(type_) {        case OT_ADD: return op1_[i] + op2_[i];        case OT_SUB: return op1_[i] - op2_[i];        case OT_MUL: return op1_[i] * op2_[i];        case OT_DIV: return op1_[i] / op2_[i];        default: throw "bad type";      }    }};template<class T1, class T2>VecSum<T1, T2> operator+(const T1& t1, const T2& t2) {  return VecSum<T1, T2>(OT_ADD, t1, t2);}template<class T1, class T2>VecSum<T1, T2> operator*(const T1& t1, const T2& t2) {  return VecSum<T1, T2>(OT_MUL, t1, t2);}int main() {  std::vector<int> v1{1, 2, 3}, v2{4, 5, 6}, v3{7, 8, 9};  auto r = v1 + v2 * v3;  for (auto i = 0; i < r.size(); ++i) {    std::cout << r[i] << " ";  }}

上面的代碼漂亮地解決了前面提到的效率問題,擴展性也很好而且對 vector 來說還是非侵入性的,雖然實現上乍看起來可能不是很直觀,除此也還有些小問題可以更完善些:

操作符重載那里很可能會影響別的類型,因此最好限制一下,只針對 vector 和 VecTmp 進行重載,這里可以用 SFINAE 來處理。

VecTmp 的 operator[] 函數中的 switch 可以優化掉,VecTmp 模板只需增加一個參數,然后對各種運算類型進行偏特化就可以了。

VecTmp 對保存的操作數是有要求的,只能是 vector 或者是 VecTmp<>,這里也應該用 SFINAE 強化一下限制,使得用錯時出錯信息好看些。

現在我們來重頭再看看這一小段奇怪的代碼,顯然關鍵在于 VecTmp 這個類,我們可以發現,它的接口其實很簡單直白,但它的類型卻可以是那么地復雜,比如說對于 v1 + v2 * v3 這個表達式,它的結果的類型是這樣的: VecTmp<vector<int>, VecTmp<vector<int>, vector<int>>>,如果表達式再復雜些,它的類型也就更復雜了,如果你看仔細點,是不是還發現這東西和哪里很像?像一棵樹,一棵類型的樹。

c語言實現四則運算,大數四則運算的c,實現,c,vector

這棵樹看起來是不是還很眼熟,每個葉子結點都是 vector,而每個內部結點則是由 VecTmp 實例化的:這是一棵類型的樹,在編譯時就確定了。這種通過表達式在編譯時得到的復雜類型有一個學名叫: Expression template。在 c++ 中每一個表達式必產生一個結果,而結果必然有類型,類型是編譯時的東西,結果卻是運行時的。像這種運算表達式,它的最終類型是由其中每一步運算所產生的結果所對應的類型組合起來所決定的,類型確定的過程其實和表達式的識別是一致的。

VecTmp 對象在邏輯上其實也是一棵樹,它的成員變量 op1_, op2_ 則分別是左右兒子結點,樹的內部結點代表一個運算,葉子結點則為操作數,一遍中序遍歷下來,得到的就是整個表達式的值。

神奇的 boost::proto

expression template 是個好東西(就正如 expression SFINAE 一樣),它能幫助你在編譯時建立非常復雜好玩的類型系統(從而實現很多高級玩意,主要是函數式)。但顯然如果什么東西都需要自己從頭開始寫,這個技術用起來還是很麻煩痛苦的,好在模板元編程實在是個太好玩的東西,已經有很多人做了很多先驅性的工作,看看 boost proto 吧,在 c++ 的世界里再打開一扇通往奇怪世界的大門


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久成年人视频| 亚洲自拍av在线| 欧美理论电影在线播放| 在线观看久久av| 日本久久91av| 亚洲精品综合精品自拍| 97视频在线观看视频免费视频| 亚洲国产婷婷香蕉久久久久久| 欧美精品18videos性欧美| 精品国产福利在线| 久久久久www| 琪琪第一精品导航| 亚洲激情中文字幕| 久久成人精品视频| 精品视频久久久久久久| 久久成人国产精品| 成年人精品视频| 欧美国产日韩一区二区三区| 国模吧一区二区| 97涩涩爰在线观看亚洲| 欧美激情精品久久久久久蜜臀| 国产精品v片在线观看不卡| 日韩黄在线观看| 国内精品美女av在线播放| 久久69精品久久久久久久电影好| 亚洲成人av资源网| 性欧美xxxx交| 国产suv精品一区二区三区88区| 欧美电影在线观看网站| 蜜月aⅴ免费一区二区三区| 色狠狠av一区二区三区香蕉蜜桃| 两个人的视频www国产精品| 一个人看的www欧美| 亚洲第一区中文99精品| 欧美日韩爱爱视频| 久久精品电影一区二区| 亚洲free性xxxx护士白浆| 国产亚洲美女精品久久久| 深夜福利一区二区| 亚洲高清av在线| 亚洲人成在线播放| 亚洲第一网站免费视频| 欧美猛交ⅹxxx乱大交视频| 亚洲女人被黑人巨大进入al| 97成人在线视频| 亚洲精品乱码久久久久久按摩观| 欧美激情奇米色| 日韩专区在线播放| 少妇精69xxtheporn| 国产精品久久久久久av福利软件| 欧美野外猛男的大粗鳮| 欧美精品情趣视频| 动漫精品一区二区| 一区二区中文字幕| 精品久久久久久久久久久久久| 亚洲国产精品成人av| 色爱精品视频一区| 亚洲激情视频在线播放| 日韩精品小视频| 91国在线精品国内播放| 欧美精品久久久久久久久久| 91精品久久久久久久久久久久久| 亚洲自拍在线观看| 91av在线播放视频| 亚洲第一色中文字幕| 亚洲国产又黄又爽女人高潮的| 久久综合免费视频| 欧美视频在线免费| 国产在线拍揄自揄视频不卡99| 日本久久久久久久久久久| 久久久久亚洲精品国产| 奇米成人av国产一区二区三区| 国产99久久精品一区二区 夜夜躁日日躁| 91精品国产高清自在线看超| 日韩欧美在线一区| 久久国产精品视频| 国产91网红主播在线观看| 欧美激情在线有限公司| 亚洲男人第一网站| 欧美综合在线观看| 俺去啦;欧美日韩| 欧美日韩另类字幕中文| 欧美日韩在线免费观看| 亚洲人成在线一二| 国产精品视频一区二区高潮| 国产91色在线|| 一本色道久久综合亚洲精品小说| 国产suv精品一区二区| 性色av一区二区三区在线观看| 色播久久人人爽人人爽人人片视av| 欧美激情视频一区| 欧美日韩在线免费观看| 色综合久综合久久综合久鬼88| 一本色道久久综合亚洲精品小说| 亚洲欧洲一区二区三区在线观看| 91精品国产高清自在线看超| 国产精品高潮呻吟久久av野狼| 热re99久久精品国产66热| 国产精品一区二区久久国产| 国产精品一区二区三区免费视频| 欧美在线精品免播放器视频| 久久综合免费视频影院| 中国china体内裑精亚洲片| 这里只有精品视频在线| 欧美夫妻性生活视频| 日韩欧美中文字幕在线播放| 亚洲小视频在线| 九九九热精品免费视频观看网站| 国产精品久久久久久久久久新婚| 亚洲人成电影网站色xx| 久久久噜噜噜久久久| 久久影院中文字幕| 一区二区三区视频在线| 亚洲人av在线影院| 亚洲free性xxxx护士白浆| 国产精品久久久| 欧美精品一本久久男人的天堂| 久久中国妇女中文字幕| 精品久久久一区二区| 97国产suv精品一区二区62| 亚洲色图35p| 91在线视频九色| 日韩精品在线免费观看视频| 在线播放亚洲激情| 青草青草久热精品视频在线观看| 日韩精品免费综合视频在线播放| 中文字幕亚洲欧美| 福利视频第一区| 色偷偷亚洲男人天堂| 一区三区二区视频| 精品日韩美女的视频高清| 日本高清视频精品| 91精品国产91久久久久久吃药| 国产日韩精品在线观看| 91在线观看免费网站| 欧洲午夜精品久久久| 性色av一区二区三区免费| 亚洲在线免费视频| 亚洲日韩中文字幕在线播放| 成人激情视频小说免费下载| 国产乱人伦真实精品视频| 精品网站999www| 国产精品夫妻激情| 国产欧美精品一区二区| 国产精品高清在线观看| 一区二区三区回区在观看免费视频| 2021久久精品国产99国产精品| 91精品综合久久久久久五月天| 亚洲一区二区三区在线视频| 色视频www在线播放国产成人| 欧美乱大交xxxxx| 亚洲国产精品高清久久久| 久久久久久亚洲| 精品色蜜蜜精品视频在线观看| 2021久久精品国产99国产精品| 亚洲欧美国产精品久久久久久久| 国产日韩欧美视频在线| 国产精品偷伦一区二区| 国产精品精品国产| 亚洲欧美国产一区二区三区| 欧美精品成人91久久久久久久| 亚洲欧美三级伦理| 亚洲在线www| 韩剧1988免费观看全集|