【转】Service Intent must be explicit的解决方法

原文网址:http://blog.csdn.net/shenzhonglaoxu/article/details/42675287

今天在学习android的Service组件的时候,在AndroidMainfest.xml中定义了

[html] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. <service  
  2.             android:name=".BindService"  
  3.             android:enabled="true"  
  4.             android:exported="true" >  
  5.             <intent-filter>  
  6.                 <action android:name="com.example.user.firstapp.FIRST_SERVICE"/>  
  7.             </intent-filter>  
  8.         </service>  

        然后在activity中用如下代码绑定service:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. final Intent intent = new Intent();  
  2. intent.setAction("com.example.user.firstapp.FIRST_SERVICE");  
  3. bindService(intent,coon,Service.BIND_AUTO_CREATE);  

        这时候会报错:

        IllegalArgumentException: Service Intent must be explicit

        经过查找相关资料,发现是因为Android5.0中service的intent一定要显性声明,当这样绑定的时候不会报错。

 

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. final Intent intent = new Intent(this,BindService.class);  
  2. bindService(intent,coon,Service.BIND_AUTO_CREATE)  

http://blog.android-develop.com/2014/10/android-l-api-21-javalangillegalargumen.html上看到一个解决方法,可以将隐性调用变成显性调用。先定义一个函数:

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. /*** 
  2.      * Android L (lollipop, API 21) introduced a new problem when trying to invoke implicit intent, 
  3.      * "java.lang.IllegalArgumentException: Service Intent must be explicit" 
  4.      * 
  5.      * If you are using an implicit intent, and know only 1 target would answer this intent, 
  6.      * This method will help you turn the implicit intent into the explicit form. 
  7.      * 
  8.      * Inspired from SO answer: http://stackoverflow.com/a/26318757/1446466 
  9.      * @param context 
  10.      * @param implicitIntent - The original implicit intent 
  11.      * @return Explicit Intent created from the implicit original intent 
  12.      */  
  13.     public static Intent createExplicitFromImplicitIntent(Context context, Intent implicitIntent) {  
  14.         // Retrieve all services that can match the given intent  
  15.         PackageManager pm = context.getPackageManager();  
  16.         List<ResolveInfo> resolveInfo = pm.queryIntentServices(implicitIntent, 0);  
  17.    
  18.         // Make sure only one match was found  
  19.         if (resolveInfo == null || resolveInfo.size() != 1) {  
  20.             return null;  
  21.         }  
  22.    
  23.         // Get component info and create ComponentName  
  24.         ResolveInfo serviceInfo = resolveInfo.get(0);  
  25.         String packageName = serviceInfo.serviceInfo.packageName;  
  26.         String className = serviceInfo.serviceInfo.name;  
  27.         ComponentName component = new ComponentName(packageName, className);  
  28.    
  29.         // Create a new intent. Use the old one for extras and such reuse  
  30.         Intent explicitIntent = new Intent(implicitIntent);  
  31.    
  32.         // Set the component to be explicit  
  33.         explicitIntent.setComponent(component);  
  34.    
  35.         return explicitIntent;  
  36.     }  

然后调用

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. final Intent intent = new Intent();  
  2. intent.setAction("com.example.user.firstapp.FIRST_SERVICE");  
  3. final Intent eintent = new Intent(createExplicitFromImplicitIntent(this,intent));  
  4. bindService(eintent,conn, Service.BIND_AUTO_CREATE);  

这样也可以解决问题。

 

PS:调用本地service是这样的,不知道其他程序隐性调用service时会不会也有类似的问题,待续。

另一种简单点的解决方法可参考另一篇日志

 

继续上一篇文章,今天发现了新的解决方法,在生命intent的时候同时调用setAction和setPackage方法,这样创建出来的intent就是显性的

 

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. final Intent intent = new Intent();  
  2. intent.setAction("com.example.user.firstapp.FIRST_SERVICE");  
  3. intent.setPackage(this.getPackageName());  
  4. bindService(intent,conn,Service.BIND_AUTO_CREATE);  


        即设置了intent的action之后还要设置service所在的包名,这里是本地调用,所以用getPackageName()方法就可以获取包名。

 

实测有效。

posted on 2015-05-26 17:03  wi100sh  阅读(3964)  评论(0编辑  收藏  举报

导航