Android推送

写这文章之前,先介绍一下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

勾选自定义行为,并添加


Android推送_第1张图片
Paste_Image.png
2.百度云推送点击通知后进入不了想要的页面

此问题跟上一个问题一样,是因为没有配置好百度推送的高级设置。

相关DEMO代码
1.AndroidManifest.xml的主线程需要添加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 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();
            }
        }
    }    

另外,小米华为手机可能要另外配置:要去小米华为官网搞配置问题。建议是用融云推送。

你可能感兴趣的:(Android推送)