基于Android平台开发,天气预报APP

1.项目功能思维导图

在这里插入图片描述

2. 项目涉及到的技术点

  1. 数据来源:和风天气API
  2. 使用okhttp网络请求框架获取api数据
  3. 使用gson库解析json数据
  4. 使用RecyclerView+adapter实现未来7天列表展示和天气指数
  5. 使用PopupMenu 实现弹出选项框
  6. 使用动画+定时器实现欢迎页倒计时和logo动画
  7. 使用TextToSpeech 实现语音播报

3.项目截图

请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述
请添加图片描述

4.部分代码功能实现

  1. 欢迎页实现
public class WelcomeActivity extends AppCompatActivity {private TextView tvCountdown;private CardView card_logo;private CountDownTimer countDownTimer;private long timeLeftInMillis = 1000; // 设置倒计时时长,单位为毫秒@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_welcome);//初始化控件tvCountdown = findViewById(R.id.tv_countdown);card_logo = findViewById(R.id.card_logo);// 启动倒计时startCountdown();//实现logo缩放动画startAnim();}private void startAnim() {ViewCompat.animate(card_logo).scaleX(1.0f).scaleY(1.0f).setDuration(1000).setListener(new ViewPropertyAnimatorListener() {@Overridepublic void onAnimationStart(View view) {}@Overridepublic void onAnimationEnd(View view) {}@Overridepublic void onAnimationCancel(View view) {}}).start();}private void startCountdown() {countDownTimer = new CountDownTimer(timeLeftInMillis, 1000) {@Overridepublic void onTick(long millisUntilFinished) {timeLeftInMillis = millisUntilFinished;int secondsRemaining = (int) (millisUntilFinished / 1000);tvCountdown.setText(secondsRemaining + "s | 跳转");}@Overridepublic void onFinish() {//跳转到登录页面(看自己逻辑想跳转哪个页面)startActivity(new Intent(WelcomeActivity.this, MainActivity.class));// 倒计时结束后的操作,例如跳转到主页面finish();}}.start();}@Overrideprotected void onDestroy() {super.onDestroy();if (countDownTimer != null) {countDownTimer.cancel();}}
}
  1. 天气指数
public class IndicesActivity extends AppCompatActivity {private String city_id;private RecyclerView recyclerView;private IndicesListAdapter mIndicesListAdapter;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_indices);//获取跳转传值city_id = getIntent().getStringExtra("city_id");//获取生活指数getWeatherIndices(city_id);//初始化控件initViews();//初始化适配器mIndicesListAdapter = new IndicesListAdapter();//设置适配器recyclerView.setAdapter(mIndicesListAdapter);//设置监听器setListener();}/*** 初始化控件*/private void initViews() {recyclerView = findViewById(R.id.recyclerView);}/*** 设置监听器*/private void setListener() {findViewById(R.id.toolbar).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {finish();}});}/*** 获取生活指数*/private void getWeatherIndices(String city_id) {OkGo.<String>get("https://devapi.qweather.com/v7/indices/1d").params("location", city_id).params("key", ApiConstants.APP_KEY).params("type", "0").execute(new StringCallback() {@Overridepublic void onStart(Request<String, ? extends Request> request) {super.onStart(request);ProgressDialogUtils.showProgressDialog(IndicesActivity.this);}@Overridepublic void onSuccess(Response<String> response) {IndicesInfo indicesInfo = new Gson().fromJson(response.body(), IndicesInfo.class);if (null != indicesInfo && indicesInfo.getCode().equals("200")) {mIndicesListAdapter.setIndicesInfoList(indicesInfo.getDaily());}}@Overridepublic void onFinish() {super.onFinish();ProgressDialogUtils.hideProgressDialog();}});}
}
  1. 城市搜索
public class SearchActivity extends AppCompatActivity {private EditText et_search_city;private RecyclerView recyclerView;private LinearLayoutCompat ll_empty;private SearchListAdapter mSearchListAdapter;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_search);// 1. 初始化控件initViews();//创建适配器mSearchListAdapter = new SearchListAdapter();//设置适配器recyclerView.setAdapter(mSearchListAdapter);// 2. 点击事件setListener();}/*** 初始化控件*/private void initViews() {et_search_city = findViewById(R.id.et_search_city);recyclerView = findViewById(R.id.recyclerView);ll_empty = findViewById(R.id.ll_empty);}/*** 点击事件*/private void setListener() {findViewById(R.id.btn_search).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {// 1. 获取输入框的值String cityName = et_search_city.getText().toString().trim();// 2. 判断是否为空if (cityName.isEmpty()) {// 提示用户Toast.makeText(SearchActivity.this, "城市名不能为空", Toast.LENGTH_SHORT).show();} else {searchCity(cityName);}}});//返回findViewById(R.id.toolbar).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {finish();}});//recyclerView点击事件mSearchListAdapter.setOnItemClickListener(new SearchListAdapter.OnItemClickListener() {@Overridepublic void onItemClick(CityLocationInfo.LocationDTO locationDTO) {// 1. 获取城市名String cityName = locationDTO.getName();Intent intent = new Intent();intent.putExtra("cityName", cityName);intent.putExtra("id", locationDTO.getId());//设置跳转回传的值setResult(1000, intent);// 3. 关闭当前界面finish();}});}/*** 城市搜索*/private void searchCity(String cityName) {OkGo.<String>get("https://geoapi.qweather.com/v2/city/lookup").params("location", cityName).params("key", ApiConstants.APP_KEY).execute(new StringCallback() {@Overridepublic void onStart(Request<String, ? extends Request> request) {super.onStart(request);ProgressDialogUtils.showProgressDialog(SearchActivity.this);}@Overridepublic void onSuccess(Response<String> response) {CityLocationInfo cityLocationInfo = new Gson().fromJson(response.body(), CityLocationInfo.class);if (null != cityLocationInfo && cityLocationInfo.getCode().equals("200")) {if (null != mSearchListAdapter) {mSearchListAdapter.setCityLocationInfoList(cityLocationInfo.getLocation());}//判断是否显示空布局if (mSearchListAdapter.getItemCount() > 0) {ll_empty.setVisibility(View.GONE);} else {ll_empty.setVisibility(View.VISIBLE);}} else {Toast.makeText(SearchActivity.this, "未查询到该城市", Toast.LENGTH_SHORT).show();}}@Overridepublic void onError(Response<String> response) {super.onError(response);}@Overridepublic void onFinish() {super.onFinish();ProgressDialogUtils.hideProgressDialog();}});}
}

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

python入门综合篇—资源爬取与exe打包(图形界面)

了解我的人都知道&#xff0c;我是一个谨言慎行且兴趣爱好广泛的IT&#xff0c;作为一个合格的前端&#xff0c;没事捣鼓一下python很合理吧&#xff0c;再没事搞搞java和php也很合乎逻辑吧&#xff0c;实在没事&#xff0c;玩玩linux服务器也是合乎常理的吧。所以&#xff0c;…

UML类图的概述

新书速览|《UML 2.5基础、建模与设计实践》新书速览|《UML 2.5基础、建模与设计实践 类是对一组具有相同属性、操作、关系和语义的对象的抽象&#xff0c;主要包括名称&#xff08;Name&#xff09;部分、属性&#xff08;Attribute&#xff09;部分和操作&#xff08;Operati…

六个Python实用技巧,大幅提升你的编程效率!

关注星标&#xff0c;每天学习Python新技能 Python作为当下最流行的编程语言之一&#xff0c;以其简洁易懂的语法和强大的第三方库支持&#xff0c;深受开发者的喜爱。在实际开发中&#xff0c;掌握一些实用的Python小技巧&#xff0c;可以大大提升编程效率和代码质量。 本文…

windwos下mysql的udf提权

MySql UDF提权介绍 UDF(User Defined Functions)即用户自定义函数&#xff0c;通过这种方式可以实现命令执行&#xff0c;其原理是通过lib_mysqludf_sys提供的函数可以执行系统命令 攻击场景:同之前利用日志写WebShell的场景&#xff0c;即堆叠注入或MySQL终端权限或类似phpMy…

钉钉扫码登录第三方

钉钉文档 实现登录第三方网站 - 钉钉开放平台 (dingtalk.com) html页面 将html放在 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>登录</title>// jquery<script src"http://code.jqu…

LeetCode刷题之HOT100之最长递增子序列

2024/7/10 晴&#xff0c;睡眠质量良好&#xff0c;到实验室时间9.18。知了在窗外聒噪&#xff0c;似乎让我安心&#xff0c;静下来。做题吧 1、题目描述 2、算法分析 给一个整数数组&#xff0c;要求出里面最长严格递增子序列的长度。遇到这种问题&#xff0c;想到的就是DP算…

【逆向基础】九、dnSpy使用技巧随记

一、dnSpy逆向工具的使用 1、反汇编适用范围&#xff1a;C#,.NET等语言编写的程序 2、工具的获取&#xff1a;dnSpy (ps:大家可自行去网页搜索下载最新版) 3、打开需要反汇编的程序&#xff0c;成功后出现如图所示的界面 4、dnSpy反汇编.NET程序后&#xff0c;可以像开发一样…

算术运算符用途解析及应用案例

文章目录 常用的算术运算符及其用途&#xff1a;运算符优先级类型转换高级用法 应用案例1. 计算器程序2. 平方根计算3. 计算平均数和标准差4. 货币兑换5. 计算几何6. 动力学模拟7. 数字图像处理8. 金融计算&#xff1a;复利计算 常用的算术运算符及其用途&#xff1a; 算术运算…

怎么简单快捷的分享文件呢?扫描二维码看文件的制作方法

怎么简单快捷的分享文件呢&#xff1f;想要快速的实现文件分享&#xff0c;那么可以将文件转成二维码之后&#xff0c;通过分享二维码让其他人扫码在手机上查看文件&#xff0c;可以将单个文件或者多个文件生成二维码&#xff0c;扫描点击文件就能够在手机上预览或者下载文件。…

【JavaScript】深入理解Promise:从基础概念到进阶用法、手写promise

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 一、引言二、Promise概述1. Promise的定义2. Promise的用途3. Promise的三种状态4. Promise的构造函数和基础结构5. Promise的优点6. Promise的实例方法7. Promise的静态方法 三、Promise的基本用法1. 创建一个Promise2. th…

居家客服人员分散,更需要统一客服话术

1、居家客服服务需求激增 近年来&#xff0c;随着线上消费的兴起&#xff0c;以及客服人员成本的不断攀升&#xff0c;越来越多的企业选择雇佣居家客服&#xff0c;以客服服务发包的形式接待客户的咨询。因此&#xff0c;居家客服人员的数量也逐渐增加。然而&#xff0c;居家办…

【探索Linux】P.38(传输层 —— TCP协议通信连接管理机制简介 | TCP连接状态转换)

阅读导航 引言一、TCP协议通信连接管理机制二、连接状态转换1. TCP状态转换图2. 状态转换过程3. 理解TIME_WAIT状态&#xff08;1&#xff09;目的和作用&#xff08;2&#xff09;状态转换&#xff08;3&#xff09;特殊情况&#xff08;4&#xff09;影响和优化 4. 理解 CLOS…

纯技术分享:淘宝商品详情原数据接口参数解析

item_get_app-获得淘宝app商品详情原数据 公共参数 名称类型必须描述keyString是调用key&#xff08;必须以GET方式拼接在URL中&#xff09;secretString是调用密钥api_nameString是API接口名称&#xff08;包括在请求地址中&#xff09;[item_search,item_get,item_search_s…

【React】Ant Design -- Table分页功能实现

实现步骤 为Table组件指定pagination属性来展示分页效果在分页切换事件中获取到筛选表单中选中的数据使用当前页数据修改params参数依赖引起接口重新调用获取最新数据 const pageChange (page) > {// 拿到当前页参数 修改params 引起接口更新setParams({...params,page})…

免费GPU——Google Colab使用

免费GPU——Google Colab使用 1、创建新的Notebook 网址&#xff1a;https://colab.research.google.com/ 点击“新建笔记本”进行创建 2、设置免费GPU 点击“更改运行时类型”&#xff0c;打开界面如下所示&#xff1a; 选择“T4 GPU”&#xff0c;然后“保存”即可使用…

通用图形处理器设计GPGPU基础与架构(一)

GPGPU背景 GPGPU&#xff08;General Purpose Graphics Processing Unit&#xff0c;通用图形处理器&#xff09;脱胎于GPU (Graphics Processing Unit&#xff0c;图形处理器&#xff09;。GPGPU由于其强大的运算能力和高度灵活的可编程性&#xff0c;已经成为深度学…

大模型应用中什么是SFT(监督微调)?

大模型应用中什么是SFT&#xff08;监督微调&#xff09;&#xff1f; 一、SFT的基本概念 监督微调&#xff08;Supervised Fine-Tuning, SFT&#xff09;是对已经预训练的模型进行特定任务的训练&#xff0c;以提高其在该任务上的表现。预训练模型通常在大量通用数据上进行训…

模型和应用,哪个更重要?

前言 2024 年 7 月 4 日&#xff0c;世界人工智能大会暨人工智能全球治理高级别会议全体会议在上海世博中心举行。百度创始人李彦宏在产业发展主论坛上发言&#xff0c;呼吁不要卷模型&#xff0c;要卷应用。 目录 四个要点 积极的观点 不合理性 总结 四个要点 李彦宏的呼吁…

【经典链表OJ】环形链表

一、题目要求 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&…

2024年上海市安全员C3证证考试题库及上海市安全员C3证试题解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年上海市安全员C3证证考试题库及上海市安全员C3证试题解析是安全生产模拟考试一点通结合&#xff08;安监局&#xff09;特种作业人员操作证考试大纲和&#xff08;质检局&#xff09;特种设备作业人员上岗证考试…