一直覺得Qt里的Model-View概念極其神秘, 因為看過很多一知半解的source code, 卻總是咋看咋不懂,急了滿頭大汗之余不禁感嘆 — 老了,腦子不夠用了!
這兩天因為在寫rssreader的關系,用到了MVC, 總算有點壓力學習學習ModelView的奧秘,而且也小有收獲。 謹以此文獻給MVC未入門的學弟學妹, 共勉!
先來講一些必備的背景知識。 在講MVC時有三個重要且基本的概念貫穿整個學習過程:Index, Data和Role。 就從Index開始。
我們見過的View有單列的List結構, 有樹狀的層次結構,還有兩維的表格結構, 歸根結底,其實這些都是層次結構的變體。 比如下面的圖:

從這張圖可以清楚的理解上文的觀點。 在這幾種結構中,都有一個隱含的根節點及與根節點聯系的層次結構。 任何一種結構中都存在這樣一個定式, 通過一個父節點及一組橫縱座標(row,column)即可唯一的確定一個子節點, 這個規律在后面會經常用到。Index可以簡單的理解成節點的指針, 前面說過通過三個要素即可唯一的確定一個節點, 所以Model提供的獲得節點index函數亦即接受row,column和parentindex三個參數, 我們在寫model時首先需要實現這樣一個函數;
第二個概念Data就更簡單了,View要顯示數據, 就要從Model中去獲取需要顯示的數據, 傳什么參數呢? 不用動腦子也想的到咯,Index肯定算一個。 但僅僅Index并不夠, 因為View要顯示的可能不止一項數據,比如我的數據包含文本, 包含圖標,包含鏈接甚至一些二進制數據, 我怎么知道View想要的是哪個呢? 這里就用到另外一個概念了 — Role, Role就用來表示View向Model索取哪個類型的數據。 View告訴Model:“我想要A節點下的N行M列數據的顯示文本; 我想要A節點下的N行M列數據的圖標…”, 這樣Model就清楚的知道應該返回什么數據了。 data()函數在這里就充當了返回數據的責任,需要我們在實現Model的時候重點實現這個函數。
目前定義好的Role可以參考下面的圖(圖中只標出了一部分Role, 其他的參見文檔DisplayRole相關章節):

作為Model必須決定為View提供多少數據,提供哪些類型的數據, 可以去滿足View的請求,也可以忽略它, 有很大的自主權。 最簡單的實現是不管什么Role都給它返回個字符串就好了。呵呵。 當然作為Model也不能太獨斷專行,因為畢竟要和View一起工作, 一定要與View的需求相配合才行。
好, 有了這些知識做基礎, 寫個Model出來其實是非常簡單的, 稍微用點心就能應付了, 首先要選對參考文檔, 如果是以寫代碼為目的, 推薦這一篇:
Creating New Models
要寫code的話這篇最實用, 前面的N多篇都在講一些概念性的內容, 大把大把的螞蟻樣的英文看了就頭大, 還是直接看這篇比較有效。 簡單來說分成幾步來做:
第一、分析需求,確定基類
先要確定你的數據是列表結構還是層次結構, 需要顯示什么樣的數據, 需不需要支持增刪或編輯功能等。 根據需求來確定從哪個Model的基類派生,如一個顯示字符串列表的Model可以采用QAbstractListModel, 樹狀層次就只能從QAbstractItemModel開始了。
第二、分析需求,確定需要實現哪些函數
根據需求的不同,需要實現的函數也不盡相同。
最簡單的只讀的列表結構只需要實現兩個基本的函數:rowCount(), data(), 也就是只需要知道一共有多少行,每行都顯示什么樣的數據即可, 十分明了吧? 多列的情況下要實現columnCount(), 需要顯示header的要去實現headerData(), 這些規則都太容易理解了。
其次,如果是層次列表,則需要確定節點之間的層次關系,就需要實現index()和parent()兩個函數, 一個是通過父指針和row,column座標確定一個子節點,一個是通過子節點知道它的父指針。
再次,如果需要修改數據, 先要通知View我的Model數據是可以被編輯的, 就是要實現flags()這個函數, 此函數返回數據的屬性,如可編輯、可被選中等; 編輯之后需要一個函數將編輯完成的數據傳遞給Model, 所以還要實現一個setData方法。
再再次, 需要增刪數據的Model還要告訴Model的底層:“我要增刪數據了!”, “我要增刪的數據是。。。”, 還有“我增刪的操作已經做完了!”, 這些分別對應:調用beginInsertRows()和endInsertRows()。 根據筆者的經驗,這部分不太好理解,而且容易出錯。 文檔里寫的是加數據的時候調用insertRows(),不過沒有提到說其實在QAbstractItemModel類里這個函數只是個空架子,根本就沒有實現, 所以你如果按照文檔去調用這個函數通知Model數據加進來了,只能得到一個return false, 不會有任何實際的作用, 很讓人困惑。 正確的做法是在你增刪數據的前后加上beginInsertRows和endInsertRows的調用,這樣底層就能正確處理數據的變化, 并且將變化及時的反應到View中。
上面提到的函數在Creating New Models這篇文章中都有具體的例子代碼可供參考,相信照著例子做一定難不倒大家。 btw,實現函數的時候要注意, 函數的聲明必須和文檔中所描述的一模一樣才能被調到, 這也是初學者經常不注意的地方。 在Qt文檔中, 下一篇就該學習如何使用View和創建自己的View了,這部分還有待研究。 等研究好了再撰文匯報!
原文鏈接:http://www.cuteqt.com/blog/?p=218