【中工开发者】——气候查询,鸿蒙OS

嚴華  论坛元老 | 2025-3-15 06:20:14 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 1010|帖子 1010|积分 3030

引言

伴随智能手机的大规模普及,气候查询已然成为人们一样平常生存里不可或缺的关键构成部分。而鸿蒙操作体系(HarmonyOS)作为华为精心打造并推出的新一代智能终端操作体系,凭借其独特的技术架构与创新特性,为广大开发者们开启了一扇全新的开发大门,带来前所未有的开发体验。本文将会全方位、细致入微地先容基于鸿蒙 OS 开发一款气候查询应用的详细步骤与方法,其内容将完整覆盖从与数据接口进行对接获取景象信息,到实现都会信息管理中都会删除功能的一整套流畅流程,旨在为开发者们提供全面且极具价值的开发指引与参考范例
1. 数据接口获取

在开发气候查询应用之前,我们需要一个可靠的气候数据接口。这里我们可以选择公开的气候API,如高德地图、和风气候等。以高德地图为例,我们需要注册账号并获取一个API密钥(API Key),用于后续的数据请求。
1.1 获取API Key
访问OpenWeatherMap官网气候查询-基础 API 文档-开发指南-Web服务 API | 高德地图API气候查询-基础 API 文档-开发指南-Web服务 API | 高德地图API气候查询-基础 API 文档-开发指南-Web服务 API | 高德地图APIOpenWeatherMap官网,注册账号并订阅相应的服务,获取API Key。


2. 对象创建

在鸿蒙OS中,我们需要创建几个关键的对象来管理气候数据和用户界面。
2.1 气候数据模型

创建一个Casts类来存储气候数据,包罗白天 温度、夜晚温度、湿度、风向、等数据信息。
  1. export class casts{
  2.   date:string=''
  3.   dayweather:string=''
  4.   nightweather:string=''
  5.   daytemp:number=0
  6.   nighttemp:number=0
  7.   daywind:string=''
  8.   daypower:string=''
  9.   daytemp_float:number=0
  10.   nighttemp_float:number=0
  11. }
复制代码
2.2 都会管理类

创建一个forecasts类来存储都会名称,代码,气候信息:
  1. export class forecasts{
  2.   city:string=''
  3.   adcode:number=0
  4.   casts:Array<casts>=[]
  5. }
  6. 2.3 城市管理类
复制代码
2.3 都会管理类

创建一个WeatherModel类来存储访问状态信息,都会信息等
  1. export class WeatherModel{
  2.   status:number=0
  3.   count:number=0
  4.   infocode:number=0
  5.   forecasts:Array<forecasts> = []
  6. }
复制代码
3. 获取数据

通过高德地图API获取指定都会的气候数据,并提供了并发请求多个都会气候的功能。代码中界说了一个getWeatherUtil类,其中包罗两个主要方法:
getWeather(cityCode):这个方法继承一个都会代码作为参数,使用高德地图API的URL模板构建请求URL,并发送HTTP GET请求。假如请求乐成(响应码200),它将解析响应结果并使用Promise的resolve函数返回解析后的气候数据对象WeatherModel。
getWeathers(cityCodes):这是一个异步方法,继承一个都会代码数组,使用getWeather方法为每个都会代码创建一个Promise,并使用Promise.all来并发实行这些Promise。当全部请求都完成后,它会遍历结果,打印每个都会的名字,并返回包罗全部气候数据的WeatherModel数组。
  1. import {casts} from "../viewmodel/casts"
  2. @Component
  3. export struct cityView{
  4.   @Builder weatherImage(dayweather: string) {
  5.     if (dayweather === "晴") {
  6.       Image($r('app.media.sun')).width(30)
  7.     }
  8.     if (dayweather === "多云") {
  9.       Image($r("app.media.cloud")).width(30)
  10.     }
  11.     if (dayweather === "阴") {
  12.       Image($r("app.media.cloud")).width(30)
  13.     }
  14.     if (dayweather.includes("雨")) {
  15.       Image($r("app.media.rain")).width(30)
  16.     }
  17.   }
  18.   //天气数据
  19.   casts:Array<casts>=[]
  20.   build() {
  21.     Column(){
  22.       ForEach(this.casts,(cast:casts)=>{
  23.         if (this.casts[0]===cast){
  24.           //上半部分 图片+当天天气
  25.           //图片
  26.           Row() {
  27.             if (cast.dayweather === "晴") {
  28.               Image($r('app.media.sun')).width(260)
  29.             }
  30.             if (cast.dayweather === "多云") {
  31.               Image($r("app.media.cloud")).width(260)
  32.             }
  33.             if (cast.dayweather === "阴") {
  34.               Image($r("app.media.cloud")).width(260)
  35.             }
  36.             if (cast.dayweather.includes("雨")) {
  37.               Image($r("app.media.rain")).width(260)
  38.             }
  39.           }.height("30%")
  40.           Column(){
  41.             //温度天气
  42.             Row() {
  43.               Text(cast.dayweather).fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  44.               Text("  " + cast.daytemp + "°~" + cast.nighttemp + "°")
  45.                 .fontSize(30)
  46.                 .fontColor(Color.White)
  47.                 .fontWeight(FontWeight.Bold)
  48.             }
  49.             //风力
  50.             Row() {
  51.               Text(cast.daywind + "风  ").fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  52.               Text(cast.daypower + "级").fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  53.             }
  54.           }.margin({ top: 10 })
  55.         }
  56.       })
  57.       //近期天气列表
  58.       Column() {
  59.         Text("查看近期天气").fontSize(26).margin({ top: 30 })
  60.         //天气列表
  61.         Row() {
  62.           ForEach(this.casts, (cast: casts) => {
  63.             Column() {
  64.               Text(cast.date.substring(5))
  65.               this.weatherImage(cast.dayweather)
  66.               Blank()
  67.               Text(cast.daytemp.toString())
  68.               Line()
  69.                 .width(20).height(80).startPoint([10, 0])
  70.                 .endPoint([10, 70]).stroke(Color.Black)
  71.                 .strokeWidth(3).strokeDashArray([10, 3])
  72.               Text(cast.nighttemp.toString())
  73.               Blank()
  74.               this.weatherImage(cast.dayweather)
  75.             }.height("90%").width("20%")
  76.           })
  77.         }
  78.         .width("80%")
  79.         .height("60%")
  80.         .backgroundColor("#ffbab8b8")
  81.         .opacity(0.5)
  82.         .justifyContent(FlexAlign.SpaceAround)
  83.       }.height("45%").width("100%")
  84.     }.width("100%").height("100%")
  85.   }
  86. }
复制代码
4. 主页气候数据展示 

4.1 主页结构

该项目主页用于展示一个都会气候信息的界面。界面答应用户在差别都会的气候信息之间切换。组件在即将表现时会异步获取气候数据,并更新界面上表现的都会列表和气候信息。用户可以通过点击添加按钮跳转到添加都会的页面,并通过Tab组件检察各个都会的气候详情。以下是主页的结构代码:
  1. import { cityView } from '../view/cityView'
  2. import getWeatherUtil from "../viewmodel/getWeatherUtil"
  3. import {WeatherModel} from "../viewmodel/WeatherModel"
  4. import { router } from '@kit.ArkUI'
  5. @Entry
  6. @Component
  7. struct Index {
  8.   //城市代码集合
  9.   @State cityCodeList :number[]=[110000,120000,130000]
  10.   //城市名字集合
  11.   @State cityNameList :string[]=[]
  12.   //城市信息集合
  13.   @State cityWeatherList :Array<WeatherModel>=[]
  14.   //当前tab组件索引
  15.   @State cityIndex: number = 0
  16.   //tab控制器
  17.   tabcontroller: TabsController = new TabsController()
  18.   //tarBar 自定义函数
  19.   @Builder tabBuild(index: number) {
  20.     Circle({ width: 10, height: 10 }).fill(this.cityIndex === index ? Color.White : Color.Gray).opacity(0.2)
  21.   }
  22.   aboutToAppear(){
  23.     this.initDate()
  24.   }
  25. async  initDate(){
  26.     // let result :Array<WeatherModel> = getWeatherUtil.getWeathers(this.cityCodeList)
  27.   let result :Array<WeatherModel>=await getWeatherUtil.getWeathers(this.cityCodeList)
  28.    for (let i = 0; i < result.length; i++) {
  29.      let ACityWeather = new WeatherModel()
  30.      ACityWeather = result[i]
  31.      this.cityWeatherList.push(ACityWeather)
  32.      let cityName = result[i].forecasts[0].city
  33.      this.cityNameList.push(cityName)
  34.    }
  35.   }
  36.   build() {
  37.     Column(){
  38.       Row(){
  39.       Button("添加")
  40.         .fontSize(25)
  41.         .fontColor(Color.Gray)
  42.         .opacity(0.7)
  43.         .backgroundColor("#87CEEB")
  44.         .margin({ bottom: 15 })
  45.         .onClick(() => {
  46.           router.pushUrl({
  47.             url: "pages/AddCity",
  48.             params: {
  49.               Codes : this.cityCodeList,
  50.               Names : this.cityNameList
  51.             }
  52.           })
  53.         })
  54.       Text(this.cityNameList[this.cityIndex]).fontSize(40).fontColor(Color.Orange)
  55.       Button("删除")
  56.         .fontSize(25)
  57.         .fontColor(Color.Gray)
  58.         .opacity(0.7)
  59.         .backgroundColor("#87CEEB")
  60.         .margin({ bottom: 15 })
  61.       }.height("10%")
  62.       .width("100%")
  63.       .justifyContent(FlexAlign.SpaceBetween)
  64.       //信息布局
  65.       Tabs({ barPosition: BarPosition.Start, controller: this.tabcontroller }) {
  66.         ForEach(this.cityWeatherList,(cityWeather:WeatherModel)=>{
  67.           TabContent(){
  68.             cityView({casts:cityWeather.forecasts[0].casts})
  69.           }.tabBar(this.tabBuild(this.cityWeatherList.findIndex(obj=>obj===cityWeather)))
  70.         })
  71.       }.barWidth(20)
  72.        .barHeight(40)
  73.       .onChange((index:number)=>{
  74.         this.cityIndex = index
  75.       })
  76.     }.backgroundColor("#87CEEB")
  77.     .width("100%").height("100%")
  78.   }
  79. }
复制代码

  • 4.2 气候信息展示
在差别的气候信息的展示中用ForEach循环来遍历气候数据数组casts,并为每条气候数据生成对应的视图。界面中包罗了气候图标、温度表现、风力描述以及日期标记,使用差别的结构和样式来优化信息的展示。
  1. import {casts} from "../viewmodel/casts"
  2. @Component
  3. export struct cityView{
  4.   @Builder weatherImage(dayweather: string) {
  5.     if (dayweather === "晴") {
  6.       Image($r('app.media.sun')).width(30)
  7.     }
  8.     if (dayweather === "多云") {
  9.       Image($r("app.media.cloud")).width(30)
  10.     }
  11.     if (dayweather === "阴") {
  12.       Image($r("app.media.cloud")).width(30)
  13.     }
  14.     if (dayweather.includes("雨")) {
  15.       Image($r("app.media.rain")).width(30)
  16.     }
  17.   }
  18.   //天气数据
  19.   casts:Array<casts>=[]
  20.   build() {
  21.     Column(){
  22.       ForEach(this.casts,(cast:casts)=>{
  23.         if (this.casts[0]===cast){
  24.           //上半部分 图片+当天天气
  25.           //图片
  26.           Row() {
  27.             if (cast.dayweather === "晴") {
  28.               Image($r('app.media.sun')).width(260)
  29.             }
  30.             if (cast.dayweather === "多云") {
  31.               Image($r("app.media.cloud")).width(260)
  32.             }
  33.             if (cast.dayweather === "阴") {
  34.               Image($r("app.media.cloud")).width(260)
  35.             }
  36.             if (cast.dayweather.includes("雨")) {
  37.               Image($r("app.media.rain")).width(260)
  38.             }
  39.           }.height("30%")
  40.           Column(){
  41.             //温度天气
  42.             Row() {
  43.               Text(cast.dayweather).fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  44.               Text("  " + cast.daytemp + "°~" + cast.nighttemp + "°")
  45.                 .fontSize(30)
  46.                 .fontColor(Color.White)
  47.                 .fontWeight(FontWeight.Bold)
  48.             }
  49.             //风力
  50.             Row() {
  51.               Text(cast.daywind + "风  ").fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  52.               Text(cast.daypower + "级").fontSize(30).fontColor(Color.White).fontWeight(FontWeight.Bold)
  53.             }
  54.           }.margin({ top: 10 })
  55.         }
  56.       })
  57.       //近期天气列表
  58.       Column() {
  59.         Text("查看近期天气").fontSize(26).margin({ top: 30 })
  60.         //天气列表
  61.         Row() {
  62.           ForEach(this.casts, (cast: casts) => {
  63.             Column() {
  64.               Text(cast.date.substring(5))
  65.               this.weatherImage(cast.dayweather)
  66.               Blank()
  67.               Text(cast.daytemp.toString())
  68.               Line()
  69.                 .width(20).height(80).startPoint([10, 0])
  70.                 .endPoint([10, 70]).stroke(Color.Black)
  71.                 .strokeWidth(3).strokeDashArray([10, 3])
  72.               Text(cast.nighttemp.toString())
  73.               Blank()
  74.               this.weatherImage(cast.dayweather)
  75.             }.height("90%").width("20%")
  76.           })
  77.         }
  78.         .width("80%")
  79.         .height("60%")
  80.         .backgroundColor("#ffbab8b8")
  81.         .opacity(0.5)
  82.         .justifyContent(FlexAlign.SpaceAround)
  83.       }.height("45%").width("100%")
  84.     }.width("100%").height("100%")
  85.   }
  86. }
复制代码

5. 都会添加与删除

提供用户界面让用户可以或许添加新的都会到都会列表,同样,提供用户界面让用户可以或许从都会列表中删除都会。
  1. import router from '@ohos.router'
  2. interface GeneratedTypeLiteralInterface_1 {
  3.   Codes: number[];
  4. }
  5. interface GeneratedTypeLiteralInterface_2 {
  6.   Names: string[];
  7. }
  8. @Entry
  9. @Component
  10. struct AddCity {
  11.   @State AllCityCodeList :Array<number> = [110000,120000,130000,140000,210000,220000,310000]
  12.   @State AllCityNameList :Array<string> = ["北京市","天津市","河北省","山西省","辽宁省","吉林省","上海市"]
  13.   //当前城市代码列表 接收传入数据的载体
  14.   @State cityCodeList :Array<number> = []
  15.   @State cityNameList :Array<string> = []
  16.   onPageShow(){
  17.     let params = router.getParams()
  18.     //(user as { name: string }).name
  19.     this.cityCodeList = (params as GeneratedTypeLiteralInterface_1).Codes
  20.     this.cityNameList = (params as GeneratedTypeLiteralInterface_2).Names
  21.   }
  22.   build() {
  23.     //加入
  24.     Column(){
  25.       Row(){
  26.         Text("添加城市列表").fontSize(35).fontColor(Color.White)
  27.         Blank()
  28.         Button("完成").backgroundColor("").fontSize(26)
  29.           .onClick(()=>{
  30.             router.back({
  31.               url:"pages/Index",
  32.               params:{
  33.                 Codes:this.cityCodeList
  34.               }
  35.             })
  36.           })
  37.       }.height("10%").width("95%")
  38.       //城市列表
  39.       Column() {
  40.         List() {
  41.           ForEach(this.AllCityNameList, (name: string) => {
  42.             ListItem() {
  43.               if (this.cityNameList.includes(name)) {
  44.                 Column() {
  45.                   Row() {
  46.                     Text(name).fontSize(35).fontColor(Color.White).width("60%")
  47.                       .margin({ top: 20, left: 30 })
  48.                     Blank()
  49.                     Text("已添加").backgroundColor("").fontSize(18)
  50.                       .margin({ right: 5 })
  51.                       .opacity(0.8)
  52.                   }.width("100%")
  53.                   Blank()
  54.                   Divider().strokeWidth(5)
  55.                 }.height(90).width("100%").margin({ top: 20 })
  56.                 .backgroundColor("#4682B4")
  57.               } else {
  58.                 Column() {
  59.                   Row() {
  60.                     Text(name).fontSize(35).fontColor(Color.White).width("60%")
  61.                       .margin({ top: 20, left: 30 })
  62.                     Blank()
  63.                     Button("添加").backgroundColor("").fontSize(18)
  64.                       .margin({ right: 5 })
  65.                       .onClick(() => {
  66.                         //根据name 获取所在索引
  67.                         let index = this.AllCityNameList.findIndex(obj => obj === name)
  68.                         console.log("index:"+index)
  69.                         //根据索引获得 城市对应的编码
  70.                         let cityCode: number = this.AllCityCodeList[index]
  71.                         console.log("cityCode= "+cityCode)
  72.                         //将编码加入列表
  73.                         this.cityCodeList.push(cityCode)
  74.                         this.cityNameList.push(name)
  75.                         console.log(this.cityCodeList.toString())
  76.                       })
  77.                   }.width("100%")
  78.                   Blank()
  79.                   Divider().strokeWidth(5)
  80.                 }.height(90).width("100%").margin({ top: 20 })
  81.                 .backgroundColor("#87CEEB")
  82.               }
  83.             }
  84.           })
  85.         }
  86.       }.width("100%").height("90%")
  87.     }.width("100%").height("100%").backgroundColor("#87CEFA")
  88.   }
  89. }
复制代码
6.功能展示


 


7.项目不足与经验总结
 


7.1项目计划不足

错误处理机制不统一:差别的请求有差别的错误处理方式,缺少统一的管理。
数据缓存策略不美满:缓存过期时间与革新机制不敷灵活,可能导致数据滞后。
网络请求与组件耦合:网络请求和 UI 组件耦合过深,应该解耦。
过度依赖外部 API:对第三方 API 依赖过高,缺少容错机制和备用方案。
UI 计划与用户体验欠缺:UI 交互和细节优化不足,加载提示和数据展示有待改进。
气候数据展示条理不清晰:展示的气候数据缺乏优先级和条理感。
缺少单元测试与主动化测试:缺少美满的单元测试和主动化测试,难以及时发现问题。
7.2经验总结

模块化计划:功能模块化进步代码可维护性和可扩展性。
公道使用数据缓存:优化缓存机制,减少网络请求,进步用户体验。
UI 交互优化:通过动态效果和反馈提拔用户体验。
提前规划错误处理:统一的错误处理机制提拔代码可靠性。
第三方 API 备选方案:避免过度依赖单一 API,增长容错机制。
清晰的数据结构界说:明确数据格式和结构,包管一致性。
可扩展性:计划时考虑将来功能扩展,保持灵活性。
参考与主要借鉴网址:GitHub - YeMengLiChou/LuckWeather-HarmonyApp: 鸿蒙课设开发——好气候App

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

嚴華

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表