浅析在QtWidget中自定义Model(beginInsertRows()和endInsertRows()是空架子,类似于一种信号,用来通知底层)
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()是空架子,类似于一种信号,用来通知底层)的更多相关文章
- 浅析在QtWidget中自定义Model
Qt 4推出了一组新的item view类,它们使用model/view结构来管理数据与表示层的关系.这种结构带来的功能上的分离给了开发人员更大的弹性来定制数据项的表示,它也提供一个标准的model接 ...
- QT内省机制、自定义Model、数据库
本文将介绍自定义Model过程中数据库数据源的获取方法,我使用过以下三种方式获取数据库数据源: 创建 存储对应数据库所有字段的 结构体,将结构体置于容器中返回,然后根据索引值(QModelIndex) ...
- iOS自定义model排序
在开发过程中,可能需要按照model的某种属性排序. 1.自定义model @interface Person : NSObject @property (nonatomic,copy) NSStri ...
- Qt自定义model
前面我们说了Qt提供的几个预定义model.但是,面对变化万千的需求,那几个model是远远不能满足我们的需要的.另外,对于Qt这种框架来说,model的选择首先要能满足绝大多数功能的需要,这就是说, ...
- Sails 自定义 model 方法
Sails 自定义 model 方法 在 Sails 中 model 提供了一些原生的静态方法,如 .create(), .update(), .destroy(), .find(), 等. 在实际业 ...
- Django中的Model(字段)
Model Django中的model是用来操作数据库的,Model是一个ORM框架,我们只需要关心model的操作,而不需要关心到底是哪一种数据库. 一.基本知识: 数据库引擎: Django中自带 ...
- Django中的Model继承
Django 中的 model 继承和 Python 中的类继承非常相似,只不过你要选择具体的实现方式:让父 model 拥有独立的数据库:还是让父 model 只包含基本的公共信息,而这些信息只能由 ...
- ASP.NET MVC4中的Model验证 移除指定验证信息
MVC中通过Model在页面间传值使的程序开发变得更加的快捷,但是很多时候,我们在数据传递的时候为了确保数据的有效性,要对Model的相关属性做基本的数据验证. 不多说直接上个代码,Model的实体类 ...
- 浅析Thinkphp框架中运用phprpc扩展模式
浅析Thinkphp框架中应用phprpc扩展模式 这次的项目舍弃了原来使用Axis2做web服务端的 方案,改用phprpc实现,其一是服务端的thinkphp已集成有该模式接口,其二是phprpc ...
随机推荐
- ssh连接上腾讯云、华为云Linux服务器,一会就自动断开
客户端向服务端发送心跳 依赖 ssh 客户端定时发送心跳,putty.SecureCRT.XShell 都有这个功能. Linux / Unix 下,编辑 ssh 配置文件: # vim /etc/s ...
- js进阶 9-5 js如何确认form的提交和重置按钮
js进阶 9-5 js如何确认form的提交和重置按钮 一.总结 一句话总结: 1.这个并不好做:onsubmit 里面的代码必须返回false才能取消onsubmit方法的执行,所以,有return ...
- 百度UEditor上传图片-再再总结一次
本周,CSDN有个网友遇到了百度UEditor上传问题,最后商定付50元钱,我帮他解决这个问题. 他最初想自己搞定这个问题,结果搞了好多次,好几天,还是没能解决. 2015年1月17日8: ...
- 电子商务系统的设计与实现(十):DWZ框架与第三方分页组件整合
晚上,就是刚刚,在后端管理系统中使用DWZ框架. 先是,直接使用官网网站的Demo,dwz-jui,与编程语言无关的纯静态的那个原始项目. 很快就搭建好了左侧菜单,打开菜单后,出现Tab页面,然后显示 ...
- 【32.26%】【codeforces 620C】Pearls in a Row
time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...
- CentOS虚拟机克隆后IP设置
1.网卡设置 vim /etc/udev/rules.d/70-persistent-net.rules 将新生成的eth1改名为eth0,并将原来的eth0注释或删除,同时复制mac地址 2.IP设 ...
- cocos2d-x 调色
在游戏开发.我们须要实现闪光的灯.照明弹效果等等,我么你能够採用混合模式来实现. 假设学习过OpenGL(ES),就知道里面使用glBlendFunc函数实现的.在cocos2d-x里肯定也有,对于精 ...
- Struts2——(3)ValueStack(值栈)
一.ValueStack 主要用于存储请求相关信息,内部结构如下 root区:被称为根存储区,是一个栈结构,栈顶元素为当前请求的Action对象. context区:被称为变量存储区,是一个Map结构 ...
- asp.net中c#求百分比
double m= 50;double n= 100; Response.Write((m/ (m+ n)).ToString("0%"));Response.Write((m/ ...
- IOC介绍及其简单实现
预备知识: Java反射原理,XML及其解析 IOC:Inversion of Control,控制反转,它最主要反映的是与传统面向对象(OO)编程的不同.通常我们编程实现某种功能都需要几个对象相 ...