android电话广播有时候失效,【案例分析】android广播接收不到原因分析

本文探讨了在Android开发中遇到的自定义广播接收不到的问题,重点在于广播访问权限。当发送广播时设置特定权限导致广播接收失败,原因是未在AndroidManifest.xml中声明自定义权限。解决方案是在Manifest文件中添加权限申明,并强调自定义权限名需包含包名。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

【概述】

本人正在学习android开发,最近在练习android自定义广播的时候,发现广播怎么也接收不到。甚是奇怪,看到网上也有很多同学遇到同样的问题。今天特别总结一下,以供大家参考。当然如有纰漏错误,望各位指正。

本次讨论的是广播访问的权限问题,对于显示或者隐式的问题,不在这里赘述。

【版本】

API版本为v21

【问题概述】

我先介绍一下我学习的例子。非常简单,通过一个button触发一个自定广播。

代码如下。

1 public class MyBroadcastReceiver extendsBroadcastReceiver {2

3 @Override4 public voidonReceive(Context context, Intent intent) {5 Log.i("TEST", "received in MyBroadcastReceiver");6

7 Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_LONG).show();8 }9

10 }

界面部分就是一个button,这里就略了。

button逻辑如下:

因为安全问题,所以设置了接收权限。sendBroadcast的第二个参数为, com.example.broadcasttest.RECV_MYBC.(这个也是本次问题的原因)

PS:如果不设置访问权限,会有Exported receiver does not require permission告警。原因是第三方应用可以通过发送该广播触发该逻辑,从而产生安全风险。具体细节可以百度一下。

com.example.broadcasttest.MY_BROADCAST为自定义的广播。

public class MainActivity extendsActivity {privateButton buttonSend;

@Overrideprotected voidonCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

buttonSend=(Button)findViewById(R.id.button_send);

buttonSend.setOnClickListener(newOnClickListener() {//

//触发自定的广播,广播定义为com.example.broadcasttest.MY_BROADCAST//设置接收权限为RECV_MYBC//@Overridepublic voidonClick(View v) { Intent intent= new Intent("com.example.broadcasttest.MY_BROADCAST"); sendBroadcast(intent,"com.example.broadcasttest.RECV_MYBC"); }

});

}

}

AndroidManifast.xml中,也对这个receiver添加权限。

现象是广播接受失败。

【问题根因】

这里的权限没有申明,凡是自定义的权限是需要在AndroidManifast.xml中定义的。

【解决方案】

在AndroidManifest.xml中添加自定义权限的权限申明和权限使用

注意几点:

1. 不设置访问权限是可以成功的。权限不是必须的。

2. 自定义权限的名字要带上包名,不带包名可能是有问题的。

练习代码下载:

http://download.csdn.net/detail/mimiduck/9430272

<think>我们正在解决Android中自定义广播和Location接收失效的问题。根据用户的问题描述和提供的引用,我们可以分析可能的原因和解决方案。首先,关于自定义广播接收失效的可能原因:1.**广播注册方式**:在Android8.0(API26)及以上版本,对隐式广播的使用进行了限制。如果自定义广播是隐式的(即没有指定接收者的包名或类名),则可能无法被接收。2.**广播注册位置**:广播接收器需要在AndroidManifest.xml中静态注册,或者在代码中动态注册。如果注册方式不正确,可能导致接收不到。3.**广播发送方式**:发送广播时,如果使用了不正确的Intent,或者没有设置ComponentName(对于显式广播),也可能导致接收不到。4.**权限问题**:发送或接收广播时可能需要权限,如果没有声明或请求权限,可能导致失败。其次,关于Location接收失效的可能原因:1.**权限问题**:从Android6.0(API23)开始,获取位置信息需要动态请求运行时权限(ACCESS_FINE_LOCATION或ACCESS_COARSE_LOCATION)。2.**位置服务未开启**:用户可能关闭了设备的位置服务。3.**位置提供器的选择**:在请求位置更新时,可能指定的位置提供器(如GPS或网络)不可用。4.**省电模式**:设备的省电模式可能会限制后台位置更新。5.**后台限制**:在Android8.0及以上版本,对后台应用获取位置信息进行了限制。参考提供的引用:引用[1]提到在Android9.0中,一些网络相关的函数和广播不再接收用户位置或个人可识别数据。因此,在Android9.0及以上版本,即使应用拥有位置权限,某些广播也可能无法携带位置信息。引用[2]展示了一个短信广播接收器的例子,其中使用了abortBroadcast()来阻断广播。但注意,在Android4.4及以上版本,有序广播的优先级设置和abortBroadcast()的使用可能受到限制,特别是对于系统广播(如短信广播)可能不再允许应用拦截。引用[3]提到了调试标记开启的安全风险,但这与广播和位置接收失效关系不大。引用[4]比较了SurfaceView和TextureView,与当前问题无关。因此,我们重点分析广播和位置接收失效的原因,并给出解决方案。解决方案:一、自定义广播接收失效的解决方案:1.**使用显式广播**:针对自定义广播,建议使用显式广播,即在发送广播时指定接收者的ComponentName(包名和类名)。这样不受隐式广播限制的影响。2.**静态注册注意事项**:如果需要在应用未启动时接收广播,可以静态注册。但注意,Android8.0及以上版本对静态注册的隐式广播有限制,所以对于自定义广播,建议使用显式意图。同时,在AndroidManifest.xml中声明接收器时,需要设置exported属性(如果只接收应用内广播,可设为false)。3.**动态注册广播**:在Activity或Service中动态注册广播接收器,并确保在合适的生命周期方法中注销。4.**权限设置**:如果广播涉及到权限,需要在发送和接收时声明相同的权限。例如,在发送广播时使用`sendBroadcast(intent,permission)`,接收方在注册时也需要声明该权限。二、位置接收失效的解决方案:1.**检查权限**:确保在AndroidManifest.xml中声明了所需的位置权限(ACCESS_FINE_LOCATION或ACCESS_COARSE_LOCATION),并且在运行时(Android6.0及以上)动态请求权限。2.**检查位置服务是否开启**:在请求位置更新前,检查设备的位置服务是否开启。如果没有开启,可以引导用户打开。3.**选择合适的定位提供器**:使用LocationManager的`getProviders()`方法获取可用的提供器,或者使用FusedLocationProviderClient(GooglePlay服务的一部分)来获取位置,它会自动选择最佳的提供器。4.**处理后台位置限制**:对于Android8.0及以上版本,后台应用接收位置更新的频率会受到限制。如果需要持续获取位置,可以考虑使用前台服务,并显示一个持续的通知。5.**省电模式处理**:检查设备是否处于省电模式,并考虑在省电模式下调整位置更新策略(如降低更新频率)。代码示例:自定义广播发送和接收(显式广播):发送广播(在Activity或Service中):```javaIntentintent=newIntent();intent.setAction("com.example.MY_CUSTOM_ACTION");//显式设置接收者intent.setComponent(newComponentName("com.example.receiverapp","com.example.receiverapp.MyBroadcastReceiver"));sendBroadcast(intent);```接收器(静态注册在AndroidManifest.xml中):```xml<receiverandroid:name=".MyBroadcastReceiver"android:enabled="true"android:exported="true"><!--如果只接收本应用广播,exported可设为false,但显式广播通常不受影响--><intent-filter><actionandroid:name="com.example.MY_CUSTOM_ACTION"/></intent-filter></receiver>```动态注册广播接收器(在Activity中):```javaprivateBroadcastReceivermReceiver=newBroadcastReceiver(){@OverridepublicvoidonReceive(Contextcontext,Intentintent){//处理广播}};@OverrideprotectedvoidonResume(){super.onResume();IntentFilterfilter=newIntentFilter("com.example.MY_CUSTOM_ACTION");registerReceiver(mReceiver,filter);}@OverrideprotectedvoidonPause(){super.onPause();unregisterReceiver(mReceiver);}```位置请求示例(使用FusedLocationProviderClient):1.添加依赖(build.gradle):implementation'com.google.android.gms:play-services-location:18.0.0'2.请求位置更新:```javaFusedLocationProviderClientfusedLocationClient=LocationServices.getFusedLocationProviderClient(this);//检查权限if(ContextCompat.checkSelfPermission(this,Manifest.permission.ACCESS_FINE_LOCATION)==PackageManager.PERMISSION_GRANTED){//创建位置请求LocationRequestlocationRequest=LocationRequest.create();locationRequest.setInterval(10000);//更新间隔locationRequest.setFastestInterval(5000);//最快更新间隔locationRequest.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);//请求位置更新fusedLocationClient.requestLocationUpdates(locationRequest,locationCallback,null);}else{//请求权限ActivityCompat.requestPermissions(this,newString[]{Manifest.permission.ACCESS_FINE_LOCATION},REQUEST_LOCATION_PERMISSION);}//定义位置回调privateLocationCallbacklocationCallback=newLocationCallback(){@OverridepublicvoidonLocationResult(LocationResultlocationResult){if(locationResult==null){return;}for(Locationlocation:locationResult.getLocations()){//处理位置更新}}};//在合适的时机移除更新请求(如onDestroy)@OverrideprotectedvoidonDestroy(){super.onDestroy();fusedLocationClient.removeLocationUpdates(locationCallback);}```注意事项:-在AndroidManifest.xml中声明位置权限:```xml<uses-permissionandroid:name="android.permission.ACCESS_FINE_LOCATION"/><!--或者使用ACCESS_COARSE_LOCATION-->```总结:1.自定义广播失效:优先使用显式广播,注意注册方式和权限。2.位置接收失效:检查权限、位置服务状态,并适配不同Android版本的后台限制。如果问题仍然存在,请检查系统版本和日志,以获取更详细的错误信息。生成相关问题:§§相关问题§§1.Android8.0及以上版本对广播有哪些限制?2.如何动态请求位置权限?3.在Android9.0及以上版本,如何获取后台位置更新?4.使用FusedLocationProviderClient有哪些优势?</think>### Android 自定义广播与位置接收失效解决方案 #### 一、自定义广播失效原因及修复 1. **Android 8.0+ 隐式广播限制** ```java // 错误:隐式广播Android 8.0+ 将被系统屏蔽) Intent intent = new Intent("com.example.MY_CUSTOM_ACTION"); sendBroadcast(intent); // 正确:显式广播(指定目标组件) Intent intent = new Intent("com.example.MY_CUSTOM_ACTION"); intent.setComponent(new ComponentName("com.example.pkg", "com.example.Receiver")); sendBroadcast(intent); ``` - **原因**:Android 8.0+ 禁止大部分隐式广播[^1] - **修复**: - 使用显式广播(指定目标包名/类名) - 对于应用内广播:使用 `LocalBroadcastManager` 2. **广播接收器注册问题** ```xml <!-- AndroidManifest.xml 静态注册 --> <receiver android:name=".MyReceiver" android:exported="true"> <intent-filter> <action android:name="com.example.MY_CUSTOM_ACTION" /> </intent-filter> </receiver> ``` - **动态注册未注销**:导致内存泄漏和接收失效 ```java // Activity 中动态注册/注销 @Override protected void onResume() { super.onResume(); registerReceiver(receiver, new IntentFilter("MY_ACTION")); } @Override protected void onPause() { super.onPause(); unregisterReceiver(receiver); // 必须注销 } ``` 3. **权限问题** ```xml <!-- 发送带权限的广播 --> <uses-permission android:name="com.example.PERMISSION" /> <!-- 接收器声明权限 --> <receiver android:name=".MyReceiver" android:permission="com.example.PERMISSION"> ``` - 发送/接收端需声明相同权限[^2] #### 二、位置接收失效原因及修复 1. **Android 9.0+ 位置权限变更** ```java // 必须同时请求前台和后台权限 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { requestPermissions(new String[]{ Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.ACCESS_BACKGROUND_LOCATION // 后台权限 }, REQUEST_CODE); } ``` - **原因**:Android 9.0+ 限制后台位置访问[^1] - **修复**: - 动态请求 `ACCESS_BACKGROUND_LOCATION` - 在 `AndroidManifest.xml` 声明: ```xml <uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" /> ``` 2. **位置服务未启用** ```java // 检查位置服务状态 LocationManager lm = (LocationManager) getSystemService(LOCATION_SERVICE); if (!lm.isProviderEnabled(LocationManager.GPS_PROVIDER)) { // 引导用户开启位置服务 startActivity(new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS)); } ``` 3. **省电模式限制** ```java // 检查省电模式 PowerManager pm = (PowerManager) getSystemService(POWER_SERVICE); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { if (pm.isDeviceIdleMode()) { // 处理省电模式下的位置获取策略 } } ``` #### 三、通用解决方案 1. **生命周期管理** - 在 `onStart()`/`onResume()` 注册接收器 - 在 `onStop()`/`onPause()` 注销接收器 - 使用 `LifecycleObserver` 自动管理 2. **后台限制处理** ```java // 使用前台服务获取后台位置 public class LocationService extends Service { @Override public int onStartCommand(Intent intent, int flags, int startId) { startForeground(NOTIFICATION_ID, createNotification()); // 启动位置监听 return START_STICKY; } } ``` 3. **广播发送优化** ```java // 使用带权限的广播发送 sendBroadcast(intent, "com.example.PERMISSION"); // 使用有序广播传递数据 sendOrderedBroadcast(intent, null, new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { // 处理最终结果 } }, null, RESULT_OK, null, null); ``` #### 四、适配建议 | 系统版本 | 广播策略 | 位置策略 | |----------|----------|----------| | Android 7.0- | 隐式广播可用 | 只需前台权限 | | Android 8.0-9 | 显式广播 | 需后台权限声明 | | Android 10+ | 组件显式注册 | 后台访问需用户手动授权 | > **关键点**:从 Android 10 开始,后台位置访问需在设置中单独开启,应用需引导用户操作: > ```java > Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS); > intent.setData(Uri.parse("package:" + getPackageName())); > startActivity(intent); > ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值