专栏电商日志财经减肥爱情
投稿投诉
爱情常识
搭配分娩
减肥两性
孕期塑形
财经教案
论文美文
日志体育
养生学堂
电商科学
头戴业界
专栏星座
用品音乐

QCustomplot使用分享加载cvs文件

  一、概述
  之前做过一款金融产品,名字叫做财联社,感兴趣的可以瞅一眼财联社产品展示,由于需要画复杂的k线图和一些辅助的图表,我个人调研了几个绘图库,包括:QWt、QCustomPlot、QtChart和directUI。最后各方考虑,决定使用QCustomPlot来做我们的基础绘图库,这里有几个方面的考虑首先QCP他是开源的代码只有2个文件,比较方便的可以引入我们的现有的代码代码可读性比较强,定制方便
  当我们的绘图库选定后,理所当然的就是去研究我们这个库了,因此我也花了几天的时间去研究了我们这个绘图库,并做了一个简单的demo,感兴趣的可以去看之前写的文章,demo都在CSDN上放着,如果没有分需要我发的可以留言。
  之前讲解的文章我在后边相关文章小节已经给出,有想法的同学也可以直接先去看之前的文章,这样更容易理解二、效果图
  如下图所示,是我做的一个测试效果图,途中包括一个简单的折线图和游标,折线图的显示模式有十几种效果,具体可以看QCustomplot使用分享(一)能做什么事这篇文章里的截图,这里我就不在贴出。
  这个效果图只是展示了一部分简单的功能,我封装的绘图控件实际上主要是用于用于加载cvs文件,然后显示相应的图表,当然了,如果你想自己获取数据添加给图表也是支持的。
  最后该绘图控件还提供了很多接口,可以获取当前绘图数据,比如:游标对于的x值、y值,最多提供了2个游标获取两个游标之间的x值数据段获取两个游标之间的y值数据段,并且可以指定折线图设置折线图颜色设置折线图类型设置4个轴的标题栏名称设置游标颜色
  下面的文章中我会分析下主要的接口和核心功能实现
  图中的展示效果测试代码如下,代码中的关键节点就2个构造ESCvsDBOperater类,并加载cvs文件通过Set接口设置数据,并设置折线图类型ESCsvDBOperatercsvDBOperaternewESCsvDBOperater(nullptr);csvDBOperaterloadCSVFile(qAppapplicationDirPath()temptest31。csv);QStringListnamescsvDBOperatergetCSVNames();autocallback〔this,names〕(constQStringname,constQVectordoubledata){intindexnames。indexOf(name);if(index!1){if(index0){uiwidgetSetGraphKey(data);}else{intlname。indexOf(();intrname。indexOf());if(l!1r!1){uiwidgetSetGraphValue(index1,name。left(l),name。mid(l1,rl1),data);uiwidgetSetGraphScatterStyle(index1,4);}else{uiwidgetSetGraphValue(index1,name,,data);}}}
  当然QCP不仅仅能显示折线图,他还可以显示各种各样的效果图,感兴趣的到QCustomplot使用分享(一)能做什么事文章中观看三、源码讲解1、源码结构
  如图所示,是工程的头文件截图,图中的文件数量比较多,但是对外我们使用的可能只是一个ESMPMultiPlot类,这个类中提供了很多接口,足够我们使用,当然了如果有特殊需求的话,我们还可以进行提供定制
  2、头文件
  如下是头文件中的接口,我只是把相关的Public接口列出来了,而这些接口也正好是我们平时使用比较多的接口,看接口名称应该都知道接口库是干什么的,因此这里不再细说voidSetGraphCount(int);voidSetGraphKey(constQVectordouble);doubleGetGraphKey(double);voidSetGraphValue(int,constQString,constQString,constQVectordouble);voidSetGraphScatterStyle(int,int);doubleGetGraphValue(int,bool);获取折线图所在游标出y值参数1:折线下标参数2:左右游标标识doubleGetGraphValue(int,double);获取折线图所在游标出y值参数1:折线下标参数2:xvoidSetGraphColor(int,constQColor);voidSetGraphColor(constQString,constQColor);voidSetGraphUnit(int,constQString);voidSetGraphTitle(int,constQString);voidRefrushGraphID(int,constQString);intGetGraphIndex(constQString)const;voidSetCursorColor(bool,constQColor);voidShowCursor(boolvisibletrue);doubleGetCursorKey(bool);boolCursorVisible();doubleGetCursorValue(bool);voidResizeKeyRange(bool);voidSetKeyRange(double,double);voidSetVauleRange(double,double);voidConfigureGraph();设置std::sharedptrGetAxisCache();3、移动游标
  如下代码所示,是移动游标的核心代码voidESMPPlot::mouseMoveEvent(QMouseEventevent){if(mbDragCursormpDragCursor){doublepixelxeventpos()。x();QCPRangekeyRangeaxisRect()axis(QCPAxis::atBottom)range();doubleminaxisRect()axis(QCPAxis::atBottom)coordToPixel(keyRange。lower);doublemaxaxisRect()axis(QCPAxis::atBottom)coordToPixel(keyRange。upper);doublelcursormmapLeftCursor。begin()。key()point1key();doublelcursorxaxisRect()axis(QCPAxis::atBottom)coordToPixel(lcursor);doublercursormmapRightCursor。begin()。key()point1key();doublercursorxaxisRect()axis(QCPAxis::atBottom)coordToPixel(rcursor);if(minpixelx){pixelxmin;}elseif(maxpixelx){pixelxmax;}if(mbLeftCursor){if(pixelxrcursorx4layer(rcursorLayer)visible()){pixelxrcursorx4;}doublekeyaxisRect()axis(QCPAxis::atBottom)pixelToCoord(pixelx);doublevalue1mpDragCursorpoint1value();doublevalue2mpDragCursorpoint2value();foreach(QCPItemStraightLinelineinmmapLeftCursor。keys()){linepoint1setCoords(key,value1);linepoint2setCoords(key,value2);}mpLeftTextsetText(QString::number(GetGraphKey(pixelx)));mpLeftTextpositionsetPixelPosition(QPoint(pixelx,axisRect()rect()。bottom()25));}else{if(pixelxlcursorx4layer(lcursorLayer)visible()){pixelxlcursorx4;}doublekeyaxisRect()axis(QCPAxis::atBottom)pixelToCoord(pixelx);doublevalue1mpDragCursorpoint1value();doublevalue2mpDragCursorpoint2value();foreach(QCPItemStraightLinelineinmmapRightCursor。keys()){linepoint1setCoords(key,value1);linepoint2setCoords(key,value2);}mpRightTextsetText(QString::number(GetGraphKey(pixelx)));mpRightTextpositionsetPixelPosition(QPoint(pixelx,axisRect()rect()。bottom()25));}eventaccept();replot();emitCursorChanged(mbLeftCursor);return;}super::mouseMoveEvent(event);}
  在ESMPPlot类中,mmapLeftCursor和mmapRightCursor分别是左右游标,为什么这里取了一个map呢?答案是:当时设计的时候是支持多个垂直摆放的游标可以进行游标同步,如果炒股的同学可能就会知道,k线和指标之间可能会有一个数值方便的线,不管在哪个绘图区进行移动,另一个图表里的线也会跟着移动
  不了解这个的同学也不要紧,我们这个控件默认的就是一个表,因此这个map里也就只存了一个指,因此可以不关心这个问题
  在ESMPMultiPlot类中,我们模拟了ESMPPlot的功能,这个时候呢?我们的坐标轴矩形只有一个了,x轴都是一样的,表示时间,对于不同曲线的y轴我们进行了平移,以达到不同的显示位置
  这里边有一个很重的技巧,那就是我们对y轴数据进行了一次单位换算,让他显示的时候可以更好显示在我们制定的区域内,可能像下面这样y1p(y1Yzero1)Ygrid1Xaxis1;核心转换公式,将原始坐标值y1转换为新坐标值y1py1;原始数值Yzero1;零点幅值,决定曲线1零点位置的变量Ygrid1;单格幅值,决定曲线1每个单元格大小的量Xaxis1;显示位置,决定曲线1在画图板中显示位置的变量
  当然了,我们转换后的坐标只是为了显示方便而已,如果我们根据UI获取原始值,我们还需要使用一个逆向公式进行转换回去。4、设置坐标轴矩形个数
  QCP他自己的逻辑是这样的,每一个QCustomPlot类都包括多个坐标轴矩形,而一个坐标轴矩形里又可以包含多个图表,因此我们这个控件是这样的:一个坐标轴矩形多个QCPGraph
  当我们设置的图表数量大于已有图表时,需要使用takeAt接口移除多余的图表;当我们设置的图表数据小于已有图表时,就需要添加新图表对象,添加时机是设置图表数据时
  由于这个函数的代码量比较大,因此这里我删除了一些异常处理代码和设置属性代码
  添加图表数据的流程可能像这面这样首先处理数据异常添加坐标轴根据当前的折线图个数,计算当前折线图的位置和一些转换可能用的系数比率添加图表所有两侧的标题栏名称,如name和unit刷新图表voidESMPMultiPlot::SetGraphCount(intcount){QCPAxisTickerTextleftTicknewQCPAxisTickerText;axisRect()axis(QCPAxis::atLeft)setTicker(QSharedPointerQCPAxisTickerText(leftTick));QCPAxisTickerTextrightTicknewQCPAxisTickerText;axisRect()axis(QCPAxis::atRight)setTicker(QSharedPointerQCPAxisTickerText(rightTick));inttickCountmiCount4;每个折线4个大刻度doubletickDistance(720100)tickCount;QMapdouble,QStringticks;for(inti0;itickCount;i){ticks〔tickDistancei〕;}leftTicksetTicks(ticks);leftTicksetSubTickCount(4);每个大刻度包含4个小刻度doublelabelDistance720miCount;mvecVerticalTick。resize(miCount);mvecNames。resize(miCount);mvecUnits。resize(miCount);doublestep1。0miCount;for(inti0;imvecVerticalTick。size();i){mvecVerticalTick〔i〕labelDistanceilabelDistance2;QCPItemTextnamenewQCPItemText(this);namepositionsetCoords(QPointF(0。01,1(stepistep2)));mvecNames〔mvecVerticalTick。size()i1〕name;QCPItemTextunitnewQCPItemText(this);unitpositionsetCoords(QPointF(0。9,1(stepistep2)));mvecUnits〔mvecVerticalTick。size()i1〕unit;}RefrushItemPosition();mgraphConfigureresize(count);}5、添加图表数据
  毫无疑问,添加图表数据是我们这个控件的非常重要的一个借口
  如下代码所示,看我们是怎么添加数据的首先排除数据异常情况更新图表的各个轴的名称然后给图表添加数据如果图表不存在则添加一个新的设置图表数据设置坐标轴信息设置折线图对应的标题栏名称voidESMPPlot::SetGraphValue(intindex,constQStringxname,constQStringyname,constQVectordoublevalues){if(indexmiCountvalues。size()0){return;}mvecIndex〔index〕xname;mvecUnit〔index〕yname;moldDatas〔index〕values;QListQCPGraphgraphsaxisRect(index)graphs();QCPGraphgraphnullptr;if(graphs。size()0){graphaddGraph(axisRect(index)axis(QCPAxis::atBottom),axisRect(index)axis(QCPAxis::atLeft));graphsetLineStyle(QCPGraph::lsLine);graphsetPen(QColor(255,0,0,200));}else{graphgraphs。at(0);}graphsetData(mvecKeys,values,true);autominiterstd::minelement(values。begin(),values。end());automaxiterstd::maxelement(values。begin(),values。end());doublepadding(maxiterminiter)0。2;axisRect(index)axis(QCPAxis::atLeft)ticker()setTickOrigin(miniterpadding);axisRect(index)axis(QCPAxis::atLeft)ticker()setTickStepStrategy(QCPAxisTicker::tssReadability);axisRect(index)axis(QCPAxis::atLeft)ticker()setTickCount(8);axisRect(index)axis(QCPAxis::atLeft)setRange(miniterpadding,maxiterpadding);axisRect(index)axis(QCPAxis::atRight)ticker()setTickOrigin(miniterpadding);axisRect(index)axis(QCPAxis::atRight)ticker()setTickStepStrategy(QCPAxisTicker::tssReadability);axisRect(index)axis(QCPAxis::atRight)ticker()setTickCount(8);axisRect(index)axis(QCPAxis::atRight)setRange(miniterpadding,maxiterpadding);intleftPaddingQFontMetrics(axisRect(index)axis(QCPAxis::atLeft)labelFont())。width(xname);axisRect(index)axis(QCPAxis::atLeft)setLabel(xname);intrightPaddingQFontMetrics(axisRect(index)axis(QCPAxis::atBottom)labelFont())。width(yname);axisRect(index)axis(QCPAxis::atBottom)setLabel(yname);}6、设置折线图类型
  QCP自带的折线图类型很多,具体我们可以参看QCPScatterStyle::ScatterShape这个枚举类型有多少voidESMPMultiPlot::SetGraphScatterStyle(intindex,intstyle){QListQCPGraphgraphsaxisRect()graphs();if(graphs。size()!0indexgraphs。size()){QCPGraphgraphgraphs。at(0);graphsetScatterStyle(QCPScatterStyle::ScatterShape(style));}}6、其他函数
  还有一些其他的方法,比如保存图表、获取图表坐标、设置图表颜色等这里就不细讲了,文章篇幅所限,不能一一的都贴出来,有需要的伙伴可以联系我,提供功能定制。
  点击领取Qt学习资料视频教程链接
  四、测试方式1、测试工程
  控件我们将的差不多了,这里把测试的代码放出来,大家参考下,首先测试工程截图如下所示,我们的测试代码,大多数都是写在了main函数中。
  2、测试文件
  这里简单说名下,我们的这个文件用途,第一列Time是代表了x轴的时间,而第二列开始的数据都是我们的折线图,一列数据代表一条折线图,并且列的名称就是我们折线图左侧的名称;列名称括号里的单位就是折线图右侧的单位。
  3、测试代码
  限于篇幅,这里我还是把无关的代码删减了很多,需要完整的源码的可以联系我。voidESMultiPlot::LoadData(){ESCsvDBOperatercsvDBOperaternewESCsvDBOperater(nullptr);csvDBOperaterloadCSVFile(qAppapplicationDirPath()temptest31。csv);QStringListnamescsvDBOperatergetCSVNames();autocallback〔this,names〕(constQStringname,constQVectordoubledata){添加图表数据};uiwidgetSetGraphCount(names。size()1);for(inti0;inames。size();i){csvDBOperaterreceiveData(names〔i〕,callback);}doublestartcsvDBOperatergetStartTime();doubleendcsvDBOperatergetEndTime();csvDBOperaterreceiveData(names〔2〕,10。201,10。412,callback);QVectordoubletiemscsvDBOperatergetRangeTimeDatas(10。201,10。412);uiwidgetSetGraphKeyRange(start,end);}
  原作者:朝十晚八orTwowords
  原文转载:https:www。cnblogs。comswarmbeesp10962588。html

鼓掌,本届世界杯中日韩都进淘汰赛了唉,题目权当开个玩笑吧,真的是羡慕嫉妒恨!随着主裁判一声哨响,现场比分定格在了21,韩国队再次战胜葡萄牙队。赛前真的谁能想到会是这样的局面,所有人都觉得韩国队肯定完了,毕竟前两场只CBA全明星单项赛预赛冷门不断,胡明轩晋级三分大赛决赛南都讯记者汪雅云2日晚,CBA全明星活动在浙江诸暨赛区展开了第一日的活动,星锐赛中一年级队以75比73战胜了二年级队,来自北控的廖三宁拿到MVP。同日进行的各单项赛预赛里则冷门不断为什么世界杯每个国家都要带3个门将参赛呢按照国际足联的要求,一支参加世界杯的球队至少要带3名门将,这个不是各队能够自己决定的。2010年的时候朝鲜队试图取个巧,结果反而把自己坑了,约等于平白无故的少了一个板凳位置。(备注冬季搭配最实用的单品黑色羽绒服,不挑年龄肤色,保暖又时髦讲真的朋友们,以我的亲身经历来看,冬天羽绒服真的不能乱选,浪费钱还穿不出想要的效果,纯纯的冤大头。在结合各方面因素后,我发现还是纯黑色羽绒服穿着带劲,冬天穿它出门轻易不影响美观,经大嘴美女舒淇,嘴巴越大越好看,除了她还有谁?娱评大赏从来没有觉得嘴巴大了会好看,自从见了舒淇才明白,原来有一种嘴巴是越大越好看,主要看是长在谁的脸上呢!这样的五官配上这样的大嘴巴才协调,不是吗?因为中国对传统好女人的要求就是赛琳娜身穿迷你裙和棕色大衣亮相纪录片放映会,看起来性感又时髦周三,赛琳娜戈麦斯现身纽约,出席她的纪录片赛琳娜戈麦斯我的思想和我的FYC放映会。这位迪士尼频道30岁的女演员还参加了问答环节,讨论了这部电影,让粉丝们近距离地了解了她6年来的心理秋季防晒正确姿势大家好我是舒舒时间一眨眼过得真快呀转眼间已经秋叶声声了今天依然给大家讲讲秋季的护肤小知识今天的护肤知识是关于秋季防晒哒很多人以为秋季的太阳不像夏天一样那么强,气温也降低了,所以就疏23分5助!山西王牌后卫闪耀全明星入选国家队无大碍洋帅鼓掌点赞在已经结束的全明星星锐赛上,星锐一年级以2分优势惊险战胜二年级的老大哥,廖三宁拿到了职业生涯首座MVP奖杯,虽然15分的表现并不是太过抢眼,但是由于一年级取得了胜利,廖三宁作为全队格力地产再度启动收购珠海免税,将于12月5日起股票停牌文梁宝欣12月2日晚,格力地产(600185。SH)发布公告称,鉴于此前导致公司重大资产重组暂停的事由已消除,公司将继续推动重大资产重组事项并对方案进行相应调整。公司拟以发行股份及丢掉苹果耳机订单损失超20亿!歌尔股份预计今年净利下滑超50视觉中国资料图歌尔股份正式下调了年度业绩预告。12月2日,歌尔股份(002241)公告称,预计2022年净利润17。1亿元21。37亿元,较上年同期下降5060。受境外某大客户暂停世界是平的,发量是厚的2亿小镇青年改变养元青年轻人涌入县城2005年,著名作者托马斯弗里德曼,曾用一种无可质疑的语气宣称,世界是平的。他认为在一个因信息技术而紧密方便的互联世界中,差异逐渐被抹平,一切产品有了被所有人共享的可
哪些游戏的游戏体验堪称满分?游戏的好坏自然是由各种因素来评判的,以下几款游戏它们的评分可是占据排行榜前几名的存在,1首先说说战神四这款单机游戏,这款游戏一经推出就受到了广大玩家的喜爱,就是因为它的剧情完整度非华硕推出RTAX82U20235G增强版路由器,首发价849元IT之家2月28日消息,华硕今日上架了一款RTAX82U魔鬼鱼电竞路由器,采用双频5400M,支持RGB电竞灯效,首发价849元。华硕RTAX82U路由器采用博通三核1。5GHz处OPPO携首款WiFi6路由器FindN2Flip等产品亮相MWC2月27日消息,在巴塞罗拉2023世界移动通信大会上,OPPO携包括OPPOFindN2Flip首款路由器OPPO45W液冷散热器在内的多款产品亮相,并在大会现场宣布海外版OPPO可控核聚变重要突破!星际旅行不再遥远!近期一则关于可控核聚变发电的新闻让许多人激动不已。2022年12月13日美国能源部敲锣打鼓的宣布喜讯,人类在可控核聚变上终于取得正能量了,这个正能量是全世界研究实验室用了大约70年真正演技炸裂的7位演员,小鲜肉们需要学一辈子,别乱炸裂了现在许多粉丝对一些偶像明星的表现,懂不懂就用演技炸裂来形容,殊不知离真正的演技炸裂还有很大的距离。说实话,真正的演技炸裂还得看以前的一些老戏骨,基本上都能做到浑然天成。今天我们就来极氪X来了,完美的皮囊加上有趣的灵魂!他会是你的菜吗?极氪X来了在头条看见彼此作为一个承载了吉利汽车电动化和智能化战略方针的高端品牌,极氪也算是不辱使命,凭借极氪001在新能源B级车市场的强势表现,无论是在口碑还是销量上都取得了双丰收过年串门试试这6条裤子,每一条都显瘦显腿长,够你穿一个冬天马上就要过年了,免不了要有串门聚会这样的场合,在穿衣搭配方面可千万不能掉链子!裤子的穿搭在整体搭配中发挥着至关重要的作用,选对适合自己的裤款修身显瘦又高级,轻松提升搭配质感。这篇文茅台原董事长多次嫖娼受贿数额巨大,沉溺赌博,细节触目惊心高卫东曾经为茅台董事长,位高权重,能做到这个位置已经非常不容易了,也是很高的官职,合法收入应该也不低,但是,高卫东却不收心,错就错在了一个字贪,他总是和熟人比吃穿,虚幻的物质生活使石林风景区早春二月,海棠花开又是一年春来到,海棠花开枝头俏。早春二月,云南石林风景区的海棠花开了,满树粉色的花儿开得熙熙攘攘。当地人穿着民族装在花下谈笑风生。海棠花花姿潇洒,花开似锦,所以自古以来是雅俗共赏的每周发班!去黄梅东山问梅村景区游玩,有直通车了长江日报大武汉客户端2月28日讯(记者黄丽娟通讯员熊丹娜)欢迎武汉游客到问梅村赏黄梅戏观岳家拳品黄梅文化。2月27日下午,由黄梅县文化和旅游局主办的东山问梅村渠道产品武汉推介会在洪海博早读丨我省这个著名景区将继续免票我省进口通关效率位列前七大外贸省市第3位。今年我省将重点支持福州厦门泉州推进高水平引才聚才平台建设。全省工会今年计划组织不少于300场次就业培训,惠及5万人次以上。今天开始,我省原
友情链接:快好找快生活快百科快传网中准网文好找聚热点快软网