【问题标题】:Android runtime permission is GRANTED but still denied已授予 Android 运行时权限,但仍被拒绝
【发布时间】:2017-11-30 07:29:20
【问题描述】:

我正在尝试编写一个发送短信的应用程序。检查我是否具有所需的权限时,它返回 true,但仍会因 SecurityException 而崩溃。

按下按钮时

private void startAutoMsg() {
    Log.d("Starting Auto Msg");
    //FIXME: Start proper loop
    if (canSendSms()) {
        sendMessage();
        mAutoMsgButton.setBackgroundColor(Color.GREEN);
    }
}

我正在使用以下函数来确定我是否拥有适当的权限

private boolean canSendSms() {
    if ((ContextCompat.checkSelfPermission(mContext, Manifest.permission.SEND_SMS)
            == PackageManager.PERMISSION_GRANTED)) {
        Log.d("Permission granted");
        return true;
    } else {
        Log.d("Permission denied");
        ActivityCompat.requestPermissions(
                mActivity, new String[]{Manifest.permission.SEND_SMS}, 101);
        return false;
    }
}

上面的代码返回true,因此SMS正在尝试用这个发送

private void sendMessage() {
    Log.d("sending message");
    PendingIntent sentPI = PendingIntent.getBroadcast(
            mContext, 0, new Intent(Constants.ACTION_SMS_SENT), 0);
    PendingIntent deliveredPI = PendingIntent.getBroadcast(
            mContext, 0, new Intent(Constants.ACTION_SMS_DELIVERED), 0);
    Thread thread = new Thread(new Runnable() {
        @Override
        public void run() {
            mSmsManager.sendTextMessage("mNumber", null, mText, sentPI, deliveredPI);
        }
    });
    thread.run();
}

在我的清单中:

<uses-permission android:name="android.permission.SEND_SMS"/>
<uses-permission android:name="android.permission.READ_SMS" />
<uses-permission android:name="android.permission.WRITE_SMS" />
<uses-permission android:name="android.permission.RECEIVE_SMS" />
<uses-permission android:name="android.permission.RECEIVE_MMS" />

崩溃日志

D/SmsApp: TelephonyFragment: canSendSms(533): Permission granted
D/SmsApp: TelephonyFragment: sendMessage(222): sending message
D/AndroidRuntime: Shutting down VM
E/AndroidRuntime: FATAL EXCEPTION: main
                  Process: com.android.phone, PID: 16736
                  java.lang.SecurityException: Sending SMS message: uid 1001 does not have android.permission.SEND_SMS.
                      at android.app.ContextImpl.enforce(ContextImpl.java:1727)
                      at android.app.ContextImpl.enforceCallingPermission(ContextImpl.java:1749)
                      at android.content.ContextWrapper.enforceCallingPermission(ContextWrapper.java:750)
                      at android.content.ContextWrapper.enforceCallingPermission(ContextWrapper.java:750)
                      at com.android.internal.telephony.IccSmsInterfaceManager.sendText(IccSmsInterfaceManager.java:410)
                      at com.android.internal.telephony.UiccSmsController.sendTextForSubscriber(UiccSmsController.java:136)
                      at android.telephony.SmsManager.sendTextMessageInternal(SmsManager.java:366)
                      at android.telephony.SmsManager.sendTextMessage(SmsManager.java:349)
                      at com.rawinc.smsapp.ui.telephony.TelephonyFragment$2.run(TelephonyFragment.java:230)
                      at java.lang.Thread.run(Thread.java:764)
                      at com.rawinc.smsapp.ui.telephony.TelephonyFragment.sendMessage(TelephonyFragment.java:233)
                      at com.rawinc.smsapp.ui.telephony.TelephonyFragment.startAutoMsg(TelephonyFragment.java:517)
                      at com.rawinc.smsapp.ui.telephony.TelephonyFragment.toggleAutoMsg(TelephonyFragment.java:507)
                      at com.rawinc.smsapp.ui.telephony.TelephonyFragment.lambda$-com_rawinc_smsapp_ui_telephony_TelephonyFragment_11555(TelephonyFragment.java:359)
                      at com.rawinc.smsapp.ui.telephony.-$Lambda$uKVldJdEkN_fZa3QWm3EZHDa2r8$2.$m$0(Unknown Source:4)
                      at com.rawinc.smsapp.ui.telephony.-$Lambda$uKVldJdEkN_fZa3QWm3EZHDa2r8$2.onClick(Unknown Source:0)
                      at android.view.View.performClick(View.java:6178)
                      at android.view.View$PerformClick.run(View.java:24416)
                      at android.os.Handler.handleCallback(Handler.java:769)
                      at android.os.Handler.dispatchMessage(Handler.java:98)
                      at android.os.Looper.loop(Looper.java:255)
                      at android.app.ActivityThread.main(ActivityThread.java:6555)
                      at java.lang.reflect.Method.invoke(Native Method)
                      at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
                      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)

【问题讨论】:

  • 哪个安卓版本?
  • 你在清单中也提到过吗?只需交叉检查一次
  • 目标版本是 API 26,但我目前正在测试 25。权限在清单中。
  • 您是否偶然使用了三星?我在三星设备上遇到了类似的问题,但需要不同的许可。结果 PermissionChecker 和 ContextCompat selfCheckPermission 返回不同的结果(拒绝与授予)
  • 您找到解决方案了吗?

标签: android sms runtime-permissions


【解决方案1】:

请这样做:

在您的应用中创建此类:

import android.content.pm.PackageManager;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;


public abstract class RuntimePermissionsActivity extends AppCompatActivity
{


    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)
    {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        int permissionCheck = PackageManager.PERMISSION_GRANTED;
        for (int permission : grantResults)
        {
            permissionCheck = permissionCheck + permission;
        }
        if ((grantResults.length > 0) && permissionCheck == PackageManager.PERMISSION_GRANTED)
        {
            onPermissionsGranted(requestCode);
        }
        else
        {
            onPermissionsDeny(requestCode);
        }
    }

    public void requestAppPermissions(final String[] requestedPermissions, final int requestCode)
    {
        int permissionCheck = PackageManager.PERMISSION_GRANTED;
        boolean shouldShowRequestPermissionRationale = false;
        for (String permission : requestedPermissions)
        {
            permissionCheck = permissionCheck + ContextCompat.checkSelfPermission(this, permission);
            shouldShowRequestPermissionRationale = shouldShowRequestPermissionRationale || ActivityCompat.shouldShowRequestPermissionRationale(this, permission);
        }
        if (permissionCheck != PackageManager.PERMISSION_GRANTED)
        {
            ActivityCompat.requestPermissions(this, requestedPermissions, requestCode);
        }
        else
        {
            onPermissionsGranted(requestCode);
        }
    }

    public abstract void onPermissionsGranted(int requestCode);
    public abstract void onPermissionsDeny(int requestCode);

}

在您的 MainActivity(或任何 Activity)中,您应该像这样从此类扩展:

public class Main_Activity extends RuntimePermissionsActivity

并在您的活动中覆盖此方法:

@Override
public void onPermissionsGranted(int requestCode) {

}

@Override
public void onPermissionsDeny(int requestCode) {

}

最后:使用此代码检查权限:

MainActivity.super.requestAppPermissions(new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE}, requestcodeWriteStorage);

你也可以检查许多权限,你应该在字符串数组中定义;

requestcodeWriteStorage 是一个 int 你应该定义它;

使用我阅读的 2 种方法管理您的回调。

例如:

@Override
public void onPermissionsGranted(int requestCode) {
    if (requestCode == requestcodeWriteStorage){
        Toast.makeText(this,"Done",Toast.LENGTH_LONG).show();

    }

【讨论】:

    【解决方案2】:

    覆盖onRequestPermissionsResult 方法

     @Override
    public void onRequestPermissionsResult(int requestCode,
                                           String permissions[], int[] grantResults) {
    
    switch (requestCode) {
            case MY_PERMISSIONS_REQUEST: {
                // If request is cancelled, the result arrays are empty.
                if (grantResults.length > 0
                        && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
    
                    // permission was granted, yay! Do the
    
                    if (ContextCompat.checkSelfPermission(this,
                            Manifest.permission.Your_permission)
                            == PackageManager.PERMISSION_GRANTED) {
    
                        //do whatever u want to do
                    }
    
                } else {
    
                    // permission denied, boo! Disable the
                    // functionality that depends on this permission.
                    Toast.makeText(this, "permission denied", Toast.LENGTH_LONG).show();
                }
                return;
            }
    
            // other 'case' lines to check for other
            // permissions this app might request
        }
    
     }
    

    【讨论】:

    • 问题是没有处理权限。问题是 API 告诉我我有它,而我没有。
    • @amoz871 问题未经许可,因为您已授予对 api > 23 正常工作的设备的权限。您能否粘贴整个代码的 sn-p,以便我们可以确切知道问题出在哪里发生?
    • 我能想到的每个权限检查都会返回 PERMISSION_GRANTED,但每当我尝试使用权限时,我都会收到 SecurityException。当 checkSelfPermission 返回 PERMISSION_GRANTED 而不是 PERMISSION_DENIED 时会出现问题。
    • 将此行添加到清单文件中,您在其中定义了活动android:exported="true"
    • 对此进行了测试,但在 Activity 和 Application 上都没有区别
    【解决方案3】:

    需要位置时检查权限

    if ((ContextCompat.checkSelfPermission(LoginActivity.this, Manifest.permission.SEND_SMS) != PackageManager.PERMISSION_GRANTED)) {
                requestPermissions(new String[]{Manifest.permission.SEND_SMS},
                        101);
            }
    

    这行增加了你的活动

    @Override
        public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
            switch (requestCode) {
                case PERMISSIONS_CODE:
                    if (grantResults.length <= 0 || grantResults[0] != PackageManager.PERMISSION_GRANTED) {
                        Toast.makeText(this, "Permission denied", Toast.LENGTH_SHORT).show();
                    }
                    break;
                default:
                    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
                    break;
            }
        }
    

    【讨论】:

    • 问题是没有处理权限。问题是 API 告诉我我有它,而我没有。
    • 你的权限没有得到去检查设置--->选择你的应用--->权限
    • 鉴于上述条件和 onRequestPermissionsResult 在检查后添加你的活动
    • 你检查你的应用权限吗
    • 6.0 后未从清单中获得更多权限,因此以编程方式添加。
    【解决方案4】:

    你需要处理onRequestPermissionResult

    @Override
        public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
            super.onRequestPermissionsResult(requestCode, permissions, grantResults);
            if(grantResults[0]== PackageManager.PERMISSION_GRANTED){
                Log.v("======>","Permission: "+permissions[0]+ "was "+grantResults[0]);
                //your task
            }
        }
    

    并检查版本

    if (Build.VERSION.SDK_INT >= 23) {
                if (checkSelfPermission(android.Manifest.permission.SEND_SMS)
                        == PackageManager.PERMISSION_GRANTED) {
                    Log.v("======>","Permission is granted");
                    return true;
                } else {
    
                    Log.v("======>","Permission is revoked");
                    ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.SEND_SMS}, 1);
                    return false;
                }
            }
            else { //permission is automatically granted on sdk<23 upon installation
                Log.v("======>","Permission is granted");
                return true;
            }
    

    并发送消息使用

    private void sendMessage() {
            try {
                SmsManager smsManager = SmsManager.getDefault();
                smsManager.sendTextMessage(number, null, "TXt", null, null);
                Toast.makeText(getApplicationContext(), "Message Sent",
                        Toast.LENGTH_LONG).show();
            } catch (Exception ex) {
                Toast.makeText(getApplicationContext(),ex.getMessage().toString(),
                        Toast.LENGTH_LONG).show();
                ex.printStackTrace();
            }
        }
    

    【讨论】:

    • 感谢您的回复。问题是 checkSelfPermission(android.Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED 行将返回 true,即使我没有权限。
    • 用 SEND_SMS 代替 READ_EXTERNA‌​L_STORAGE
    • 我明白你的意思并使用了 SEND_SMS :) 正如在我原来的帖子中看到的那样,我已经在 canSendSms() 中进行了此检查,但检查给了我一个误报,这就是我的问题。
    • 使用我的代码检查权限并查看,它只会请求一次权限
    • 它打印 Log.v("======>","Permission is grant");在日志中然后崩溃
    【解决方案5】:

    在 5.0 及更高版本上,您还必须授予您的应用在设置中读取短信的权限。

    【讨论】:

    • 设置 -> 应用程序 -> SmsApp -> 权限:未请求权限,Screenshot
    • 我不知道不同手机上的权限是什么样的,但您可以使用此步骤来实现。在设置中查找权限,您会看到不同权限的列表。单击其中任何一个时,您将看到当前请求该权限的应用程序列表。只需查找读取短信权限,您应该会看到您的应用列出并单击它,您会看到允许开关,只需将其打开。
    • 我想我找到了你的意思。但是,我的应用在 SMS 许可下甚至不可见,screenshots
    • 尝试重启设备
    • 我试过重启它。我已经尝试卸载/重新安装该应用程序。我已经刷新了我的设备并在其他几台设备上进行了尝试。
    【解决方案6】:

    确保您的应用程序使用了 android Android 6.0 和更高版本的 SDK,我遇到了同样的问题,我发现我使用的是 android 4.1 kitkat sdk,一旦我将 min sdk 级别更改为 21,问题就解决了

    【讨论】:

    • 感谢您的评论。我的 Android 设备运行 Android 8.0。 minSdkVersion 版本设置为 24,targetSdkVersion 设置为 27。
    猜你喜欢
    • 1970-01-01
    • 2014-09-27
    • 2021-10-18
    • 2011-10-17
    • 2021-12-21
    • 1970-01-01
    • 2011-08-28
    • 2019-03-10
    • 2017-05-14
    相关资源
    最近更新 更多