QT实现天气预报

1. MainWindow类设计的成员变量和方法

public:
    MainWindow(QWidget* parent = nullptr);
    ~MainWindow();

   protected:

形成文本菜单来用来右键关闭窗口
    void contextMenuEvent(QContextMenuEvent* event);

鼠标被点击之后此事件被调用

    void mousePressEvent(QMouseEvent *ev);

移动窗口
    void mouseMoveEvent(QMouseEvent* ev);
    //重写过滤器方法
    bool eventFilter(QObject *watched, QEvent *event);

private slots://信号槽函数
    void onReplied(QNetworkReply* reply);

    void on_btnSearch_clicked();

public:

    void getWeaterInfo(QString cityCode);
    //解析json数据
    void parseJson(QByteArray& byteArray);
    int parseString(QString str);
    //更新UI
    void UpdateUi();
    //检测空气质量
   int checkedQulity(int api);
    //绘制高低温曲线
    void paintHighCurve();
   void paintLowCurve();


   private:
    Ui::MainWindow* ui;

    QMenu* mExitMenu;   // 右键退出的菜单
    QAction* mExitAct;  // 退出的行为
    QPoint m_offset;//鼠标和左上角坐标的偏移距离
    QNetworkAccessManager *m_netAccessManager;
    Today m_today;
    Day m_day[7];

    //星期和日期
    QList<QLabel*> m_weekList;
    QList<QLabel*> m_dateList;

    //天气和天气图标
    QList<QLabel*> m_typeList;
    QList<QLabel*> m_typeIconList;

    //天气污染指数
    QList<QLabel*> m_ApiList;

    //风力和风向
    QList<QLabel*> m_FLList;
    QList<QLabel*> m_FXList;

把UI控件都放到QList集合里面

//将UI控件放到数组里边,方便使用循环进行处理m_dateList << ui->lblDate0 << ui->lblDate1 << ui->lblDate2 << ui->lblDate3 << ui->lblDate4 << ui->lblDate5;m_weekList << ui->lblWeek0 << ui->lblWeek1 << ui->lblWeek2 << ui->lblWeek3 << ui->lblWeek4 << ui->lblWeek5;m_ApiList << ui->lblQuality0 << ui->lblQuality1 << ui->lblQuality2 << ui->lblQuality3 << ui->lblQuality4 << ui->lblQuality5;m_typeList << ui->lblType0 << ui->lblType1 << ui->lblType2 << ui->lblType3 << ui->lblType4 << ui->lblType5;m_typeIconList << ui->lblTypeIcon0 << ui->lblTypeIcon1 << ui->lblTypeIcon2 << ui->lblTypeIcon3 << ui->lblTypeIcon4 << ui->lblTypeIcon5;m_FXList << ui->lblFx0 << ui->lblFx1 << ui->lblFx2 << ui->lblFx3 << ui->lblFx4 << ui->lblFx5;m_FLList << ui->lblFl0 << ui->lblFl1 << ui->lblFl2 << ui->lblFl3 << ui->lblFl4 << ui->lblFl5;

    QMap <QString,QString> m_typeMap;

2. 右键退出

void contextMenuEvent(QContextMenuEvent* event);

  mExitMenu->exec(QCursor::pos());event->accept();

QMenu* mExitMenu;   右键退出的菜单

 // 右键菜单:退出程序
    mExitMenu = new QMenu(this);//添加一个菜单框
    mExitAct = new QAction();
    mExitAct->setText(tr("退出"));
    mExitAct->setIcon(QIcon(":/res/close.png"));
    mExitMenu->addAction(mExitAct);//把这个行为设置到菜单里面去

    connect(mExitAct, &QAction::triggered, this, [=]() {
        qApp->exit(0); 
    });

3.存放天气信息的类


class Today
{
public:Today(){date = "2022-10-20";city = "广州";ganmao = "感冒指数";wendu = 0;shidu = "0%";pm25 = 0;quality = "无数据";type = "多云";fl = "2级";fx = "南风";high = 30;low = 18;}QString date;QString city;QString ganmao;int wendu;QString shidu;int pm25;QString quality;QString type;QString fx;QString fl;int high;int low;
};class Day
{
public:Day(){date = "2022-10-20";week = "周五";type = "多云";high = 0;low = 0;fx = "南风";fl = "2级";aqi = 0;}QString date;QString week;QString type;QString ymd;int high;int low;QString fx;QString fl;int aqi;
};

4.get请求数据

void MainWindow::getWeaterInfo(QString cityName)
{QString cityCode=weatherTool::getCityCode(cityName);qDebug()<<"cityCode"<<cityCode;QUrl url("http://t.weather.itboy.net/api/weather/city/"+cityCode);m_netAccessManager->get(QNetworkRequest(url));//发送get请求
}

5.重写void onReplied函数

void onReplied(QNetworkReply* reply);在get请求成功之后此函数被调用

void MainWindow::onReplied(QNetworkReply *reply)
{qDebug()<<"onReplied success";int status_code=reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();qDebug()<<"参数: "<<reply->operation();qDebug()<<"状态码: "<<status_code;qDebug()<<"url: "<<reply->url();qDebug()<<"请求头: "<<reply->rawHeaderList();if(reply->error()!=QNetworkReply::NoError||status_code!=200){qDebug()<<reply->errorString().toLatin1().data();QMessageBox::warning(this,"天气预报","数据请求失败",QMessageBox::Ok);}else{QByteArray all=reply->readAll();qDebug()<<"read all : "<<all.data() ;parseJson(all);}reply->deleteLater();}

在此函数里把请求到的数据全部读出来

6.读出类之后在调用parseJson(all);来进行解析JSON数据

void MainWindow::parseJson(QByteArray &byteArray)
{QJsonParseError err;QJsonDocument doc=QJsonDocument::fromJson(byteArray,&err);if(err.error!=QJsonParseError::NoError){qDebug()<<"解析Json出错了";return ;}//Json对象,包括了请求的全部数据QJsonObject rootObj = doc.object();//qDebug() << rootObj;QJsonObject dataObj = rootObj.value("data").toObject();QJsonObject yesterdayObj = dataObj.value("yesterday").toObject();//Json数组,未来几天的数据QJsonArray forecastArray = dataObj.value("forecast").toArray();//1.解析日期和城市QString str = rootObj.value("time").toString();m_today.date= str.split(" ").at(0);qDebug()<<"今天:"<<m_today.date;m_today.city = rootObj.value("cityInfo").toObject().value("city").toString();//2.解析yesterdaym_day[0].date = yesterdayObj.value("date").toString();//日期m_day[0].aqi = yesterdayObj.value("aqi").toDouble();//空气质量,toDouble包括整数型m_day[0].type = yesterdayObj.value("type").toString();//天气类型m_day[0].week = yesterdayObj.value("week").toString();//星期m_day[0].ymd = yesterdayObj.value("ymd").toString();//完整年月日//解析最低温度和最高温度/*QString str = yesterdayObj.value("low").toString();//最低温度str = str.split(" ").at(1);//以空格为分割符,去除前面的文字,取出第二个X℃,str = str.left(str.length() - 1);//去除后面的摄氏度符号mDay[0].low = str.toInt();//传入最低温度str.clear();//清除内容str = yesterdayObj.value("high").toString();//最高温度str = str.split(" ").at(1);//以空格为分割符,去除前面的文字,取出第二个X℃,str = str.left(str.length() - 1);//去除后面的摄氏度符号mDay[0].high = str.toInt();//传入最高温度*///简化版m_day[0].low = parseString(yesterdayObj.value("low").toString());//最低温度m_day[0].high = parseString(yesterdayObj.value("high").toString());//最高温度m_day[0].fx = yesterdayObj.value("fx").toString();//风向m_day[0].fl = yesterdayObj.value("fl").toString();//风力//3.解析forecast中的6天数据for (int i = 0; i < 5; ++i) {QJsonObject forecastData = forecastArray[i].toObject();//跳过0,因为0已经赋值给昨天的数据了m_day[i + 1].date = forecastData.value("date").toString();//日期//qDebug()<<"time: "<<m_day[i + 1].date;m_day[i + 1].aqi = forecastData.value("aqi").toDouble();//空气质量,toDouble包括整数型m_day[i + 1].type = forecastData.value("type").toString();//天气类型m_day[i + 1].week = forecastData.value("week").toString();//星期m_day[i + 1].low = parseString(forecastData.value("low").toString());//最低温度m_day[i + 1].high = parseString(forecastData.value("high").toString());//最高温度m_day[i + 1].fx = forecastData.value("fx").toString();//风向m_day[i + 1].fl = forecastData.value("fl").toString();//风力m_day[i + 1].ymd = forecastData.value("ymd").toString();//完整年月日//ymd分割成MM/dd//QDateTime::fromString(forecastData.value("ymd").toString(), "yyyy-MM-dd").toString("MM/dd");//日期//qDebug() << i << mDay[i + 1].date << mDay[i + 1].aqi << mDay[i + 1].type << mDay[i + 1].week << mDay[i + 1].low << mDay[i + 1].high << mDay[i + 1].fx << mDay[i + 1].fl;//qDebug() << i << mDay[i].date;//qDebug() << mDay[i + 1].ymd;}//4.解析今天的数据m_today.shidu = dataObj.value("shidu").toString();m_today.quality = dataObj.value("quality").toString();m_today.ganmao = dataObj.value("ganmao").toString();m_today.pm25 = dataObj.value("pm25").toDouble();m_today.wendu = dataObj.value("wendu").toString().toInt();//qDebug() << dataObj;//qDebug() << mToday.shidu << mToday.quality << mToday.ganmao << mToday.pm25 << mToday.wendu;//注意:forecast中的第2个元素也是今天的数据m_today.type = m_day[1].type;qDebug()<<"今天的天气是: "<<m_today.type;m_today.low = m_day[1].low;m_today.high = m_day[1].high;m_today.fx = m_day[1].fx;m_today.fl = m_day[1].fl;//qDebug() << mToday.type << mToday.low << mToday.high << mToday.fx << mToday.fl;UpdateUi();//更新曲线在搜索城市完成后手动更新曲线ui->lblHighCurve->update();ui->lblLowCurve->update();
}

解析完JSON数据之后在更新ui(UpdateUi();)

7.更新UI

void MainWindow::UpdateUi()
{ui->lblDate->setText(QDateTime::fromString(m_today.date,"yyyyMMdd").toString("yyyy/MM/dd")+" "+m_day[1].week);ui->lblCity->setText(m_today.city);//更新今天数据ui->lblTypeIcon->setPixmap(m_typeMap[m_today.type]);//更新图片ui->lblTemp->setText(QString ::number(m_today.wendu));ui->lblType->setText(m_today.type);ui->lblLowHigh->setText(QString ::number(m_today.low)+"℃ ~ "+QString::number(m_today.high)+"℃");ui->lblGanMao->setText("感冒指数: "+m_today.ganmao);ui->lblWindFl->setText(m_today.fl);ui->lblWindFx->setText(m_today.fx);ui->lblPM25->setText(QString::number(m_today.pm25));ui->lblShiDu->setText(m_today.shidu);ui->lblQuality->setText(m_today.quality);//更新六天for(int i=0;i<6;i++){//更新日期和时间m_weekList[i]->setText("周"+m_day[i].week.right(1));//从右边取第一个ui->lblWeek0->setText("昨天");ui->lblWeek1->setText("今天");ui->lblWeek2->setText("明天");//         //把日期提取出来变成 10/10格式QStringList slist=m_day[i].ymd.split("-");qDebug()<<slist;m_dateList[i]->setText(slist.at(1)+"/"+slist.at(2));//更新天气类型m_typeList[i]->setText(m_day[i].type);m_typeIconList[i]->setPixmap(m_typeMap[m_day[i].type]);//更新图片
//空气质量switch (checkedQulity(m_day[i].aqi)) {case 1:m_ApiList[i]->setText("优");m_ApiList[i]->setStyleSheet("background-color: rgb(121,184,0);");break;case 2:m_ApiList[i]->setText("良");m_ApiList[i]->setStyleSheet("background-color: rgb(255,187,23);");break;case 3:m_ApiList[i]->setText("轻度");m_ApiList[i]->setStyleSheet("background-color: rgb(255,87,97);");break;case 4:m_ApiList[i]->setText("中度");m_ApiList[i]->setStyleSheet("background-color: rgb(235,17,27);");break;case 5:m_ApiList[i]->setText("重度");m_ApiList[i]->setStyleSheet("background-color: rgb(170,0,0);");break;case 6:m_ApiList[i]->setText("严重");m_ApiList[i]->setStyleSheet("background-color: rgb(110,0,0);");break;}//风力风向m_FXList[i]->setText(m_day[i].fx);m_FLList[i]->setText(m_day[i].fl);}}//检查空气质量
int MainWindow::checkedQulity(int aqi)
{if (aqi >= 0 && aqi <= 50) {return 1;}else if(aqi > 50 && aqi <= 100){return 2;}else if(aqi > 100 && aqi <= 150){return 3;}else if(aqi > 150 && aqi <= 200){return 4;}else if(aqi > 200 && aqi <= 250){return 5;}else{return 6;}return -1;
}

8.实现搜索框搜索城市

获取LeCity搜索框输入的数据

void MainWindow::on_btnSearch_clicked()
{QString cityName=ui->leCity->text();getWeaterInfo(cityName);
}

天气工具类

//天气工具类
class weatherTool{
private://内联静态变量 在占用调用处的内存inline static QMap<QString,QString> m_cityMap={};static void initCityMap(){QFile fd("E:\\code\\citycode.json");fd.open(QIODevice::ReadOnly|QIODevice::Text);QByteArray allJson=fd.readAll();fd.close();QJsonParseError err;//读出的数据转换为json的QJsonDocument doc=  QJsonDocument::fromJson(allJson,&err);if(err.error!=QJsonParseError::NoError||!doc.isArray()){//不是数组就错误qDebug()<<"json数据转换失败...";return;}  //转换成json数组QJsonArray rootArray=doc.array();for(int i=0;i<rootArray.size();i++){QString cityName = rootArray[i].toObject().value("city_name").toString();QString cityCode = rootArray[i].toObject().value("city_code").toString();//判断编号是否存在if(cityCode.size()>0){m_cityMap.insert(cityName,cityCode);}}}public :QString static getCityCode(QString cityName){qDebug()<<"cityName"<<cityName;if(m_cityMap.isEmpty()){qDebug()<< m_cityMap.isEmpty();initCityMap();}if(cityName==""){QMessageBox::warning(nullptr,"错误(null)","输入的城市不能为空...",QMessageBox::Ok);return "";}//在map寻找传入的城市名称,返回值可以遍历//返回的是map中的多个元素QMap<QString, QString>::iterator it = m_cityMap.find(cityName);//到结尾了,未找到,加上市,再搜一遍if (it == m_cityMap.end()) {it = m_cityMap.find(cityName + "市");}//找到了if (it != m_cityMap.end()) {qDebug()<<" if (it != m_cityMap.end())";return it.value();}QMessageBox::warning(nullptr,"城市名字错误","输入的城市名字错误...",QMessageBox::Ok);//否则return "";}};

读取这个文件里面的数据QFile fd("E:\\code\\citycode.json");把city_name和city_code取出来放到QMap集合里面去,之后在和输入的CityName进行查找获得city_code.

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/42914.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Leetcode每日一题:1444. 切披萨的方案数(2023.8.17 C++)

目录 1444. 切披萨的方案数 题目描述&#xff1a; 实现代码与解析&#xff1a; 二维后缀和 动态规划 原理思路&#xff1a; 1444. 切披萨的方案数 题目描述&#xff1a; 给你一个 rows x cols 大小的矩形披萨和一个整数 k &#xff0c;矩形包含两种字符&#xff1a; A …

Spring(三):Spring中Bean的生命周期和作用域

前言 在 Spring 中&#xff0c;那些组成应用程序的主体及由 Spring IOC 容器所管理的对象&#xff0c;被称之为 bean。简单地讲&#xff0c;bean 就是由 IOC 容器初始化、装配及管理的对象&#xff0c;除此之外&#xff0c;bean 就与应用程序中的其他对象没有什么区别了。而 b…

LeetCode 热题 100(四):48. 旋转图像、240. 搜索二维矩阵 II、234. 回文链表

一.48. 旋转图像 题目要求&#xff1a;就是一个顺时针的旋转过程。 思路&#xff1a;观察矩阵&#xff0c;得出翻转前第i行的第J个元素 等于 翻转后倒数第i列的第J个元素&#xff0c;举例说明&#xff0c;第1行第2个元素为“2”&#xff0c;翻转后到了 倒数第1列的第2个元素…

MAC环境,在IDEA执行报错java: -source 1.5 中不支持 diamond 运算符

Error:(41, 51) java: -source 1.5 中不支持 diamond 运算符 (请使用 -source 7 或更高版本以启用 diamond 运算符) 进入设置 修改java版本 pom文件中加入 <plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin&l…

vue项目预览pdf功能(解决动态文字无法显示的问题)

最近&#xff0c;因为公司项目需要预览pdf的功能&#xff0c;开始的时候找了市面上的一些pdf插件&#xff0c;都能用&#xff0c;但是&#xff0c;后面因为pdf变成了需要根据内容进行变化的&#xff0c;然后&#xff0c;就出现了需要动态生成的文字不显示了。换了好多好多的插件…

Flink安装与使用

1.安装准备工作 下载flink Apache Flink: 下载 解压 [dodahost166 bigdata]$ tar -zxvf flink-1.12.0-bin-scala_2.11.tgz 2.Flinnk的standalone模式安装 2.1修改配置文件并启动 修改&#xff0c;好像使用默认的就可以了 [dodahost166 conf]$ more flink-conf.yaml 启动 …

【办公自动化】使用Python批量生成PPT版荣誉证书

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…

RocketMQ消费者可以手动消费但无法主动消费问题,或生成者发送超时

1.大多数是配置问题 修改rocketmq文件夹broker.conf 2.配置与集群IP或本地IPV4一样 重启 在RocketMQ独享实例中支持IPv4和IPv6双栈&#xff0c;主要是通过在网络层面上同时支持IPv4和IPv6协议栈来实现的。RocketMQ的Broker端、Namesrv端和客户端都需要支持IPv4和IPv6协议&…

Python土力学与基础工程计算.PDF-螺旋板载荷试验

python 求解代码如下&#xff1a; 1. import numpy as np 2. 3. # 已知参数 4. p_a 100 # 标准压力&#xff0c; kPa 5. p np.array([25, 50, 100, 200) # 荷载&#xff0c; kPa 6. s np.array([2.88, 5.28, 9.50, 15.00) / 10 # 沉降量&#xff0c; cm 7. D 10 # 螺旋板直…

C语言:选择+编程(每日一练)

目录 选择题&#xff1a; 题一&#xff1a; 题二&#xff1a; 题三&#xff1a; 题四&#xff1a; 题五&#xff1a; 编程题&#xff1a; 题一&#xff1a;尼科彻斯定理 示例1 题二&#xff1a;等差数列 示例2 本人实力有限可能对一些地方解释和理解的不够清晰&…

windows下redis服务启动及.bat文件中中redis服务的启动

windows windows下redis服务的启动 1、不配置环境变量 找到redis服务的安装目录进入命令行窗口并输入命令redis-server.exe redis.windows.conf2、配置环境变量 将redis安装目录配置在path环境变量中之后就可以在cmd窗口的任意位置输入redis-server命令就可以启动redis服务…

材料行业可以转IC设计后端吗?

近来有许多材料行业的小伙伴通过后台来问我对于职业规划的看法&#xff0c;甚至有些小伙伴直接点明了某个行业适不适合自己&#xff0c;那么我这边仅以近年来比较热门的数字芯片设计来展开讲讲&#xff0c;材料适不适合转行做IC呢。 对于理工科的同学而言&#xff0c;选择哪个…

公告:微信小程序备案期限官方要求

备案期限要求 1、若微信小程序未上架&#xff0c;自2023年9月1日起&#xff0c;微信小程序须完成备案后才可上架&#xff0c;备案时间1-20日不等&#xff1b; 2、若微信小程序已上架&#xff0c;请于2024年3月31日前完成备案&#xff0c;逾期未完成备案&#xff0c;平台将按照…

Android Studio实现列表展示图片

效果&#xff1a; MainActivity 类 package com.example.tabulation;import android.content.Intent; import android.os.Bundle; import android.view.View;import androidx.appcompat.app.AppCompatActivity; import androidx.recyclerview.widget.LinearLayoutManager; im…

JVM——引言+JVM内存结构

引言 什么是JVM 定义: Java VirtualMachine -java 程序的运行环境 (ava 二进制字节码的运行环境) 好处: 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收功能数组下标越界检查&#xff0c;多态 比较: jvm jre jdk 学习jvm的作用 面试理解底层实现原理中…

SpringBoot-lombok

为什么要使用lombok? Lombok是一个通过注解以达到减少代码的Java库,如通过注解的方式减少getter,setter方法,构造方法等。通过注解的形式自动生成构造器、getter/setter、equals、hashcode、toString等方法&#xff0c;并可以自动化生成日志变量&#xff0c;简化java开发、提高…

NLP文本匹配任务Text Matching [有监督训练]:PointWise(单塔)、DSSM(双塔)、Sentence BERT(双塔)项目实践

NLP文本匹配任务Text Matching [有监督训练]&#xff1a;PointWise&#xff08;单塔&#xff09;、DSSM&#xff08;双塔&#xff09;、Sentence BERT&#xff08;双塔&#xff09;项目实践 0 背景介绍以及相关概念 本项目对3种常用的文本匹配的方法进行实现&#xff1a;Poin…

模型预测笔记(一):数据清洗分析及可视化、模型搭建、模型训练和预测代码一体化和对应结果展示(可作为baseline)

模型预测 一、导入关键包二、如何载入、分析和保存文件三、修改缺失值3.1 众数3.2 平均值3.3 中位数3.4 0填充 四、修改异常值4.1 删除4.2 替换 五、数据绘图分析5.1 饼状图5.1.1 绘制某一特征的数值情况&#xff08;二分类&#xff09; 5.2 柱状图5.2.1 单特征与目标特征之间的…

Mysql性能优化:什么是索引下推?

导读 索引下推&#xff08;index condition pushdown &#xff09;简称ICP&#xff0c;在Mysql5.6的版本上推出&#xff0c;用于优化查询。 在不使用ICP的情况下&#xff0c;在使用非主键索引&#xff08;又叫普通索引或者二级索引&#xff09;进行查询时&#xff0c;存储引擎…

容斥原理 博弈论(多种Nim游戏解法)

目录 容斥原理容斥原理的简介能被整除的数&#xff08;典型例题&#xff09;实现思路代码实现扩展&#xff1a;用DPS实现 博弈论博弈论中的相关性质博弈论的相关结论先手必败必胜的证明Nim游戏&#xff08;典型例题&#xff09;代码实现 台阶-Nim游戏&#xff08;典型例题&…