重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
极光推送能实现;所谓的消息推送就是服务器和移动端保持连接,通过定期传送一定的信息。比如一些关于新闻客户端,每隔一段时间就会收到一条或者多条的信息通知,这就是从服务器推送过来的消息。
为浈江等地区用户提供了全套网页设计制作服务,及浈江网站建设行业解决方案。主营业务为成都网站设计、网站制作、浈江网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!
消息推送的好处
1、提高用户的活跃度;消息推送是获得用户的关注,打开App激活使用的绝佳途径。
2、提高用户的留存率;就是唤醒沉睡用户,挽留流失用户,对于提高用户留存率都是有重要意义。
3、带动功能模块使用率;用户对新功能的点击、发现率低,而消息推送正是一个引导用 户关注及使用体验的途径。
使用消息推送系统推荐使用深圳极光家的系统。是中国领先的开发者服务提供商,专注于为开发者提供稳定高效的消息推送、一键认证以及流量变现等服务,助力开发者的运营、增长与变现。
目前极光推送的船新版本是3.0.3,具体的集成步骤就不多说了,按照官网的步骤来就可以了.
之前用自定义的消息去写,那么自己就要写通知,当时的思路是在自定义的消息里面去写,这样拓展程度高.自己定义的receiver如下:
结果到最后,感觉还是用极光的推送好一些,我在小米4的测试机上面测试自定义的消息,当自定义的推送消息推送过来的时候,不需要点击就执行了这个自定义消息的点击意图,不知道为什么,在别的模拟器上面没有发现这个问题,然后自己写的Notification也有缺陷,悬浮式通知不能自己隐藏,而且自己写的通知只能显示一条,新推送的消息会覆盖掉上一条,应该是 mNotifyMgr.notify(buiderID,notify);的时候没有指定不同的buiderID,可以将buiderID每次都指定不同的,这样应该就不会覆盖了,但是没有去验证.
极光推送技术挺不错。极光推送具有以下优势:
1、便捷的使用体验
快速集成SDK,简单易用的控制台和API
2、灵活的目标筛选
提供用户自定义的标签和别名系统,以及极光自己根据数据分析出的分类目标
3、高效稳定的系统
支持10亿级的高并发访问,多点备份保证系统稳定
4、专业的支持
有专业的技术支持团队,及时响应客户的需求和问题
极光可定制的私有云:对于安全性要求更高,希望推送数据和系统存储在自己服务器的客户,及个性化需求需要定制开发的,性能更高要求的,或者想拥有自己推送平台的甚至要求源码授权二次开发的开发者,极光提供全功能的私有云解决方案。
极光推送安全包:为金融、新闻、政务及其他对推送安全要求极高的客户提供安全严谨、稳定可靠的信息推送解决方案。
应用场景:在我们的项目中我们用的是创建视频会议和预警消息通知推送这两种情况,首先创建视频会议和中途邀请人参加会议我采用的是自定义推送消息,因为此应用场景传递的参数比较多,在推送过去之后android和ios可以很方便的将参数传递过去,但是我们考虑到邀请的所有人都在登录状态的时候才可以创建会议房间,我们采用的是腾讯视频会议的sdk,因为我们采用的是别名推送,当用户在登录的时候安卓端将用户uuid设置为别名,退出时取消别名,我在服务端根据传递过来的uuid利用别名查询registration_ids的数组大小是否为空判断当前用户是否设置别名,当邀请的所有人都设置了别名之后我才调用创建会议房间的接口。而预警消息推送我采用的是通知推送,当气象预警触发的时候我会采用通知推送的方式推送给app,这种方式比较简单。
极光推送是给app推送消息的,我们首先需要在服务端集成maven依赖
!-- 极光推送 --
groupIdcn.jpush.api
artifactIdjpush-client
version3.2.3
/dependency
接着我们需要了解极光推送有哪些推送方式,对于安卓和ios都适用的情况,我在项目中使用的是别名推送alias,(还有标签推送tag),推送方式又分为通知推送和自定义推送,通知推送能够显示在手机提示框中,而自定义推送却不能,我看安卓他们做的能跟微信视频一样的弹出一个会话框,看着挺不错的,这跟微信不同的是可以邀请多个,之前用阿里云的好像只能一对一,所以才换成腾讯视频会议的,好了,废话不多说,上我写的一个工具类,
package com.jpxx.homepage.homePage.service.utils;
import cn.jpush.api.JPushClient;
import cn.jpush.api.push.PushResult;
import cn.jpush.api.push.model.Options;
import cn.jpush.api.push.model.Platform;
import cn.jpush.api.push.model.PushPayload;
import cn.jpush.api.push.model.audience.Audience;
import cn.jpush.api.push.model.notification.Notification;
import cn.jpush.api.push.model.Message;
import cn.jpush.api.push.model.PushPayload.Builder;
public class SendMessageUtils {
private static StringAppKey="8a7880c6fb81ad494b224078";
/**
* JPush MasterSecret 极光推送平台生成的密钥
*/
private static StringMasterSecret="c0fc675c4c48f9bf35269cf4";
//两个参数分别填写你申请的masterSecret和appKey
private static JPushClientjPushClient=new JPushClient(MasterSecret,AppKey);
/**
* 通知推送
* 备注:推送方式不为空时,推送的值也不能为空;推送方式为空时,推送值不做要求
* @param type 推送方式:1、“tag”标签推送,2、“alias”别名推送
* @param value 推送的标签或别名值
* @param alert 推送的内容
*/
public static StringpushNotice(String type,String title,String value,String alert,int roomId,String MessageType,String name,String promoterAdavter,String meetingTitle,String sig){
Builder builder= PushPayload.newBuilder();
builder.setPlatform(Platform.all());//设置接受的平台,all为所有平台,包括安卓、ios、和微软的
//设置如果用户不在线、离线消息保存的时间
Options options=Options.sendno();
options.setTimeToLive(86400l); //设置为86400为保存一天,如果不设置默认也是保存一天
builder.setOptions(options);
builder.setMessage(Message.newBuilder()
.setMsgContent(value)
.setTitle(title)
.addExtra("roomId",roomId)
.addExtra("MessageType",MessageType)
.addExtra("name",name)
.addExtra("promoterAdavter",promoterAdavter)
.addExtra("meetingTitle",meetingTitle)
.addExtra("userSig",sig)
.build());
//设置推送方式
if(type.equals("alias")){
builder.setAudience(Audience.alias(value));//根据别名推送
}else if(type.equals("tag")){
builder.setAudience(Audience.tag(value));//根据标签推送
}else{
builder.setAudience(Audience.all());//Audience设置为all,说明采用广播方式推送,所有用户都可以接收到
}
//设置为采用通知的方式发送消息
builder.setNotification(Notification.alert(alert));
PushPayload pushPayload=builder.build();
System.out.println("ggggggg"+pushPayload);
try{
//进行推送,实际推送就在这一步
//System.out.println("zzzzzzzzz "+jPushClient.sendPush(pushPayload));
PushResult pushResult=jPushClient.sendPush(pushPayload);
return "success";
}catch(Exception e){
System.out.println("异常 "+e);
e.printStackTrace();
return "fail";
}
}
/**
* 自定义消息推送
* 备注:推送方式不为空时,推送的值也不能为空;推送方式为空时,推送值不做要求
* @param type 推送方式:1、“tag”标签推送,2、“alias”别名推送
* @param value 推送的标签或别名值
* @param alert 推送的内容
*/
public static StringpushMsg(String type,String title,String value,String alert,int roomId,String MessageType,String name,String promoterAdavter,String meetingTitle,String sig){
Builder builder= PushPayload.newBuilder();
builder.setPlatform(Platform.all());//设置接受的平台
if(type.equals("alias")){
builder.setAudience(Audience.alias(value));//别名推送
}else if(type.equals("tag")){
builder.setAudience(Audience.tag(value));//标签推送
}else{
builder.setAudience(Audience.all());//Audience设置为all,说明采用广播方式推送,所有用户都可以接收到
}
Message.Builder newBuilder=Message.newBuilder();
newBuilder.setMsgContent(alert);//消息内容
newBuilder.setTitle(title);
newBuilder.addExtra("roomId",roomId);
newBuilder.addExtra("MessageType",MessageType);
newBuilder.addExtra("name",name);
newBuilder.addExtra("promoterAdavter",promoterAdavter);
newBuilder.addExtra("meetingTitle",meetingTitle);
newBuilder.addExtra("userSig",sig);
Message message=newBuilder.build();
builder.setMessage(message);
PushPayload pushPayload=builder.build();
try{
PushResult pushResult=jPushClient.sendPush(pushPayload);
System.out.println(pushResult.isResultOK());
return "success";
}catch(Exception e){
e.printStackTrace();
return "fail";
}
}
public static void main(String[] args) {
//给标签为kefu的用户进行消息推送
//SendMessageUtils.pushMsg("alias","标题","5b9022746e284ea0992e3baa983035dc","你有新的任务,请及时处理",111,"meetingType","name","avater","meetingTitle","");
//sendNotificationWirhAlias_Ios("zzzzzzzzzzzz","uuid");
//String result = SendMessageUtils.pushNotice("alias","标题","5b9022746e284ea0992e3baa983035dc","你有新的任务,请及时处理",111,"meetingType","name","avater","meetingTitle","");
//System.out.println("返回结果"+result);
/*String result = SendMessageUtils.pushNotice("alias","预警标题","5b9022746e284ea0992e3baa983035dc","dddd",0,"warnType","","","",""); //userDto.getUuId()
System.out.println("result "+result);*/
//根据uuid查询别名信息是否存在
String result = HttpRequestUtil.JGUtil("", "8a7880c6fb81ad494b224078","c0fc675c4c48f9bf35269cf4");//get请求
//String result = HttpRequestUtil.interfaceUtil("", "","ydswtapp","eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJ7XCJjbGllbnRJZFwiOlwieWRzd3RhcHBcIixcImxvZ2luVGltZVwiOjE1NTYwMDY1NDU4MzksXCJ1c0lkXCI6MjM5OSxcInVzZXJSb2xlXCI6XCJBRE1JTlwiLFwidXVJZFwiOlwiNDAyODgxZTUzYzdmMGRkZTAxM2M3ZjI5ZWQ4ZTAwMTZcIn0iLCJpc3MiOiJhdXRoMCIsImV4cCI6MTg3MTM2NjU0NSwiaWF0IjoxNTU2MDA2NTQ1fQ.Cy3-eDD4OEYhJlldvtJsymALRVGwP466TmBrSQJQGUo");//get请求
System.out.println("result "+result);
}
}
因为我要根据前端传来的uuid判断当前用户是否登录,所以我对http方法进行了稍微改装,根据极光的api文档(文档写的不是很好找),以下是我改装的
/**
*
* 极光专用
*/
public static StringJGUtil(String path,String appKey,String masterSecret) {
String str ="";
try {
URL url =new URL(path);
//打开和url之间的连接
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
PrintWriter out =null;
/**设置URLConnection的参数和普通的请求属性****start***/
String base64String=appKey+":"+masterSecret;
System.out.println("拼接的 "+base64String);
String str2=base64String;
String encode =new BASE64Encoder().encode(str2.getBytes());
System.out.println("编码过后:"+encode);
String auth ="Basic "+encode;
System.out.println("最终的 "+auth);
conn.setRequestProperty("Authorization", auth);
conn.setRequestProperty("Content-Type", "application/json");
//conn.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");
conn.setRequestProperty("accept", "*/*");
conn.setRequestProperty("connection", "Keep-Alive");
conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");
/**设置URLConnection的参数和普通的请求属性****end***/
//设置是否向httpUrlConnection输出,设置是否从httpUrlConnection读入,此外发送post请求必须设置这两个
//最常用的Http请求无非是get和post,get请求可以获取静态页面,也可以把参数放在URL字串后面,传递给servlet,
//post与get的 不同之处在于post的参数不是放在URL字串里面,而是放在http请求的正文内。
conn.setDoOutput(true);
conn.setDoInput(true);
conn.setRequestMethod("GET");//GET和POST必须全大写
/**GET方法请求*****start*/
/**
* 如果只是发送GET方式请求,使用connet方法建立和远程资源之间的实际连接即可;
* 如果发送POST方式的请求,需要获取URLConnection实例对应的输出流来发送请求参数。
* */
conn.connect();
/**GET方法请求*****end*/
/***POST方法请求****start*/
/*out = new PrintWriter(conn.getOutputStream());//获取URLConnection对象对应的输出流 out.print(data);//发送请求参数即数据 out.flush();//缓冲数据*/
/***POST方法请求****end*/
//获取URLConnection对象对应的输入流
InputStream is = conn.getInputStream();
//构造一个字符流缓存
BufferedReader br =new BufferedReader(new InputStreamReader(is));
while ((str = br.readLine()) !=null) {
str =new String(str.getBytes(), "UTF-8");//解决中文乱码问题
System.out.println("wwwww"+str);
return str;
}
//关闭流
is.close();
//断开连接,最好写上,disconnect是在底层tcp socket链接空闲时才切断。如果正在被其他线程使用就不切断。
//固定多线程的话,如果不disconnect,链接会增多,直到收发不出信息。写上disconnect后正常一些。
conn.disconnect();
System.out.println("完整结束");
}catch (Exception e) {
e.printStackTrace();
}
return "success";
}
好了,以后再写类似的就会写了,这里附上极光文档的链接
2.选择推送服务
厂商通道在 onCreate() 中获取 String json = getIntent().getExtras().getString("JMessageExtra");
极光通道在 receiver 中的方法 onNotifyMessageOpened() 获取 String json = notificationMessage.notificationExtras;
1、首先在极光后台用你的app包名创建应用获取极光的appkey和appid,然后点击推送配置,
2、在小米开放平台通过你的包名创建你app的应用 获取小米这里的appkey和appid
3、在极光后台找到推送配置,把小米创建的appid和appkey 还有一个App Secret 填进去然后启用
4、在你应用的app路径的build.grile将极光和小米开放平台的appid和appkey都配置进去
我本来以为这样就完成了,然后兴冲冲的去测试推送,然后就遇到以下错误
xiaomi sdk appkey or appid was empty,please check your manifest config
如果你遇到这个错误的话,就一定记住你的appid和appkey前面一定要加上
MI-你的设备
然后我我跑起来发现,又报了如下错误
register fail 22022
5、 登录小米开放平台,点击应用服务,立即体验,然后找你的应用,把推送服务给我开启
然后我又运行,发现又报错了
xiao mi push register failed - errorCode:66108,reason:appid does not match packagename!
最后排查发现小米开放平台和极光后台的包名不一致 然后 就不报错了