1. 程式人生 > >淺析在QtWidget中自定義Model(beginInsertRows()和endInsertRows()是空架子,類似於一種信號,用來通知底層)

淺析在QtWidget中自定義Model(beginInsertRows()和endInsertRows()是空架子,類似於一種信號,用來通知底層)

cti ron 初學者 開發 http 沒有 insert ati 學習

Qt 4推出了一組新的item view類,它們使用model/view結構來管理數據與表示層的關系。這種結構帶來的功能上的分離給了開發人員更大的彈性來定制數據項的表示,它也提供一個標準的model接口,使得更多的數據源可以被這些item view使用。這裏對model/view的結構進行了描述,結構中的每個組件都進行了解釋.。

一直覺得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,實現函數的時候要註意, 函數的聲明必須和文檔中所描述的一模一樣才能被調到, 這也是初學者經常不註意的地方。

小結:QtWidget中自定義Model的內容介紹完了,希望本篇對你有幫助。Model與數據源通訊,並提供接口給結構中的別的組件使用。通訊的性質依賴於數據源的種類
model實現的方式。

轉自:http://mobile.51cto.com/symbian-270166.htm

http://www.cnblogs.com/bingcaihuang/archive/2011/07/24/2115612.html

淺析在QtWidget中自定義Model(beginInsertRows()和endInsertRows()是空架子,類似於一種信號,用來通知底層)