文档章节

利用C++11的function和bind功能,实现QStandardItemModel的通用遍历函数

尘中远
 尘中远
发布于 2016/05/12 23:51
字数 1153
阅读 18
收藏 0
在使用Qt的树形视图和表格视图QTableView和QTreeView时,经常需要遍历所有条目,每种功能都写一个遍历函数既麻烦又不符合编程最简原则,因此,写一个通用的遍历函数是很必要的(类似于std::for_each),对于遍历时实现的功能不一样,可以通过函数指针来代替需要实现的功能。

在讲这个之前,需要了解 std::function和std::bind,具体可以自行谷歌。

由于这个遍历函数可能经常用到,因此可以写一个类的静态函数或者是全局函数。

下面把这个功能封装在一个类里

类的头文件如下:
#include <QStandardItemModel>
#include <functional>
class QStandardItemModelEx
 {
 public:
     QStandardItemModelEx(){}
     ~QStandardItemModelEx(){}
     ///
     /// \brief callback_ergodicFun_ptr 回调函数指针,bool f(QStandardItem*),bool用于决定是否继续,如果为true就继续递归,如果为false就停止递归
     ///
     typedef std::function<bool(QStandardItem*)> callback_ergodicFun_ptr;

     //typedef void(*callback_ergodicFun_ptr)(QStandardItem*);
     static void ergodicAllItem(QStandardItemModel* model,callback_ergodicFun_ptr pFun);
     static bool ergodicItem(QStandardItem* item,callback_ergodicFun_ptr pFun);
 private:
 };

这里使用C++11的新特性function(环境 VS2010),定义一个函数指针用于回调实现功能。
函数 ergodicAllItem用于遍历所有项目, ergodicItem用于遍历项目下的所有子项目。回调函数返回bool用于决定是否继续,如果为true就继续遍历,如果为false就停止遍历
为了简单起见,下面用递归实现遍历( ergodicItem是尾递归函数
 
void QStandardItemModelEx::ergodicAllItem(QStandardItemModel* model,callback_ergodicFun_ptr pFun)
{
    int rows = model->rowCount();
    int column = model->columnCount();
    for (int i=0;i<rows;++i)
    {
        for(int j=0;j<column;++j)
        {
            QStandardItem* item = model->item(i,j);
            if (item)
            {
                if(!ergodicItem(item,pFun))
                    return;
            }
        }
    }
}

bool QStandardItemModelEx::ergodicItem(QStandardItem* item,callback_ergodicFun_ptr pFun)
{
    int rows = item->rowCount();
    int column = item->columnCount();
    for (int i=0;i<rows;++i)
    {
        for(int j=0;j<column;++j)
        {
            QStandardItem* childItem = item->child(i,j);
            if (childItem)
            {
                if(!ergodicItem(childItem,pFun))
                    return false;
            }
        }
    }
    return pFun(item);
}




用这两个函数就可以遍历所有的项目了,而遍历时需要的动作,就通过回调函数实现。

回调函数即可写在类里,也可以写为全局函数,若除了 QStandardItem * 还需要别的参数,那么可以用std::bind帮忙。

下面举一个例子
例子功能是通过关键字高亮条目。在使用QTreeView或QTableView时,在条目多的情况下,需要搜索某些条目,对搜索结果进行高亮,这时,就需要遍历整个View里的条目,对符合的条目进行高亮
遍历的函数已经写好,现在缺少的是回调的函数,由于遍历的函数使用的函数指针是 std : : function ,因此回调函数即可写为全局函数也可以写为类成员函数
回调函数声明:
bool callback_hightLightItem(QStandardItem* item,const QStringList keys);
回调函数实现:
bool callback_hightLightItem(QStandardItem* item,const QStringList keys)
{   
    QString str = item->text();
    if (is_match_string(str,keys))
    {
        item->setData(QVariant(QColor(237,100,100,180)),Qt::BackgroundRole);
    }
    else
    {
        item->setData(QVariant(),Qt::BackgroundRole);
    }
    return true;
}

使用QStandardItem的setData函数,设置角色为Qt::BackgroundRole,即可设置背景颜色。
实现函数中is_match_string函数是用来检测是否符合关键字,如果符合关键字,就返回true,符合关键字就给条目背景赋予不同的颜色,否则就把颜色消除。
回调函数返回true,意味着一直遍历,直到所有遍历完成。

通过按钮pushButton_search来触发搜索功能。 pushButton_search函数的槽如下:
void MainWindow : :on_pushButton_search_clicked()
{
    QString str  = ui - >lineEdit_search - >text(); //获取关键字
     if (str.isEmpty())
         return;
    QStringList keyWords  = str.split(QString( " "));
    StandardItemModel : :ergodicAllItem(
        qobject_cast <QStandardItemModel * >(ui - >treeView - >model())
        ,std : :bind(callback_hightLightItem,std : :placeholders : :_1,keyWords)); //对于callback_hightLightItem是全局函数的情况下
}

由于回调函数是两个参数QStandardItem* item,const QStringList keys,而函数指针只是一个参数,因此需要绑定一个参数,使其变为参数只有一个的函数指针。这时就需要std::bind来实现。
占位符_1表示这个参数是对应的函数指针的第一个参数所在位置,如当前函数是:
bool callback_hightLightItem(QStandardItem* item,const QStringList keys);
我们要把他转换为如下类型的函数指针
bool(QStandardItem*)
第一个参数是正好对应,所以占位符_1就位于第一个位置,后面的参数直接传递进去就行。

如果回调写成:
bool callback_hightLightItem(const QStringList keys,QStandardItem* item);
那么写法变为:
std::bind(callback_hightLightItem,keyWords,std::placeholders::_1);
如果callback_hightLightItem是类成员函数,那么其写法如下
StandardItemModel : :ergodicAllItem(
        qobject_cast <QStandardItemModel * >(ui - >treeView - >model())
        ,std : :bind( &MainWindow : :callback_hightLightItem, this,std : :placeholders : :_1,keyWords));

bind也能轻松搞定类成员函数指针,注意要把类的指针传递过去。

© 著作权归作者所有

共有 人打赏支持
尘中远
粉丝 1
博文 26
码字总数 47436
作品 0
朝阳
程序员
c++11实现异步定时器

c++11提供了丰富的时间和线程操作函数,比如 std::this_thread::sleep, std::chrono::seconds等。可以利用这些来很方便的实现一个定时器。 实现 ifndef TIMERH define TIMERH include includ...

moki_oschina
05/23
0
0
(转)现代C++函数式编程

本文转自:http://geek.csdn.net/news/detail/96636 {public: { 评论 添加评论 发布链接 发布图文 返回顶部 发布到 主题 发布 评论

wangxiaocvpr
2016/08/22
0
0
PyQt5系列教程(30): 文本输入栏(QLineEdit)4

这期我们继续介绍一下文本输入栏(QLineEdit),我们已经介绍了其常用属性、信号、函数还有自定义密码输入框,今天是最后一章,文本框自动补全。 若我们想用到QLineEdit自动补全的功能,必须...

学点编程吧
03/02
0
0
使用C++11封装线程池ThreadPool

读本文之前,请务必阅读: 使用C++11的function/bind组件封装Thread以及回调函数的使用 Linux组件封装(五)一个生产者消费者问题示例 线程池本质上是一个生产者消费者模型,所以请熟悉这篇文...

inevermore
2015/03/19
0
0
c++11特性与cocos2d-x 3.0之std::bind与std::function

原文地址:http://www.cnblogs.com/slysky/p/3822640.html 作者:阳光下的蒲公英 c++11特性与cocos2d-x 3.0之std::bind与std::function 昨天同事让帮忙写一小功能,才发现cocos2d-x 3.0 和 ...

智捷课堂
2015/08/21
0
0

没有更多内容

加载失败,请刷新页面

加载更多

shell特殊符号、cut、sort、uniq、wc、tee、tr、split命令

10月15日任务 8.10 shell特殊符号cut命令 8.11 sort_wc_uniq命令 8.12 tee_tr_split命令 8.13 shell特殊符号下 cut 命令 cut作用:截取字符串 用法如下:cat /etc/passwd |head -2 |cut -d ...

hhpuppy
10分钟前
0
0
Springboot实现filter拦截token验证和跨域

背景 web验证授权合法的一般分为下面几种 1使用session作为验证合法用户访问的验证方式 使用自己实现的token 使用OCA标准 在使用API接口授权验证时,token是自定义的方式实现起来不需要引入其...

funnymin
44分钟前
1
0
linux使用ntfs-3g操作ntfs格式硬盘

Linux内核目前只支持对微软NTFS文件系统的读取。 NTFS-3G 是微软 NTFS 文件系统的一个开源实现,同时支持读和写。NTFS-3G 开发者使用 FUSE 文件系统来辅助开发,同时对可移植性有益。 安装 ...

linuxprobe16
今天
1
0
kubeadm部署kubernetes集群

一、环境要求 这里使用RHEL7.5 master、etcd:192.168.10.101,主机名:master node1:192.168.10.103,主机名:node1 node2:192.168.10.104,主机名:node2 所有机子能基于主机名通信,编辑...

人在艹木中
今天
13
0
Shell特殊符号总结以及cut,sort,wc,uniq,tee,tr,split命令

特殊符号总结一 * 任意个任意字符 ? 任意一个字符 # 注释字符 \ 脱义字符 | 管道符 # #号后的备注被忽略[root@centos01 ~]# ls a.txt # 备注 a.txt[root@centos01 ~]# a=1[root@centos01...

野雪球
今天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部