应用程序在请求 READ_PHONE_STATE 权限时崩溃并出现 IllegalStateException

Posted

技术标签:

【中文标题】应用程序在请求 READ_PHONE_STATE 权限时崩溃并出现 IllegalStateException【英文标题】:App crashes with IllegalStateException while requesting READ_PHONE_STATE permission 【发布时间】:2018-06-06 08:49:23 【问题描述】:

我需要捕获设备的 IMEI 和 IMSI 以及您已经知道的情况,这需要权限 "android.permission.READ_PHONE_STATE"。碰巧当这个对话框窗口向用户请求这个权限时,我的应用程序被停止了。怎么解决?

@SuppressLint("MissingPermission")
public void ListarApps(View view) throws PackageManager.NameNotFoundException 

    String myDeviceModel = android.os.Build.MODEL;
    String myDeviceProduct = android.os.Build.MANUFACTURER;
    String myVersion = Build.VERSION.RELEASE;
    System.out.println("Manufacturer: " + myDeviceProduct +
            " - Model: " + myDeviceModel + " - Android: " + myVersion);

    // Code For IMEI AND IMSI NUMBER ("android.permission.READ_PHONE_STATE")

    final int MY_PERMISSIONS_REQUEST_READ_PHONE_STATE = 999;

    ActivityCompat.requestPermissions(this,
            new String[]Manifest.permission.READ_PHONE_STATE, MY_PERMISSIONS_REQUEST_READ_PHONE_STATE);
        
        
    String serviceName = Context.TELEPHONY_SERVICE;
    TelephonyManager m_telephonyManager = (TelephonyManager) getSystemService(serviceName);
        
    String IMEI, IMSI;
        
    /*IMEI = m_telephonyManager.getDeviceId();
    IMSI = m_telephonyManager.getSubscriberId();  

    System.out.println("IMEI: " + IMEI);
    System.out.println("IMSI: " + IMSI);*/

更新(logcat 输出):

12-24 15:31:43.787 27853-27853/com.testando.teste E/AndroidRuntime: 致命异常: main 进程:com.testando.teste,PID:27853 java.lang.IllegalStateException:无法执行 android:onClick 的方法 在 android.support.v7.app.AppCompatViewInflater$DeclaredOnClickListener.onClick(AppCompatViewInflater.java:293) 在 android.view.View.performClick(View.java:5640) 在 android.view.View$PerformClick.run(View.java:22455) 在 android.os.Handler.handleCallback(Handler.java:751) 在 android.os.Handler.dispatchMessage(Handler.java:95) 在 android.os.Looper.loop(Looper.java:154) 在 android.app.ActivityThread.main(ActivityThread.java:6165) 在 java.lang.reflect.Method.invoke(本机方法) 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:888) 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:778) 引起:java.lang.reflect.InvocationTargetException 在 java.lang.reflect.Method.invoke(本机方法) 在 android.support.v7.app.AppCompatViewInflater$DeclaredOnClickListener.onClick(AppCompatViewInflater.java:288) 在 android.view.View.performClick(View.java:5640) 在 android.view.View$PerformClick.run(View.java:22455) 在 android.os.Handler.handleCallback(Handler.java:751) 在 android.os.Handler.dispatchMessage(Handler.java:95) 在 android.os.Looper.loop(Looper.java:154) 在 android.app.ActivityThread.main(ActivityThread.java:6165) 在 java.lang.reflect.Method.invoke(本机方法) 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:888) 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:778) 引起:java.lang.SecurityException:getDeviceId:用户 10141 和当前进程都没有 android.permission.READ_PHONE_STATE。 在 android.os.Parcel.readException(Parcel.java:1684) 在 android.os.Parcel.readException(Parcel.java:1637) 在 com.android.internal.telephony.ITelephony$Stub$Proxy.getDeviceId(ITelephony.java:4684) 在 android.telephony.TelephonyManager.getDeviceId(TelephonyManager.java:866) 在 com.testando.teste.MainActivity.ListarApps(MainActivity.java:49) 在 java.lang.reflect.Method.invoke(本机方法) 在 android.support.v7.app.AppCompatViewInflater$DeclaredOnClickListener.onClick(AppCompatViewInflater.java:288) 在 android.view.View.performClick(View.java:5640) 在 android.view.View$PerformClick.run(View.java:22455) 在 android.os.Handler.handleCallback(Handler.java:751) 在 android.os.Handler.dispatchMessage(Handler.java:95) 在 android.os.Looper.loop(Looper.java:154) 在 android.app.ActivityThread.main(ActivityThread.java:6165) 在 java.lang.reflect.Method.invoke(本机方法) 在 com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:888) 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:778)

更新 2(AndroidManifest.xml 文件):

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.testando.teste">

    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
    
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

【问题讨论】:

发布您的 manifest.xml 文件,顺便说一句,您是否在清单中添加了权限? @JeffersonFarias 你打电话给m_telephonyManager.getDeviceId(); ? @Munir 他没有在清单文件中添加权限。 @InziKhan Lolz..:) @JeffersonFarias 你没有在清单中添加权限吗?? 【参考方案1】:

好的,因为我了解您的代码,您首先需要检查是否授予权限,然后为 getDeviceId 编写代码

if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_PHONE_STATE)
                != PackageManager.PERMISSION_GRANTED) 
    // ask user permission 
 else 
    // READ_PHONE_STATE permission is already been granted.
    TelephonyManager mngr = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
    IMEI = mngr.getDeviceId();

您还需要覆盖onRequestPermissionsResult

@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) 
    if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) 
        // READ_PHONE_STATE permission is already been granted.
        TelephonyManager mngr = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
        IMEI = mngr.getDeviceId();
    

【讨论】:

【参考方案2】:
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_PHONE_STATE)
                != PackageManager.PERMISSION_GRANTED) 
            // ask user permission 
         else 
   // READ_PHONE_STATE permission is already been granted.
            TelephonyManager mngr = (TelephonyManager)getSystemService(Context.TELEPHONY_SERVICE);
            IMEI = mngr.getDeviceId();                    
        

【讨论】:

以上是关于应用程序在请求 READ_PHONE_STATE 权限时崩溃并出现 IllegalStateException的主要内容,如果未能解决你的问题,请参考以下文章

READ_PHONE_STATE 的权限被拒绝 [重复]

如何解决此错误:android.permission.READ_PHONE_STATE?

Play 控制台上的测试版发布错误

应用程序请求权限时崩溃

为啥 SmsManager 在某些设备上需要 READ_PHONE_STATE 权限,而在其他设备上不需要?

使用 Build.VERSION.SDK_INT 会自动添加 READ_PHONE_STATE 权限吗?