写这文章之前,先介绍一下Android与IOS推送的区别
Android
APP未运行,不可以接收到通知。
APP已运行,放置在后台,可以接收到通知。
APP已运行,正在前台使用,可以接收到通知。
结束APP进程,不可以接收到通知。
IOS
APP未运行,可以接收到通知。
APP已运行,放置在后台,可以接收到通知。
APP已运行,正在前台使用,不可以接收到通知。
通过上面的对比我们很明显看到IOS的推送做得比较好,尤其在安卓6.0+各种权限下,经常出现推送不到或者推送延迟的情况。有一些公司使用自己平台的推送框架,如果是做自己公司的推送框架,需要注意一下以下几点:
1.采用什么协议?目前常用就是XMPP以及MQTT或者自定义的二进制协议。
2.推送的内容?(单纯的存文本还是图文或者是包含语言?)
3.长连接的保活问题。
4.数据的加密解密
5.传输过程数据是否会丢失?或者客户端网络请求不稳定如何处理?
常用的一些第三方推送平台
融云
小米推送(MiPush)
华为推送(华为Push)
友盟推送(U-Push)
个推
极光推送
阿里云移动推送(Alibaba Cloud Channel Service)
腾讯信鸽推送
百度云推送
Bmob云推送
平台那么多我们该选哪家?
上面的推送,大致分为手机厂商推送、平台推送、第三方推送。
手机厂商推送:
他们的推送服务在他们自己生产的手机上属于系统级别的服务,手机系统对他们自家的推送限制最小。如小米手机上,不在系统自启动名单里的App,在手机重启后,App声明的后台Service并不会自动运行。但是,小米推送作为手机系统级服务,仍然是可以收到推送的。又如华为,内部团队放话:华为Push,在华为手机上是系统级别的服务,稳定性等各方面肯定都会好些。
第三方平台推送:
第三方平台推送主要看平台的规模,例如百度推送,用百度推送的公司多了,就算你APP在后台,依然可以收到推送。至于用哪种推送好,就看你们自己了,小编我就只用过两种推送:百度推送和极光推送,感觉也能满足公司的要求了。写这篇文章的时候用的是:Baidu-Push-SDK-Android-L2-5.6.0.30。
几个新闻端的推送
2016.12.31的版本。
友盟推送(今日头条、央视新闻、一点资讯、ZAKER新闻)
个推推送(新浪新闻、网易新闻)
百度推送 (百度新闻)
信鸽推送的(掌上英雄联盟、欢乐斗地主 )
关于推送的一些细节问题
1、如果主应用已经被杀死并且不存在任何Activity后就无法跳到用户指定面板;
2、只有在主应用未被杀死的情况下可以跳到用户上次依然打开的面板;
3、如果主应用被杀死后点击通知栏依然可以重新启动主应用效果;
4、从推送平台的Service外部进程打开Activity必须指定FLAG_ACTIVITY_NEW_TASK标记;
1.百度推送收到推送后点开通知栏会重新打开APP
勾选自定义行为,并添加<action android:name="com.baidu.android.pushservice.action.notification.CLICK" />
2.百度云推送点击通知后进入不了想要的页面
此问题跟上一个问题一样,是因为没有配置好百度推送的高级设置。
相关DEMO代码
1.AndroidManifest.xml的主线程需要添加android:launchMode="singleTask"
<activity
android:name=".MainActivity"
android:label="@string/app_name"
android:launchMode="singleTask"
>
2.判断APP进程是否存在
/**
* 判断应用是否已经启动
* @param context 一个context
* @param packageName 要判断应用的包名
* @return boolean
*/
public static boolean isAppAlive(Context context, String packageName){
ActivityManager activityManager =
(ActivityManager)context.getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningAppProcessInfo> processInfos
= activityManager.getRunningAppProcesses();
for(int i = 0; i < processInfos.size(); i++){
if(processInfos.get(i).processName.equals(packageName)){
//app进程存在
return true;
}
}
//app还没启动
return false;
}
3.启动页
public class SplashActivity extends AppCompatActivity{
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_splash);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
Intent intent = new Intent(SplashActivity.this, MainActivity.class);
if(getIntent().getBundleExtra(Constants.EXTRA_BUNDLE) != null){
//判断是否推送进来的
intent.putExtra(Constants.EXTRA_BUNDLE,
getIntent().getBundleExtra(Constants.EXTRA_BUNDLE));
}
startActivity(intent);
finish();
}
}, 2000);
}
}
4.主页
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Bundle bundle = getIntent().getBundleExtra(Constants.EXTRA_BUNDLE);
if(bundle != null){
//推送进来 打开详情页
String goodsId= bundle.getString("goodsId");
Intent detialIntent = new Intent(MainActivity.this,Detial.class);
detialIntent.putExtra("goodsId",goodsId);
startActivity(detialIntent);
}
}
}
5.百度的MyPushMessageReceiver
@Override
public void onNotificationClicked(Context context, String title,
String description, String customContentString) {
// 通知点击
// 自定义内容获取方式,mykey和myvalue对应通知推送时自定义内容中设置的键和值
if (!TextUtils.isEmpty(customContentString)) {
JSONObject customJson = null;
try {
customJson = new JSONObject(customContentString);
String goodsId = null;
//校验传过来的值
if (!customJson.isNull("goodsId")) {
goodsId = customJson.getString("goodsId");
//判断app进程是否存活
if(UiUtils.isAppAlive(context)){
//如果存活的话,就直接启动DetailActivity,但要考虑一种情况,就是app的进程虽然仍然在
//但Task栈已经空了,比如用户点击Back键退出应用,但进程还没有被系统回收,如果直接启动
//DetailActivity,再按Back键就不会返回MainActivity了。所以在启动
//DetailActivity前,要先启动MainActivity。
Intent mainIntent = new Intent(context, GuideActivity.class);
//将MainAtivity的launchMode设置成SingleTask, 或者在下面flag中加上Intent.FLAG_CLEAR_TOP,
//如果Task栈中有MainActivity的实例,就会把它移到栈顶,把在它之上的Activity都清理出栈,
//如果Task栈不存在MainActivity实例,则在栈顶创建
//清除MainActivity上所有界面
//mainIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
mainIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Intent detailIntent = new Intent(context, DetialActivity.class);
detailIntent.putExtra("goodsId", "goodsId");
Intent[] intents = {mainIntent, detailIntent};
context.startActivities(intents);
}else{
//如果app进程已经被杀死,先重新启动app,将DetailActivity的启动参数传入Intent中,参数经过
//SplashActivity传入MainActivity,此时app的初始化已经完成,在MainActivity中就可以根据传入 //参数跳转到DetailActivity中去了
Intent launchIntent = context.getPackageManager().
getLaunchIntentForPackage("包名");
launchIntent.setFlags(
Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
Bundle args = new Bundle();
args.putString("goodsId", "goodsId");
launchIntent.putExtra(Constants.EXTRA_BUNDLE, args);
context.startActivity(launchIntent);
}
}
} catch (JSONException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
另外,小米华为手机可能要另外配置:要去小米华为官网搞配置问题。建议是用融云推送。