Android之SD卡插拔广播

本文介绍了在Android中实现SD卡插拔广播监听的具体方法,包括在Activity中及整个应用程序中注册广播接收器的方式,并提供了代码示例。

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

  监听SD卡的插拔这样的需求你肯定会遇到,今天就总结一下关于系统广播的监听。
   比如SD卡的插拔监听流程:

1. 如果是单个Activity监听广播,则在Activity的onCreate方法里面,用下面的代码注册广播

    IntentFilter iFilter = new IntentFilter();  
    iFilter.addAction(Intent.ACTION_MEDIA_EJECT);  
    iFilter.addDataScheme("file");  
    registerReceiver(mBroadcastReceiver , iFilter);  
2. 如果是整个程序监听广播,则在Android manifest用下面的方法注册广播

<receiver android:name=".activities.widget.UsbBroadCastReceiver">  
    <intent-filter android:priority="1000">  
        <action android:name="android.intent.action.MEDIA_MOUNTED"/>  
        <action android:name="android.intent.action.MEDIA_EJECT" />  
        <data android:scheme="file"/>  
    </intent-filter>  
</receiver> 

3. 如果广播别别的程序截获,导致你无法收到广播,给intent-filter加上一个android:priority="1000"的属性就行,添加位置,参照上面的代码。

4. 我的项目里面是整个程序监听广播,

①Manifest的代码如下:

    <receiver android:name=".activities.widget.UsbBroadCastReceiver">  
        <intent-filter android:priority="1000">  
            <action android:name="android.intent.action.MEDIA_MOUNTED"/>  
            <action android:name="android.intent.action.MEDIA_EJECT" />  
            <data android:scheme="file"/>  
        </intent-filter>  
    </receiver>  

Android Manifest一定要在intent-filter注意加上<data android:scheme=”file”>,否则无法监听到SD卡插拔广播,如果你在Activity里面用代码注册广播,那一定要加iFilter.addDataScheme("file")
②广播的代码如下:
public class UsbBroadCastReceiver extends BroadcastReceiver {  
    @Override  
    public void onReceive(Context context, Intent intent) {  
        String action = intent.getAction();  
          
        if(action.equals(Intent.ACTION_MEDIA_EJECT)){  
            ToastUtil.ToastShort(context, R.string.usb_sdeject);  
        }else if(action.equals(Intent.ACTION_MEDIA_MOUNTED)){  
            ToastUtil.ToastShort(context, R.string.usb_sdconnect);  
        }  
    }  
}  
  类似的广播监听方式都是一样的,比如开机广播的监听,插拔耳机的监听,屏幕亮灭的监听,电量低的监听...
  Android系统源码中有哪些系统广播? 我在之前的博客中有总结,感兴趣的可以看一下Android源码中常用的系统广播








<think>在Android13中监听SIM插拔事件,可以使用广播接收器(BroadcastReceiver)来监听SIM状态变化。根据引用[2]和引用[3],关键点在于注册广播接收器并处理`ACTION_SIM_STATE_CHANGED`意图(Intent)。需要注意,从Android8.0(API级别26)开始,对隐式广播的使用进行了限制,但系统广播(如SIM状态变化)通常仍可接收。###步骤实现:1.**创建广播接收器(BroadcastReceiver)**:重写`onReceive`方法,在SIM状态变化时接收通知。2.**注册广播接收器**:在Activity或Fragment的生命周期中注册和注销广播接收器,避免内存泄漏。3.**处理SIM状态**:从意图中获取SIM状态信息。###代码示例:####1.创建广播接收器```javapublicclassSimStateReceiverextendsBroadcastReceiver{publicstaticfinalStringACTION_SIM_STATE_CHANGED="android.intent.action.SIM_STATE_CHANGED";@OverridepublicvoidonReceive(Contextcontext,Intentintent){if(intent==null||intent.getAction()==null)return;if(ACTION_SIM_STATE_CHANGED.equals(intent.getAction())){intsimState=SubscriptionManager.getDefaultSubscriptionId();//获取默认SIM状态//或者使用intent中的额外信息,如引用[3]提到的SUBSCRIPION_KEYintsubId=intent.getIntExtra("subscription",SubscriptionManager.INVALID_SUBSCRIPTION_ID);//判断SIM状态TelephonyManagertelephonyManager=(TelephonyManager)context.getSystemService(Context.TELEPHONY_SERVICE);intstate=telephonyManager.getSimState();switch(state){caseTelephonyManager.SIM_STATE_ABSENT://SIM被拔出break;caseTelephonyManager.SIM_STATE_PRESENT://SIM插入break;default://其他状态(如未知、准备中等)}}}}```####2.在Activity/Fragment中注册和注销```javapublicclassMainActivityextendsAppCompatActivity{privateSimStateReceiversimStateReceiver;@OverrideprotectedvoidonResume(){super.onResume();registerSimReceiver(true);}@OverrideprotectedvoidonPause(){super.onPause();registerSimReceiver(false);}privatevoidregisterSimReceiver(booleanregister){if(register){simStateReceiver=newSimStateReceiver();IntentFilterfilter=newIntentFilter(SimStateReceiver.ACTION_SIM_STATE_CHANGED);registerReceiver(simStateReceiver,filter);}else{if(simStateReceiver!=null){unregisterReceiver(simStateReceiver);simStateReceiver=null;}}}}```###注意事项:-**权限**:确保在`AndroidManifest.xml`中声明`READ_PHONE_STATE`权限(如果应用目标API级别为Android10或更低,还需声明`READ_PRIVILEGED_PHONE_STATE`,但后者仅系统应用可用)。对于Android11及以上,需使用运行时权限请求。```xml<uses-permissionandroid:name="android.permission.READ_PHONE_STATE"/>```-**动态注册**:广播接收器必须动态注册(在代码中注册),因为`ACTION_SIM_STATE_CHANGED`是隐式广播,且受Android8.0+的限制(静态注册无效)。-**兼容性**:不同Android版本中,SIM状态的可能取值略有差异,需参考[TelephonyManager文档](https://developer.android.com/reference/android/telephony/TelephonyManager)。-**多SIM**:若设备支持多SIM,需通过`subId`区分不同槽事件(如引用[3])。###相关问题:1.在Android13中,如何检查是否拥有`READ_PHONE_STATE`权限?2.如何适配Android10(API29)以上的分区存储和权限变化?3.多SIM设备中,如何精确识别哪张SIM插拔?</think> ### Android 13 监听SIM插拔事件实现方法 在Android 13中监听SIM插拔事件,主要通过**广播接收器**实现。以下是完整实现步骤: #### 1. 创建广播接收器 ```java public class SimStateReceiver extends BroadcastReceiver { public static final String ACTION_SIM_STATE_CHANGED = TelephonyManager.ACTION_SUBSCRIPTION_CARRIER_IDENTITY_CHANGED; @Override public void onReceive(Context context, Intent intent) { if (ACTION_SIM_STATE_CHANGED.equals(intent.getAction())) { int slotIndex = intent.getIntExtra(SubscriptionManager.EXTRA_SLOT_INDEX, -1); int subId = intent.getIntExtra(SubscriptionManager.EXTRA_SUBSCRIPTION_INDEX, SubscriptionManager.INVALID_SUBSCRIPTION_ID); TelephonyManager tm = context.getSystemService(TelephonyManager.class); int simState = tm.getSimState(slotIndex); switch (simState) { case TelephonyManager.SIM_STATE_ABSENT: // SIM拔出处理 handleSimRemoved(slotIndex); break; case TelephonyManager.SIM_STATE_READY: // SIM插入处理 handleSimInserted(slotIndex, subId); break; } } } } ``` #### 2. 在Activity/Fragment中动态注册广播 ```java public class MainActivity extends AppCompatActivity { private SimStateReceiver simReceiver; @Override protected void onStart() { super.onStart(); registerSimReceiver(); } @Override protected void onStop() { super.onStop(); unregisterSimReceiver(); } private void registerSimReceiver() { simReceiver = new SimStateReceiver(); IntentFilter filter = new IntentFilter(); filter.addAction(SimStateReceiver.ACTION_SIM_STATE_CHANGED); registerReceiver(simReceiver, filter); } private void unregisterSimReceiver() { if (simReceiver != null) { unregisterReceiver(simReceiver); simReceiver = null; } } } ``` #### 3. 添加必要权限 在 `AndroidManifest.xml` 中添加: ```xml <uses-permission android:name="android.permission.READ_PHONE_STATE"/> <uses-permission android:name="android.permission.READ_PRIVILEGED_PHONE_STATE"/> ``` > **注意**:Android 13对SIM访问权限控制更严格: > 1. `READ_PRIVILEGED_PHONE_STATE` 仅系统应用可用 > 2. 普通应用需要动态申请 `READ_PHONE_STATE` 权限: > ```java > ActivityCompat.requestPermissions( > this, > new String[]{Manifest.permission.READ_PHONE_STATE}, > REQUEST_CODE > ); > ``` #### 4. 关键改进(针对Android 13) ```java // 使用新的CarrierConfigManager检测SIM变化 CarrierConfigManager ccm = getSystemService(CarrierConfigManager.class); ccm.registerCarrierConfigChangeListener(Runnable::run, (slotIndex, subId) -> { // 处理SIM配置变更 }); // 使用SubscriptionManager监听订阅变化 SubscriptionManager sm = getSystemService(SubscriptionManager.class); sm.addOnSubscriptionsChangedListener(new SubscriptionManager.OnSubscriptionsChangedListener() { @Override public void onSubscriptionsChanged() { // 当SIM状态变化时触发 } }); ``` ### 补充说明: 1. **双设备处理**:通过`slotIndex`区分不同槽 2. **SIM状态常量**: - `SIM_STATE_UNKNOWN` - `SIM_STATE_ABSENT`(拔出) - `SIM_STATE_READY`(就绪) - `SIM_STATE_NETWORK_LOCKED` 3. **兼容性**:需同时处理 `ACTION_SIM_CARD_STATE_CHANGED` 广播 > 实测中插入SIM会收到 `SIM_STATE_READY` (3) 事件,拔出会收到 `SIM_STATE_ABSENT` (1) 事件 --- ### 相关问题: 1. Android 13 如何动态申请`READ_PHONE_STATE`权限? 2. 如何在多SIM设备中区分不同的槽事件? 3. Android 13 对SIM权限管理有哪些新限制? 4. 如何检测SIM切换(如禁用/启用)状态? 5. 如何在无界面组件(如Service)中监听SIM变化?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值