程序员用10分钟写了个旅游管家APP,女友用了直呼贴心

1

呐,你知道吗? 听说樱花飘落的速度是秒速五厘米哦。

听到阿珍又念起这句经典台词,阿强,这个对自然界的花期不太敏感的程序员,也收到了“樱花开了”的讯号。

春天的樱花不能错过,赏樱是写进阿珍价值观的春日打卡动作。视频通话里,阿珍计划了一场远行——周末就回母校武大看樱花。但这一次,在外出差的阿强不能陪她一起去。

于是阿强,依然是拿出了自己吃饭的本事,将自己的周到关怀写进一个智能旅游管家APP,陪阿珍去武汉看樱花。即使不在身边,也要给她无微不至的出游体验:

  1. 出发行程不用记:出发前为她提醒出行时间,让她不再错过重要行程;
  2. 目的地天气早知道:根据目的地气象预报,提前给到出游装备和穿搭建议;
  3. 攻略玩法全推荐:让她省去做攻略的时间,抵达目的地,推送所需的游玩攻略和优惠政策……

以上管家式智能出行服务能力,有了华为情景感知服务(Awareness kit)的时间感知和天气感知能力、定位服务(Location Kit)的地理围栏能力,以及推送服务(Push kit)的加持,实现起来并不难。

效果示例

点击视频查看示例

原理解释

情景感知服务能感知用户当前的时间、位置、活动状态、耳机状态、天气状况、环境光、车载连接状态、信标连接状态等场景,并通过能常驻后台运行的围栏能力,并且可自由组合这些感知能力,建立组合围栏。

定位服务采用GNSS、Wi-Fi、基站等多途径的混合定位模式进行定位,能快速、精准获取位置信息,实现全球定位服务能力。

推送服务是华为为开发者提供的消息推送平台,建立了从云端到终端的消息推送通道。开发者通过集成Push Kit可以实时推送消息到用户终端。

代码实战

1、集成情景感知服务

 开发准备

情景感知服务的集成需如下3个关键步骤,可以参考华为开发者联盟的文档

1. AppGallery Connect配置

2.集成HMS Awareness SDK

3.配置混淆脚本

https://developer.huawei.com/consumer/cn/doc/development/HMS-Guides/awareness-preparation?ha_source=hms1

代码开发关键步骤

1、在Manifest指定权限:

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />

2. 实现通过城市名获取天气信息:

String city = edCity.getText().toString();
if (city != null && !city.equals("")) {WeatherPosition weatherPosition = new WeatherPosition();weatherPosition.setCity(city);//传入指明传入地址用的语言类型,例如“zh_CN”、“en_US”等,“语言码_国家码”的格式。weatherPosition.setLocale("zh_CN");
// 获取Awareness Kit的“Capture Client”,调用天气查询能力Awareness.getCaptureClient(getApplicationContext()).getWeatherByPosition(weatherPosition).addOnSuccessListener(new OnSuccessListener<WeatherStatusResponse>() {@Overridepublic void onSuccess(WeatherStatusResponse weatherStatusResponse) {// 处理返回的天气数据WeatherStatus weatherStatus = weatherStatusResponse.getWeatherStatus();WeatherSituation weatherSituation = weatherStatus.getWeatherSituation();Situation situation = weatherSituation.getSituation();String weather;// 将天气id与天气名称匹配weather = getApplicationContext().getResources().getStringArray(R.array.cnWeather)[situation.getCnWeatherId()];// 更新UI((TextView) findViewById(R.id.tv_weather)).setText(weather);((TextView) findViewById(R.id.tv_windDir)).setText(situation.getWindDir());((TextView) findViewById(R.id.tv_windSpeed)).setText(situation.getWindSpeed() + " km/h");((TextView) findViewById(R.id.tv_temperature)).setText(situation.getTemperatureC() + "℃");}}).addOnFailureListener(new OnFailureListener() {@Overridepublic void onFailure(Exception e) {}});
}

3. 实现定时提醒和进入目的地后,接受到推送的消息:
(1) 因为需要在应用被杀死时也能接受通知,需要注册静态广播:
在Manifest中:

<receiver android:name=".BarrierReceiver">
<intent-filter><action android:name="com.test.awarenessdemo.TimeBarrierReceiver.BARRIER_RECEIVER_ACTION"/>
</intent-filter>
</receiver>

 在java代码中:

Intent intent = new Intent();
intent.setComponent(new ComponentName(MainActivity.this, BarrierReceiver.class));
mPendingIntent = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);

(2) 定义时间Barrier(围栏)和其对应的标签Label,然后添加Barrier:

// 获取输入的时间
String timeHour = edTimeHour.getText().toString();
String timeMinute = edTimeMinute.getText().toString();
int hour = 0;
int minute = 0;
if (!timeHour.equals("")) {hour = Integer.parseInt(timeHour);if (!timeMinute.equals("")) {minute = Integer.parseInt(timeMinute);}
}
long oneHourMilliSecond = 60 * 60 * 1000L;
long oneMinuteMilliSecond = 60 * 1000L;
// 定义"duringPeriodOfDay"围栏,在指定时区的指定时间段内发送通知
AwarenessBarrier periodOfDayBarrier = TimeBarrier.duringPeriodOfDay(TimeZone.getDefault(),// 这里设置的是提前2小时通知(hour - 2) * oneHourMilliSecond + minute * oneMinuteMilliSecond,hour * oneHourMilliSecond + minute * oneMinuteMilliSecond);String timeBarrierLabel = "period of day barrier laber";
// 定义更新围栏的请求
BarrierUpdateRequest.Builder builder = new BarrierUpdateRequest.Builder();
BarrierUpdateRequest request = builder.addBarrier(timeBarrierLabel, periodOfDayBarrier, mPendingIntent).build();
Awareness.getBarrierClient(getApplicationContext()).updateBarriers(request).addOnSuccessListener(new OnSuccessListener<Void>() {@Overridepublic void onSuccess(Void aVoid) {}}).addOnFailureListener(new OnFailureListener() {@Overridepublic void onFailure(Exception e) {}});

(3) 定义地理Barrier(围栏)和其对应的标签Label,然后添加Barrier:

if (city != null && !city.equals("")) {//根据城市名称获取城市经纬度,这里是从内部文件中获取String data = cityMap.get(city);if (data != null){int flag = data.indexOf(",");double latitude = Double.parseDouble(data.substring(flag+1));double longitude = Double.parseDouble(data.substring(0,flag));double radius = 50;long timeOfDuration = 5000;// 定义"stay"围栏,在进入指定区域并驻留超过指定时间后触发围栏事件上报AwarenessBarrier stayBarrier = LocationBarrier.stay(latitude, longitude, radius, timeOfDuration);String stayBarrierLabel = "stay barrier label";// 定义更新围栏的请求BarrierUpdateRequest.Builder builder = new BarrierUpdateRequest.Builder();BarrierUpdateRequest request = builder.addBarrier(stayBarrierLabel, stayBarrier, mPendingIntent).build();Awareness.getBarrierClient(getApplicationContext()).updateBarriers(request).addOnSuccessListener(new OnSuccessListener<Void>() {@Overridepublic void onSuccess(Void aVoid) {}}).addOnFailureListener(new OnFailureListener() {@Overridepublic void onFailure(Exception e) {}});}
}

(4) 定义广播接收器,用于监听Barrier事件,收到事件后进行应用的业务处理:

class BarrierReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {BarrierStatus barrierStatus = BarrierStatus.extract(intent);String label = barrierStatus.getBarrierLabel();int barrierPresentStatus = barrierStatus.getPresentStatus();String city = intent.getStringExtra("city");switch (label) {case DURING_PERIOD_OF_DAT_BARRIER_LABEL:if (barrierPresentStatus == BarrierStatus.TRUE) {initNotification(context,"1","time_channel","行程提醒","离出发还剩2小时");} else if (barrierPresentStatus == BarrierStatus.FALSE) {showToast(context, "It's not between ");} else {showToast(context, "The time status is unknown.");}break;case STAY_BARRIER_LABEL:if (barrierPresentStatus == BarrierStatus.TRUE) {initNotification(context,"2","area_channel","欢迎来到"+city,"查看旅行攻略");} else if (barrierPresentStatus == BarrierStatus.FALSE) {showToast(context,"You are not staying in the area set by locationBarrier" +" or the time of duration is not enough.");} else {showToast(context, "The location status is unknown.");}break;}}
}

2、基于位置的信息推送

 开发准备

1. 在项目级gradle里添加华为maven仓
AndroidStudio项目级build.gradle文件,增量添加如下maven地址:

buildscript {repositories {maven { url 'http://developer.huawei.com/repo/'}}
dependencies {...// 增加agcp配置。classpath 'com.huawei.agconnect:agcp:1.4.2.300'}
}allprojects {repositories {maven { url 'http://developer.huawei.com/repo/'}}
}

2. 在应用级的build.gradle里面加上SDK依赖

dependencies {implementation 'com.huawei.hms:location:5.0.2.300'implementation 'com.huawei.hms:push: 5.0.2.301'
}

关键步骤说明

 1.在AndroidManifest.xml文件里面声明系统权限
因华为定位服务采用GNSS、Wi-Fi、基站等多种混合定位模式进行定位,需要用到网络,精确的位置权限,粗略的位置权限,如果需要应用程序在后台执行时也具备持续定位能力,需要在Manifest文件中申请ACCESS_BACKGROUND_LOCATION权限:

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="com.huawei.hms.permission.ACTIVITY_RECOGNITION" />
<uses-permission android:name="android.permission.ACTIVITY_RECOGNITION" />

注:由于ACCESS_FINE_LOCATION,WRITE_EXTERNAL_STORAGE和READ_EXTERNAL_STORAGE是危险的系统权限,因此,需要动态的申请这些权限。如果权限不足,Location Service将会拒绝为应用开启定位。

2. 创建触发围栏
先根据需要创建围栏\围栏组,填好相关参数。

LocationSettingsRequest.Builder builders = new LocationSettingsRequest.Builder();
builders.addLocationRequest(mLocationRequest);
LocationSettingsRequest locationSettingsRequest = builders.build();
// Before requesting location update, invoke checkLocationSettings to check device settings.
Task<LocationSettingsResponse> locationSettingsResponseTasks = mSettingsClient.checkLocationSettings(locationSettingsRequest);
locationSettingsResponseTasks.addOnSuccessListener(new OnSuccessListener<LocationSettingsResponse>() {@Overridepublic void onSuccess(LocationSettingsResponse locationSettingsResponse) {Log.i(TAG, "check location settings success");mFusedLocationProviderClient.requestLocationUpdates(mLocationRequest, mLocationCallbacks, Looper.getMainLooper()).addOnSuccessListener(new OnSuccessListener<Void>() {@Overridepublic void onSuccess(Void aVoid) {LocationLog.i(TAG, "geoFence onSuccess");}}).addOnFailureListener(new OnFailureListener() {@Overridepublic void onFailure(Exception e) {LocationLog.e(TAG,"geoFence onFailure:" + e.getMessage());}});}
})

3. 设置触发推送通知
在GeoFenceBroadcastReceiver的onReceive接收到围栏触发成功提示后发送推送通知,在通知栏接收通知并展示。

if (geofenceData != null) {int errorCode = geofenceData.getErrorCode();int conversion = geofenceData.getConversion();ArrayList<Geofence> list = (ArrayList<Geofence>) geofenceData.getConvertingGeofenceList();Location myLocation = geofenceData.getConvertingLocation();boolean status = geofenceData.isSuccess();sb.append("errorcode: " + errorCode + next);sb.append("conversion: " + conversion + next);if (list != null) {for (int i = 0; i < list.size(); i++) {sb.append("geoFence id :" + list.get(i).getUniqueId() + next);}}if (myLocation != null) {sb.append("location is :" + myLocation.getLongitude() + " " + myLocation.getLatitude() + next);}sb.append("is successful :" + status);LocationLog.i(TAG, sb.toString());Toast.makeText(context, "" + sb.toString(), Toast.LENGTH_LONG).show();//new PushSendUtils().netSendMsg(sb.toString());
}

注:此代码围栏创建成功后会触发两次回调,conversion:1和4,分别代表进入触发回调和驻留触发回调一次。代码中输入的Trigger输入7的含义表示回调包括所有情况,即进入、驻留、离开等各种情况。

 

>>访问华为开发者联盟官网,了解更多相关内容

>>获取开发指导文档

>>华为移动服务开源仓库地址:GitHub、Gitee

关注我们,第一时间了解华为移动服务最新技术资讯~

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

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

相关文章

python实现垃圾分类程序,对于要处理的垃圾,判断该投放到哪个类别的垃圾桶中

python实现垃圾分类程序&#xff0c;对于要处理的垃圾&#xff0c;判断该投放到哪个类别的垃圾桶中 一、编程题目 编程题目&#xff1a;输入要处理的垃圾&#xff0c;空格分隔&#xff0c;判断并输出各个垃圾应该投放到哪个类别的垃圾桶中。其中垃圾有以下分类&#xff1a; 废…

Python之美[从菜鸟到高手]--Python垃圾回收机制及gc模块详解

Python中的垃圾回收是以引用计数为主&#xff0c;标记-清除和分代收集为辅。引用计数最大缺陷就是循环引用的问题&#xff0c;所以Python采用了辅助方法。本篇文章并不详细探讨Python的垃圾回收机制的内部实现&#xff0c;而是以gc模块为切入点学习Python的垃圾回收机制&#x…

【PythonGUI小程序】相信我,这是最in的n种骰子梭哈小游戏新玩法,好玩到丧心病狂~(文中有惊喜)

导语 哈喽&#xff01;大家晚上好&#xff0c;我是木木子吖&#xff0c;很久没给大家更新游戏代码的类型啦~ 所有文章完整的素材源码都在&#x1f447;&#x1f447; 粉丝白嫖源码福利&#xff0c;请移步至CSDN社区或文末公众hao即可免费。 在长沙这个美食遍地的城市&#xff…

李峋同款心跳Python代码

李峋同款心跳Python代码【按头安利《点燃我温暖你》】 import random from math import sin,cos,pi,log from tkinter import * CANVAS_WIDTH 640 CANVAS_HEIGHT 640 CANVAS_CENTER_X CANVAS_WIDTH / 2 CANVAS_CENTER_Y CANVAS_HEIGHT / 2 IMAGE_ENLARGE 11 HEART_COLOR …

【Python案例】Python实现垃圾分类APP(附带微信小程序)

嗨嗨&#xff0c;大家好呀&#xff0c;我是小圆~ 今天给你们分享一个有趣的东西 如何利用现有的工具来实现一个垃圾分类的应用 主要做了三个核心内容&#xff1a; 对比现有垃圾分类服务&#xff0c;挑选一个合适并编码实现开发桌面版垃圾分类APP开发垃圾分类微信小程序 上…

【python】制作李峋同款爱心代码,也不是很难嘛~

前言 嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! 最近&#xff0c;一部名叫《点燃我&#xff0c;温暖你》得电视剧冲进了大家得视野~ 其中李峋用代码做出的红色跳动的爱心&#xff0c;一下子跳到朱韵的心坎里&#xff0c;同样也跳到我们的心坎 今天&#xff0c;我们就用py…

太棒了 | 手把手教你用Python做一个 “举牌小人” 生成器!

教你如何使用Selenium库 本文禁止转载&#xff0c;如有违反&#xff0c;严肃处理&#xff01; 前几天写了一个婴儿级别的爬虫图文教程&#xff0c;大家很喜欢。戳我查看&#xff1a;3000字 “婴儿级” 爬虫图文教学 | 手把手教你用Python爬取 “实习网”&#xff01; 趁热打铁…

李峋的跳动爱心代码(python)

运行效果 代码&#xff1a; """ author:Athena Geng """ import random from math import sin, cos, pi, log from tkinter import *CANVAS_WIDTH 640 # 画布的宽 CANVAS_HEIGHT 480 # 画布的高 CANVAS_CENTER_X CANVAS_WIDTH / 2 # 画布中…

李峋爱心Python代码

李峋爱心Python代码&#xff1a; # codinggbk import random from math import sin, cos, pi, log from tkinter import * CANVAS_WIDTH 640 CANVAS_HEIGHT 480 CANVAS_CENTER_X CANVAS_WIDTH / 2 CANVAS_CENTER_Y CANVAS_HEIGHT / 2 IMAGE_ENLARGE 11 HEART_COLOR "…

李峋同款爱心python实现

爱心运行结果截图&#xff1a; import random from math import sin, cos, pi, log from tkinter import *CANVAS_WIDTH 840 # 画布的宽 CANVAS_HEIGHT 680 # 画布的高 CANVAS_CENTER_X CANVAS_WIDTH / 2 # 画布中心的X轴坐标 CANVAS_CENTER_Y CANVAS_HEIGHT / 2 # 画…

OpenAI CEO创建的Worldcoin正式推出「世界币」:AI 时代的数字通行证?

编辑&#xff1a;秦晋 据外媒Semafor独家报道&#xff0c;知情人士透露&#xff0c;由 OpenAI 首席执行官 Sam Altman 创建的Worldcoin 代币将于今日推出。 在硅谷引发争议的Worldcoin 代币试图解决两个棘手问题&#xff1a;在线身份认证与收入不平等问题。 据知情人士透露&…

华为鸿蒙HarmonyOS 4定档8月;ChatGPT之父的加密货币正式上线;微软必应聊天将推出重新生成答案功能|极客头条...

「极客头条」—— 技术人员的新闻圈&#xff01; CSDN 的读者朋友们早上好哇&#xff0c;「极客头条」来啦&#xff0c;快来看今天都有哪些值得我们技术人关注的重要新闻吧。 整理 | 梦依丹 出品 | CSDN&#xff08;ID&#xff1a;CSDNnews&#xff09; 一分钟速览新闻点&#…

数据分析实战(六):英国电商用户行为分析

案例&#xff1a;英国电商用户行为数据分析 Part 1. 数据获取 1.1 数据集简介 https://archive.ics.uci.edu/ml/datasets/onlineretail# 该数据集为英国在线零售商在2010年12月1日至2011年12月9日间发生的所有网络交易订单信息。 1.2 数据集内容 数据集为xlsx格式&#xff0c…

亚马逊跟卖分析与经验分享

亚马逊设置跟卖机制&#xff0c;也是给了卖家一个机会&#xff0c;对于新手来说自己制作产品的listing花费大量的时间与精力大多数出单效果很不理想&#xff0c;这个时候把握得当跟卖就会取得很大的优势。 首先它能快速的获得流量 亚马逊的跟卖就是产品的listing共享&#xff…

Lazada和Shopee选品分析之马来西亚电商市场详解-海鲸跨境

马来西亚基础信息: 马来西亚是东南亚第三大经济体,国民富足;并且年轻人众多,对中国的产品非常喜爱。 国民经济:2019你那GDP 3543亿美元,增长4.7% 人均收入:10460美元,仅次于新加坡 年龄结构:30岁以下年轻人占人口53% 电商市场规模与潜力: 马来西亚电商用户数量…

为Lazada商家量身定做的精细化运营数据分析软件,Ushop BI

跨境电商数据可视化BI系统现在在市面上也是比较普遍&#xff0c;但是针对Lazada平台的目前只有一个Ushop BI系统&#xff0c;Ushop BI系统能够把Lazada平台数据可视化做的非常好&#xff0c;其功能覆盖面广&#xff0c;操作简单&#xff0c;是Lazada商家不可多得的好帮手&#…

亚马逊商品销售数据爬虫分析报告

家电产业和消费者升级悄然地展开。 市场的这种变化使消费者对家用电器的期望不再仅仅是一个简单的功能满足&#xff0c;而是更多的细节体验和技术创新。 通过洞察家用电器的消费特点&#xff0c;有利于确定市场的未来趋势&#xff0c;从而积极应对市场变化。 ▼ 我们围绕亚马逊…

Lazada数据大分析:帮你正确解析Lazada市场!

Lazada数据大分析&#xff1a;帮你正确解析Lazada市场&#xff01; 2021年是跨境电商飞速发展的一年&#xff0c;各行各业都选择了加入跨境电商这个行业&#xff0c;而如今在互联网的大趋势下国内电商行业的发展已经非常成熟&#xff0c;而跨境电商行业就后来居上&#xff0c;…

解决TimeWait过多的问题

原文链接&#xff1a;https://www.tiejiang.org/27040.html 服务端客户端通信连接示意图 1、 time_wait的作用&#xff1a; TIME_WAIT状态存在的理由&#xff1a; 1&#xff09;可靠地实现TCP全双工连接的终止 在进行关闭连接四次挥手协议时&#xff0c;最后的ACK是由主动关闭…

Confluence 调整会话超时(session timeout)

文章目录 前言一、概括二、实际场景应用1.更改空闲超时2.更改记住我 cookie 的生命周期3.在用户通过身份验证后的某个时间强制注销用户 总结 前言 在 Confluence 中有两个会话 Cookie&#xff1a; JSESSIONID: 由 Tomcat 使用和管理。 默认情况下&#xff0c;这被视为会话 co…