关键词:
文章目录
前言
今天来讲讲推送这件小事,事虽小,要做好却不容易。
推送难,难于上青天。
我们在讨论 Android 手机上的推送时,大多数情况是在说集成第三方推送,因为即使是像微信这样的大厂,也需要厂商加到启动白名单里才能保持在线。
iOS 手机使用 APNs(Apple Push Notification service
)进行推送,而 Android 手机,也是有 GCM(Google Cloud Messaging
)作为 Google 官方的推送支持的,但是在国内需要翻墙才能使用,并且需要手机安装了 Google Service
,条件比较苛刻。
GCM 最新版本叫 FCM(Firebase Cloud Messaging
)
推送的实现方式
1. C2DM
Cloud to Device Messaging,云端推送,是Android系统级别的消息推送服务(Google出品)
- 原理
基于Push方式 - 具体描述
C2DM服务负责处理诸如消息排队等事务,并向运行于目标设备上的应用程序分发这些消息。如下图:
- C2DM原理
-
优点
C2DM提供了一个简单的、轻量级的机制,允许服务器可以通知移动应用程序直接与服务器进行通信,以便于从服务器获取应用程序更新和用户数据。 -
缺点
1.依赖于Google官方提供的C2DM服务器,但在国内使用Google服务需要翻墙,成本较大;
2.需要用户手机安装Google服务。但由于Android机型、系统的碎片化 & 国内环境,国内的Android系统都自动去除Google服务,假如要使用C2DM服务,这意味着用户还得去安装Google服务,成本较大。
-
2. 轮询
-
原理
基于Pull方式 -
具体描述
应用程序隔固定时间主动与服务器进行连接并查询是否有新的消息 -
优点
实时性好 -
缺点
成本大,需要自己实现与服务器之间的通信,例如消息排队等;
到达率不确定,考虑轮询的频率:太低可能导致消息的延迟;太高,更费客户端的资源(CPU资源、网络流量、系统电量)和服务器资源(网络带宽)
3. SMS信令推送
- 原理
基于Push方式 - 具体描述
服务器有新消息时,发送1条类似短信的信令给客户端,客户端通过拦截信令,解析消息内容 / 向服务器获取信息 - 优点
可实现完全的实时操作 - 缺点
成本高(主要是短信资费的支出)
4. MQTT协议
- 定义
轻量级的消息发布/订阅协议 - 原理
基于Push方式,wmqtt.jar 是IBM提供的MQTT协议的实现,原理如下图:
- 项目实例
AndroidPushNotificationsDemo
5. XMPP协议
-
定义
Extensible Messageing and Presence Protocol
,可扩展消息与存在协议,是基于可扩展标记语言(XML)的协议,是目前主流的四种IM协议之一 -
原理流程
-
优点
1.开源:可通过修改其源代码来适应我们的应用程序。
2.简单:XML易于解析和阅读;将复杂性从客户端转移到了服务器端
3.可拓展性强:继承了在XML环境中灵活的发展性,可进一步对协议进行扩展,实现更为完善的功能。 -
缺点
如果将消息从服务器上推送出去,则不管消息是否成功到达客户端手机上。
6. 使用第三方平台
现今主流的推送平台分为:
- 手机厂商类:小米推送、华为推送。
- 第三方平台类:友盟推送、极光推送、云巴(基于MQTT)
- BAT大厂的平台推送:阿里云移动推送、腾讯信鸽推送、百度云推送
推荐阅读:Android消息推送:第三方消息推送平台详细解析
Android 中 MQTT 的使用
Android中使用MQTT需要使用到Paho Android Service库,Paho Android Service是一个用Java编写的MQTT客户端库。
GitHub地址:https://github.com/eclipse/paho.mqtt.android
1. 集成
- 在 module 的 build.gradle 文件中添加依赖:
repositories
maven
url "https://repo.eclipse.org/content/repositories/paho-snapshots/"
dependencies
compile 'org.eclipse.paho:org.eclipse.paho.client.mqttv3:1.1.0'
compile 'org.eclipse.paho:org.eclipse.paho.android.service:1.1.1'
- 在
AndroidManifest.xml
添加限权
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
- 在 AndroidManifest.xml 注册 Service
<service android:name="org.eclipse.paho.android.service.MqttService" /> <!--MqttService-->
<service android:name="com.dongyk.service.MyMqttService"/> <!--MyMqttService-->
2. 具体代码
2.1 Android中使用MQTT最主要的就是以下几个方法:
- connect:连接MQTT服务器,这里主要讲3个参数的方法,如下:
@Override
public IMqttToken connect(MqttConnectOptions options, Object userContext,
IMqttActionListener callback) throws MqttException
//...
参数options:用来携带连接服务器的一系列参数,例如用户名、密码等。
参数userContext:可选对象,用于向回调传递上下文。一般传null即可。
参数callback:用来监听MQTT是否连接成功的回调
- publish:发布消息,这里使用四个参数的方法,如下:
@Override
public IMqttDeliveryToken publish(String topic, byte[] payload, int qos,
boolean retained) throws MqttException, MqttPersistenceException
//...
参数topic:发布消息的主题
参数payload:消息的字节数组
参数qos:提供消息的服务质量,可传0、1或2
参数retained:是否在服务器保留断开连接后的最后一条消息
- subscribe:订阅消息,这里主要讲2个参数的方法,如下:
@Override
public IMqttToken subscribe(String topic, int qos) throws MqttException,
MqttSecurityException
//...
参数topic:订阅消息的主题
参数qos:订阅消息的服务质量,可传0、1或2
2.2 MQTT服务——MyMqttService
public class MyMqttService extends Service
public final String TAG = MyMqttService.class.getSimpleName();
private static MqttAndroidClient mqttAndroidClient;
private MqttConnectOptions mMqttConnectOptions;
public String HOST = "tcp://192.168.0.102:61613";//服务器地址(协议+地址+端口号)
public String USERNAME = "admin";//用户名
public String PASSWORD = "password";//密码
public static String PUBLISH_TOPIC = "tourist_enter";//发布主题
public static String RESPONSE_TOPIC = "message_arrived";//响应主题
@RequiresApi(api = 26)
public String CLIENTID = Build.VERSION.SDK_INT >= Build.VERSION_CODES.O
? Build.getSerial() : Build.SERIAL;//客户端ID,一般以客户端唯一标识符表示,这里用设备序列号表示
@Override
public int onStartCommand(Intent intent, int flags, int startId)
init();
return super.onStartCommand(intent, flags, startId);
@Nullable
@Override
public IBinder onBind(Intent intent)
return null;
/**
* 开启服务
*/
public static void startService(Context mContext)
mContext.startService(new Intent(mContext, MyMqttService.class));
/**
* 发布 (模拟其他客户端发布消息)
*
* @param message 消息
*/
public static void publish(String message)
String topic = PUBLISH_TOPIC;
Integer qos = 2;
Boolean retained = false;
try
//参数分别为:主题、消息的字节数组、服务质量、是否在服务器保留断开连接后的最后一条消息
mqttAndroidClient.publish(topic, message.getBytes(), qos.intValue(), retained.booleanValue());
catch (MqttException e)
e.printStackTrace();
/**
* 响应 (收到其他客户端的消息后,响应给对方告知消息已到达或者消息有问题等)
*
* @param message 消息
*/
public void response(String message)
String topic = RESPONSE_TOPIC;
Integer qos = 2;
Boolean retained = false;
try
//参数分别为:主题、消息的字节数组、服务质量、是否在服务器保留断开连接后的最后一条消息
mqttAndroidClient.publish(topic, message.getBytes(), qos.intValue(), retained.booleanValue());
catch (MqttException e)
e.printStackTrace();
/**
* 初始化
*/
private void init()
String serverURI = HOST; //服务器地址(协议+地址+端口号)
mqttAndroidClient = new MqttAndroidClient(this, serverURI, CLIENTID);
mqttAndroidClient.setCallback(mqttCallback); //设置监听订阅消息的回调
mMqttConnectOptions = new MqttConnectOptions();
mMqttConnectOptions.setCleanSession(true); //设置是否清除缓存
mMqttConnectOptions.setConnectionTimeout(10); //设置超时时间,单位:秒
mMqttConnectOptions.setKeepAliveInterval(20); //设置心跳包发送间隔,单位:秒
mMqttConnectOptions.setUserName(USERNAME); //设置用户名
mMqttConnectOptions.setPassword(PASSWORD.toCharArray()); //设置密码
// last will message
boolean doConnect = true;
String message = "\\"terminal_uid\\":\\"" + CLIENTID + "\\"";
String topic = PUBLISH_TOPIC;
Integer qos = 2;
Boolean retained = false;
if ((!message.equals("")) || (!topic.equals("")))
// 最后的遗嘱
try
mMqttConnectOptions.setWill(topic, message.getBytes(), qos.intValue(), retained.booleanValue());
catch (Exception e)
Log.i(TAG, "Exception Occured", e);
doConnect = false;
iMqttActionListener.onFailure(null, e);
if (doConnect)
doClientConnection();
/**
* 连接MQTT服务器
*/
private void doClientConnection()
if (!mqttAndroidClient.isConnected() && isConnectIsNomarl())
try
mqttAndroidClient.connect(mMqttConnectOptions, null, iMqttActionListener);
catch (MqttException e)
e.printStackTrace();
/**
* 判断网络是否连接
*/
private boolean isConnectIsNomarl()
ConnectivityManager connectivityManager = (ConnectivityManager) this.getApplicationContext().getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo info = connectivityManager.getActiveNetworkInfo();
if (info != null && info.isAvailable())
String name = info.getTypeName();
Log.i(TAG, "当前网络名称:" + name);
return true;
else
Log.i(TAG, "没有可用网络");
/*没有可用网络的时候,延迟3秒再尝试重连*/
new Handler().postDelayed(new Runnable()
@Override
public void run()
doClientConnection();
, 3000);
return false;
//MQTT是否连接成功的监听
private IMqttActionListener iMqttActionListener = new IMqttActionListener()
@Override
public void onSuccess(IMqttToken arg0)
Log.i(TAG, "连接成功 ");
try
mqttAndroidClient.subscribe(PUBLISH_TOPIC, 2);//订阅主题,参数:主题、服务质量
catch (MqttException e)
e.printStackTrace();
@Override
public void onFailure(IMqttToken arg0, Throwable arg1)
arg1.printStackTrace();
Log.i(TAG, "连接失败 ");
doClientConnection();//连接失败,重连(可关闭服务器进行模拟)
;
//订阅主题的回调
private MqttCallback mqttCallback = new MqttCallback()
@Override
public void messageArrived(String topic, MqttMessage message) throws Exception
Log.i(TAG, "收到消息: " + new String(message.getPayload()));
//收到消息,这里弹出Toast表示。如果需要更新UI,可以使用广播或者EventBus进行发送
Toast.makeText(getApplicationContext(), "messageArrived: " + new String(message.getPayload()), Toast.LENGTH_LONG).show();
//收到其他客户端的消息后,响应给对方告知消息已到达或者消息有问题等
response("message arrived");
@Override
public void deliveryComplete(IMqttDeliveryToken arg0)
@Override
public void connectionLost(Throwable arg0)
Log.i(TAG, "连接断开 ");
doClientConnection();//连接断开,重连
;
@Override
public void onDestroy()
try
mqttAndroidClient.disconnect(); //断开连接
catch (MqttException e)
e.printStackTrace();
super.onDestroy();
该 MyMqttService
类的大概逻辑就是开启服务后,调用init()
方法初始化各个参数,包括服务器地址、用户名、密码等等,然后调用doClientConnection()
方法连接MQTT服务器,iMqttActionListener
用来监听MQTT是否连接成功,连接成功则订阅主题。mqttCallback
为订阅主题的回调,收到消息后会执行该回调中的messageArrived()
方法,拿到消息后进行UI更新,并调用response()
方法响应给对方告知消息已到达或者消息有问题等。
2.3 开启服务
在MainActivity
中开启服务,这里为了方便不做UI更新,所以就一行开启服务的代码,如下:
public class MainActivity extends AppCompatActivity
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
MyMqttService.startService(this); //开启服务
3. 项目地址
androidapp必备高级功能,消息推送之mqtt(代码片段)
...公众号《鸿洋》原创首发,转载请务必注明出处。1.Android端实现消息推送的几种方式轮询:客户端定时向服务器请求数据。伪推送。缺点:费电,费流量。拦截短信消息。服务器需要向客户端发通知时,发送... 查看详情
uniappapp消息推送方案(代码片段)
...限*/setPermissionsInform()//#ifdefAPP-PLUSif(plus.os.name=='Android')//判断是Androidvarmain=plus.android.runtimeMainActivity();varpkName=main.getPackageName();varuid=main.getApplicationInfo().plusGetAttribute("uid");varNotificationManagerCompat=plus 查看详情
xamarin学习系列之极光消息推送(代码片段)
...orms的消息推送集成的资料非常少,下面就说下Xamarin.Forms(Android)怎么集成极光推送准备工作:1、了解极光推送原理:https://docs.jiguang.cn/jpush/client/Android/android_sdk/2、下载极 查看详情
阿里云推送服务(代码片段)
移动推送提供从云端到移动终端的优质推送服务,支持Android和iOS平台的通知/消息的推送功能.推送内容及模式通知:会自动在手机端弹出通知栏,用户可以打开或者清除通知栏。iOS的通知走APNs(苹果官方推送通知服务),android走... 查看详情
php实现app消息推送(代码片段)
如何用php实现APP消息推送现在有很多的消息推送厂商,比如阿里云的消息推送,极光推送,融云的消息推送。他们的原理都是把sdk内置在app里面,达到消息推送的目的,通过一张图来了解一下,看不懂不要... 查看详情
php实现app消息推送(代码片段)
...;话不多说开干,详看官方文档API部分php使用极光推送项目引入极光SDKcomposerrequirejpush/jpush简单使用useJPush\\ClientasJPush;useJPush\\Client; //测试推送publicfunctionjpush()$app_key="111111";//这是app密钥,填你自己 查看详情
ios推送手机消息背后的技术(代码片段)
作者:allenzzhao,腾讯 IEG运营开发工程师消息推送我们几乎每天都会用到,但你知道iOS中的消息推送是如何实现的吗?本文将从推送权限申请,到本地和远程消息推送,再到App对推送消息的处理等多个步骤,详细介绍iOS中消... 查看详情
开发者必备个推《app消息推送白皮书》正式发布|附下载
消息到达率、点击率,如何提升?智能、精细、友好的用户触达,如何实现?促活跃、增留存、降成本,有何方法?......答案尽在个推《APP消息推送白皮书》。3月29,个推《APP消息推送白皮书》(... 查看详情
android小米推送服务集成(代码片段)
...面看了。我的小米手机型号是MI8,MIUI版本12.0.4,Android版本10.0。正 查看详情
android小米推送服务集成(代码片段)
...面看了。我的小米手机型号是MI8,MIUI版本12.0.4,Android版本10.0。正 查看详情
android项目必备(四十)-->app实现定位功能(代码片段)
...录前言1.添加权限2.定位大致分为三大类3.实例代码前言在Android开发中。常需要用到定位功能,尤其是依赖于地理位置功能的应用。非常多人喜欢使用百度地图,高德地图提供的SDK。开放API,可是在只须要经纬度,... 查看详情
程序员debug必备神器—每日bug看板(附源码)(代码片段)
...云实现在线通知功能,并见证一下MQTT+即时通知在项目开发管理中是如何敏捷而出色地满足多人云协作需求的!随着数字化服务的发展,应用系统与终端用户之间需要构建稳定消息通道,且消息的灵活性与实效... 查看详情
android--项目必备app实现定位功能(代码片段)
前言在Android开发中。常需要用到定位功能,尤其是依赖于地理位置功能的应用。非常多人喜欢使用百度地图,高德地图提供的SDK。开放API,可是在只须要经纬度,或者城市,街道地址等信息。并不须要提供预... 查看详情
android集成友盟推送功能(代码片段)
友盟是中国最大的移动开发者服务平台,为移动开发者提供免费的应用统计分析、社交分享、消息推送、自动更新、在线参数、移动推广效果分析、微社区等app开发和运营解决方案。如何快速集成友盟推送功能:1.注册友盟账... 查看详情
推送通知项目记录
1.对于App设置中的开关通知有对应的unregisterForRemoteNotifications取消注册2.iOS我在通知栏里清除了所有通知,但是应用上的角标还是存在的.点击icon进入app是不会获取到通知栏的消息的。在通知栏那里点击清除消息也没法影响到badge... 查看详情
h5做的app怎么实现将消息推送到状态栏,求大神指点
...做了一个消息推送到状态栏的Demo,但是一放入用H5做的app项目中运行后没有任何效果追答肯定是代码有问题,因为不管是codover还是其他苹果,打包都是基于原生项目的。 查看详情
个推测试结果+前后端如何分工(代码片段)
1、下图是我在个推平台中设置推送,ios和android的接收测试情况:测试机型和版本:ios:iphone6splus,ios12.1.4android:红米5plus,android8.1.0注意:1.1、根据项目需求,选择推送的消息类型。比如,我们的项目需求是,ios和android在前台运... 查看详情
java开源项目消息推送平台发送一条短信(代码片段)
...员👨🏻💻常年被誉为优质八股文选手austin项目实现的第一个渠道::从发送短信开始01、短信介绍在项目介绍的时候,已经定义了austin项目的核心功能:发送消息我认为,短信是在一整个消息推... 查看详情