美文网首页移动开发狂热者(299402133)android开发技巧安卓知识
Android实现点击通知栏后,先启动应用再打开目标Activi

Android实现点击通知栏后,先启动应用再打开目标Activi

作者: slimhippo | 来源:发表于2015-08-06 18:29 被阅读68604次

    情况简述

    在开发Android app的过程中,遇到这样一个需求:app中启动一个Service,该Service在独立进程中运行,与服务器保持长连接,将服务器推送过来的消息在通知栏中显示,并设置点击动作,点击后跳转到app中对应的Activity。目前遇到的问题是Service以独立进程运行,在收到消息并弹出通知后,app本身的进程有两种情况:

    1. app正在运行
    2. app已退出

    对于第一种情况,处理就非常简单了,直接将参数传入Intent并打开对应的Activity即可。

    但第二种情况比较复杂,因为app已经退出,而要打开的Activity中的某些操作是需要依赖app的初始化的,这些初始化操作是在app启动过程中进行的。举个例子,一个购物应用推送了某个新商品的消息,用户点击通知后进入商品详情的Activity,而该Activity中有个订购Button,点击该Button后就会从本地中获取用户的Id等信息并发一条消息给服务器,告诉服务器某用户订购了该商品。这些用户信息是在app启动时与服务器进行一系列交互后取得的。如果app退出后直接进入详情Activity并点击购买,就会因为获取不到用户信息而出错。

    所以目前要解决的问题时,在Notification中设置点击动作,如果app本身正在运行,直接跳转到目标Activity;如果app已经退出,先启动app完成初始化,再跳转到目标Activity。

    方案和思路

    我们假设目前有三个Activity:

    1. SplashActivity 用于显示app大图,同时进行用户登录等操作,服务器返回数据后跳转到MainActivity。
    2. MainActivity app的主Activity。
    3. DetailActivity MainActivity中点击Button进入的Activity,用于显示某件商品详情。

    而弹出通知的Service在另外一个进程中。

    我们要达到的目的是:

    1. 点击通知栏通知,假如app正在运行,则直接跳转到DetailActivity显示具体内容,在DetailActivity中按Back键返回MainActivity
    2. 点击通知栏通知,假如app已经退出,先从SplashActivity进入,显示app启动界面,初始化操作完成后进入MainActivity再跳转到DetailActivity显示具体内容,在DetailActivity中按Back键返回MainActivity。

    初步的思路是先判断app进程是否存在,如果存在的话,就利用startActivities启动MainActivity和DetailActivity。为什么还要启动MainActivity而不直接只启动DetailActivity?因为有如下情况,进程中的所有Activity都已经退出了,但进程还没有被系统回收,这时判断进程是否存在返回true,然后只启动DetailActivity的话,按Back键任务栈就直接到底,返回桌面了。而我们要的效果是按Back键返回上一级Activity,也就是MainActivity。

    如果app进程已经退出,不存在了,此时就用一个Intent启动应用,该Intent中包含一个Bundle, Bundle中存有启动DetailActivity所需的参数,这个Intent传入SplashActivity后,再由SplashActivity传给MainActivity,在MainActivity中加入判断,如果有该参数,则表示应用是从通知栏启动的,要进行跳转到DetailActivity的操作,否则就是常规启动。

    代码实现

    有了大概的实现思路后,大家来个demo实际操作一下。
    首先,我们的demo有简单的组件:

    1. PushService,在新进程中启动的Service,负责监听服务器,收到服务器的信息后将消息广播出去,在本demo中,为了简化,只是简单的广播一个消息
    2. ShowNotificationReceiver,在新进程中注册的BroadcastReceiver,收到PushService发的消息后,会在通知栏弹出通知
    3. NotificationReceiver, 在新进程中注册的BroadcastReceiver,用来设置点击通知栏通知的动作,打开app中的某个Activity
    4. SplashActivity, app启动页面,先是启动图片,3s后进入MainActivity
    5. MainActivity,app的主Activity
    6. DetailActivity,app中显示详情的Activity

    PushService.java

    首先是PushService,要在新进程中启动,要在AndroidManifest.xml中加入以下注册Service的代码

    <service android:name=".PushService"
                     android:process=":push"/>
    

    PushService的工作很简单,启动后发一个广播在通知栏显示通知,然后常驻在后台

    public class PushService extends Service{
        @Nullable
        @Override
        public IBinder onBind(Intent intent) {
            return null;
        }
    
        @Override
        public void onCreate() {
            super.onCreate();
            Log.i("PushService", "PushService onCreate");
            //用AlarmManager定时发送广播
            AlarmManager am = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
            
            Intent intent = new Intent(this, ShowNotificationReceiver.class);
    
            PendingIntent pendingIntent =
                    PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
    
            am.set(AlarmManager.ELAPSED_REALTIME, SystemClock.currentThreadTimeMillis(), pendingIntent);
    
        }
    
    }
    

    ShowNotificationReceiver.java

    这个广播类用来在通知栏弹出通知

    public class ShowNotificationReceiver extends BroadcastReceiver{
        private static final String TAG = "RepeatReceiver";
        @Override
        public void onReceive(Context context, Intent intent) {
            Log.d(TAG, "ShowNotificationReceiver onReceive");
            //设置点击通知栏的动作为启动另外一个广播
            Intent broadcastIntent = new Intent(context, NotificationReceiver.class);
            PendingIntent pendingIntent = PendingIntent.
                    getBroadcast(context, 0, broadcastIntent, PendingIntent.FLAG_UPDATE_CURRENT);
            
            NotificationCompat.Builder builder = new NotificationCompat.Builder(context);
            builder.setContentTitle("这就是通知的头")
                    .setTicker("这是通知的ticker")
                    .setContentIntent(pendingIntent)
                    .setSmallIcon(android.R.drawable.ic_lock_idle_charging);
    
            Log.i("repeat", "showNotification");
            NotificationManager manager = (NotificationManager)context.getSystemService(Context.NOTIFICATION_SERVICE);
            manager.notify(2, builder.build());
        }
    }
    

    NotificationReceiver.java

    点击通知栏后,会发送一个广播,NotificationReceiver收到该广播后,就会判断,app进程是否仍然存活,根据app进程的不同状态,定义不同的app启动方式

    public class NotificationReceiver extends BroadcastReceiver{
    
        @Override
        public void onReceive(Context context, Intent intent) {
            //判断app进程是否存活
            if(SystemUtils.isAppAlive(context, "com.liangzili.notificationlaunch")){
                //如果存活的话,就直接启动DetailActivity,但要考虑一种情况,就是app的进程虽然仍然在
                //但Task栈已经空了,比如用户点击Back键退出应用,但进程还没有被系统回收,如果直接启动
                //DetailActivity,再按Back键就不会返回MainActivity了。所以在启动
                //DetailActivity前,要先启动MainActivity。
                Log.i("NotificationReceiver", "the app process is alive");
                Intent mainIntent = new Intent(context, MainActivity.class);
                //将MainAtivity的launchMode设置成SingleTask, 或者在下面flag中加上Intent.FLAG_CLEAR_TOP,
                //如果Task栈中有MainActivity的实例,就会把它移到栈顶,把在它之上的Activity都清理出栈,
                //如果Task栈不存在MainActivity实例,则在栈顶创建
                mainIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    
                Intent detailIntent = new Intent(context, DetailActivity.class);
                detailIntent.putExtra("name", "电饭锅");
                detailIntent.putExtra("price", "58元");
                detailIntent.putExtra("detail", "这是一个好锅, 这是app进程存在,直接启动Activity的");
    
                Intent[] intents = {mainIntent, detailIntent};
    
                context.startActivities(intents);
            }else {
                //如果app进程已经被杀死,先重新启动app,将DetailActivity的启动参数传入Intent中,参数经过
                //SplashActivity传入MainActivity,此时app的初始化已经完成,在MainActivity中就可以根据传入             //参数跳转到DetailActivity中去了
                Log.i("NotificationReceiver", "the app process is dead");
                Intent launchIntent = context.getPackageManager().
                        getLaunchIntentForPackage("com.liangzili.notificationlaunch");
                launchIntent.setFlags(
                        Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
                Bundle args = new Bundle();
                args.putString("name", "电饭锅");
                args.putString("price", "58元");
                args.putString("detail", "这是一个好锅, 这是app进程不存在,先启动应用再启动Activity的");
                launchIntent.putExtra(Constants.EXTRA_BUNDLE, args);
                context.startActivity(launchIntent);
            }
        }
    }
    

    SplashActivity.java

    SplashActivity.java先是app启动的图片,3s后进入MainActivity, 如果启动SplashActivity的Intent中带有参数,就将参数取出,放入启动MainActivity的Intent中

    public class SplashActivity extends AppCompatActivity{
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_splash);
            //隐藏ActionBar
            getSupportActionBar().hide();
            //使用handler倒数3秒后进入MainActivity
            new Handler().postDelayed(new Runnable() {
                @Override
                public void run() {
                    Intent intent = new Intent(SplashActivity.this, MainActivity.class);
                    //如果启动app的Intent中带有额外的参数,表明app是从点击通知栏的动作中启动的
                    //将参数取出,传递到MainActivity中
                    if(getIntent().getBundleExtra(Constants.EXTRA_BUNDLE) != null){
                        intent.putExtra(Constants.EXTRA_BUNDLE,
                                getIntent().getBundleExtra(Constants.EXTRA_BUNDLE));
                    }
                    startActivity(intent);
                    finish();
                }
            }, 3000);
        }
    }
    

    MainActivity.java

    MainActivity中,如果有参数传入,就在初始化结束后,根据参数启动DetailActivity,如果没有参数传入,就此结束自己的任务

    public class MainActivity extends AppCompatActivity {
        private static final String TAG = "MainActivity";
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            Intent intent = new Intent(this, PushService.class);
            startService(intent);
            setTitle("MainActivity");
            Bundle bundle = getIntent().getBundleExtra(Constants.EXTRA_BUNDLE);
            if(bundle != null){
                //如果bundle存在,取出其中的参数,启动DetailActivity
                String name = bundle.getString("name");
                String price = bundle.getString("price");
                String detail = bundle.getString("detail");
                SystemUtils.startDetailActivity(this, name, price, detail);
                Log.i(TAG, "launchParam exists, redirect to DetailActivity");
            }
        }
    
        @Override
        protected void onResume() {
            super.onResume();
        }
    
        @Override
        public boolean onCreateOptionsMenu(Menu menu) {
            // Inflate the menu; this adds items to the action bar if it is present.
            getMenuInflater().inflate(R.menu.menu_main, menu);
            return true;
        }
    
        @Override
        public boolean onOptionsItemSelected(MenuItem item) {
            // Handle action bar item clicks here. The action bar will
            // automatically handle clicks on the Home/Up button, so long
            // as you specify a parent activity in AndroidManifest.xml.
            int id = item.getItemId();
    
            //noinspection SimplifiableIfStatement
            if (id == R.id.action_settings) {
                return true;
            }
    
            return super.onOptionsItemSelected(item);
        }
    }
    

    DetailActivity.java

    比较简单,显示传入的参数即可:-D

    public class DetailActivity extends AppCompatActivity{
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_detail);
            getSupportActionBar().setTitle("DetailActivity");
            String name = getIntent().getStringExtra("name");
            String price = getIntent().getStringExtra("price");
            String detail = getIntent().getStringExtra("detail");
    
            ((TextView)findViewById(R.id.name)).setText(name);
            ((TextView)findViewById(R.id.price)).setText(price);
            ((TextView)findViewById(R.id.detail)).setText(detail);
        }
    }
    

    效果展示

    http://v.youku.com/v_show/id_XMTMwMjgyNTUwMA==.html?from=y1.7-1.2

    demo下载

    https://github.com/slimhippo/androidcode

    相关文章

      网友评论

      • nevgip:楼主,我想问个问题,就是进程被关掉后,发通知的时候还能收到吗?或者说还是显示在手机屏幕上吗?
      • 大柚子的故事:简主,这个github找不到代码库
      • krmao:app 被杀死,启动 splash 会初始化 application ?
      • 917a0938d905:为啥写这么多广播,不写广播难道不会省掉这一个步骤么
      • 8a1d14916178:楼主,你好!按照你的步骤,能够正常跳转到指定页面,但是,有一个问题是,当我在DetailActivity点击返回按钮时,还是一直返回到DetailActivity页面,而不是跳转到MainActivity中,请问这是什么原因?
      • ThinkNuo:在android 7.0 通知栏有多条通知会聚合成一条 这时点击通知只会跳转到MainActivity
      • 曹半斤:进程被杀死了是不是可以不考虑 。。。
      • nixi8888:太好了,找了老长时间终于找到了:smile:
      • 你是暖阳:特别感谢简主,刚好头疼这个需求。:wink:
      • 粉条好吃不啦:楼主,当进程被杀死的时候服务还存在吗?如果服务不在,又如何能接收推送消息
        30aa02bae807:可能在收到推送消息后才被杀死的呀
      • 608924568ff8:现在遇到一个问题 点击通知栏进入app和点击launcher app icon进入app效果不一样,通过点击通知栏进入app方式 按home的效果会和回退效果一样了
      • 0208a5be9de6:可以把主activity设置单例的,然后把notice的intent直接发送到这个activity上是不是也可以
      • 0f38058111bf:点击通知,无法跳转到Activity
        NotificationManager mNotifyMgr = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);

        Intent intent;
        int smallIcon;
        if (type == 1) {
        intent = new Intent(context, ChattingActivity.class);
        smallIcon = R.drawable.ic_launcher;
        } else {
        intent = new Intent(context, SystemMessageActivity.class);
        smallIcon = R.drawable.icon_message_system;
        }
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        PendingIntent contentIntent=PendingIntent.getActivity(context, 110, intent, PendingIntent.FLAG_UPDATE_CURRENT);
        NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(context).setSmallIcon(smallIcon)
        .setContentTitle(title).setContentText(content).setFullScreenIntent(contentIntent, true)
        .setAutoCancel(true);
        mNotifyMgr.notify(999, mBuilder.build());
      • 王人冉:楼主,你说的长连接怎么实现的?进程被杀了呢?
      • 景阳_jy:如果我有30个页面,我要跳转到第30个页面,一直这样intent传参?

        问题1:你知道我上次是从哪个页面按home键,进入后台的?
        问题2:就算知道,像这么一层一层的传得多大的工作量?
        问题3:进程是存活的,但界面全都finish掉了;比如退出app后手动停止进程,然后在双进程拉活的模式下又激活了,这种情况,你怎么能跳MainActivity?
        景阳_jy:@三棵柚子 都尼玛什么时候的评论,还不是看到头像还真不知道是我当初写下的
        三棵柚子:我不是楼主:看到你近乎"作为测试态度提问题"的问题, 给出我的想法!
        对于前两个问题:楼主说的进入主页面是因为要拿到用户信息,你弄三十个页面是什么意思呢?难道你要同时初始化三十个页面才能进到想要进入的页面?难道你的Activity不能单独的开启么?为了能够准确的打开,一般都会提前埋点, 而不是要通过无数次intent传递 ,那样非常low.
      • 36cf8531f05d:我目前测试出的情况是,只要你的receiver起来了,就说明你的应用进程已经存在了。所以在简主这种判断方式下,我的进程是已经存在的。所以在点击通知时,我使用了android里activity栈的管理方式,如果activity栈大小为0,此时去启动应用;否则如果栈中不存在MainActivity,说明此时是在登录界面或者初始界面,再使用简主的方法就可以实现简主的目的了。
      • 36cf8531f05d:使用第三方推送比如极光推送,通过几个应用之间互相拉起推送服务,这时候收到推送点击,用简主的方法,会判断成app进程是存在的,导致没有去启动应用,这该怎么解决呢
        32a81ff89504:我也是貌似这种,一直判断进程在运行的
      • ed1e218bf627:楼主思路不错,想请教另一个问题,你的service在一个单独的进程,你在项目中,service所在的进程和主进程是如果进行数据交互的?哪怕是简单的数据彼此之间的获取?
      • 呃哈哈:楼主请问一下,这个是不是还需要判断安卓系统?我在4.0上在appkill时候不能用 但是在5.0系统以后就可以,能给点意见吗
        呃哈哈:@Lxzzzz 没有办法,谷歌已经对这个屏蔽了。
        635a1473907f:@我是不是你大爷 请问你解决了吗亲
      • 囧_囧:思路很清晰,3Q
      • 7e4c836c667e:谢谢楼主^-^
      • 242e60066e39:楼主,你好。我遇到个问题:在应用关闭的时候,一下子推送过来多条notification通知,点击其中一条能正常跳转,然后再点其他的没反应。应用在前台(正在和用户交互)或者后台(按home键到桌面)却不会发生这种bug。这是为什么呀,困惑了我好久。
        3866b3f2e166:@kieedi 大神,怎么解决的,求具体说一下,谢谢!
        242e60066e39:@夏日的金鱼花火 我的问题解决了,是个超级奇怪的坑。如果遇到,请在未来意图中加这个Intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
        d133872055db:@kieedi 你每条通知的id没设好吧
      • 小池laucherish:思路很清晰,代码很详细,赞一个。
        是否可以考虑创建任务栈实现呢?
        TaskStackBuilder
      • 木木潇潇:楼主,我的是从创建在;桌面快捷方式进去的,怎么搞啊
      • 4ca6cba16a9c:楼主你好,跟你的思路写了,但是不知为什么我长按home键将应用给关掉后,后面的广播通知也收不到了,请问你那会有这样的情况么,如果会,可否教下怎么让把程序关掉后依然可以收到通知,谢谢
        逍遥wqy:@telic 如果你是动态注册的广播,进程被杀后就不行了,你修改为静态注册广播试下
      • 爱吃薯片的人:根据楼主的思路又写了个实现点击跳转的NotificationReceiver广播,但是没有点击通知栏消息的时候,NotificationReceiver也会收到广播,造成每次点击通知栏,进入的都是最后一次通知的地方。
        b95acc560dbf:我也遇到这个问题,你解决了吗,如何解决
      • MrLuo:用了楼主的思路,但是在4.2中Android加入了多用户 ,4.2以上的版本好像就不用从通知栏进入到APP了。
      • 67d4f5752320:麻烦楼主收到回复一下 谢谢。
      • pomelom:收藏了0.0
      • 4037db44dc4e:不错 ,和微信的流程 一样 有没有更优雅的方式 ,楼主可以分享一下
        4037db44dc4e:@slimhippo 期待楼主 更好作品
        slimhippo:@吸烟有益健康 原理大概就是这样了,实现方面可以封装一下,封装成接口让Activity继承,最近实在是忙,忙完了再写一篇相关的好了 :blush:
      • 6f4d7532ffae:感谢楼主,一只在研究这方面的技术,终于找到了
        67d4f5752320:@slimhippo 楼主有新的方法吗?我也是按照你那种方法,当应用进程杀死之后,应用并没有重新启动,只能直接跳到了那个通知页面去了。
        浩林Leon:@slimhippo 有一个更优雅的方法吗?最近我也有这方面的一个需求
        slimhippo:@shawshank 离写完这篇文章又过去了一个多月,随着app的功能扩展,觉得这种定死了的实现方式实在不方面扩展,正在想有没有更优雅的方法,或者做成一个library :smile:
      • dde9cb1ddd6f:你好,你的demo在什么环境开发的,eclipse跑步了
        slimhippo:@zhongliping 换用Android Studio吧亲

      本文标题:Android实现点击通知栏后,先启动应用再打开目标Activi

      本文链接:https://www.haomeiwen.com/subject/sbhbqttx.html