【问题标题】:open failed: EACCES (Permission denied) api23?打开失败:EACCES(权限被拒绝)api23?
【发布时间】:2017-07-10 07:16:51
【问题描述】:

我知道运行时权限,我已考虑到这一点,但它仍然不起作用。我已将这些行添加到清单中:

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

这是我的 onCreate 方法:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_new);
    if (shouldAskPermissions()) {
        askPermissions();
    }
    generateNoteOnSD(this,fileName,msg);
    readFile(fileName);
}

这是为了请求和授予权限

protected static boolean shouldAskPermissions() {
    return (Build.VERSION.SDK_INT > Build.VERSION_CODES.LOLLIPOP_MR1);
}

@TargetApi(23)
protected void askPermissions() {
    String[] permissions = {
            "android.permission.READ_EXTERNAL_STORAGE",
            "android.permission.WRITE_EXTERNAL_STORAGE"
    };
  /*  int requestCode = 200;
    requestPermissions(permissions, requestCode);*/
    // Check if we have write permission
    int permission = ActivityCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE);

    if (permission != PackageManager.PERMISSION_GRANTED) {
        // We don't have permission so prompt the user
        ActivityCompat.requestPermissions(
                this, permissions, 1);
    }
}

这些是写作和阅读方法

 public static void generateNoteOnSD(Context context, String sFileName, String sBody) {
        try {
            File root = new File(String.valueOf(Environment.getExternalStorageDirectory()));
            Log.e("file path is ", String.valueOf(Environment.getExternalStorageDirectory()));
            if (!root.exists()) {
                root.mkdirs();
            }
            File gpxfile = new File(root, sFileName);
            String separator = System.getProperty("line.separator");

            FileWriter writer = new FileWriter(gpxfile.getAbsoluteFile(), true);
            writer.append(separator); // this will add new line ;

            //FileWriter writer = new FileWriter(gpxfile);
            writer.append(sBody);
            writer.flush();
            writer.close();
            Toast.makeText(context, "Saved", Toast.LENGTH_SHORT).show();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    public void readFile(String file1){
        //Find the directory for the SD Card using the API
//*Don't* hardcode "/sdcard"
        File sdcard = Environment.getExternalStorageDirectory();

//Get the text file
        File file = new File(sdcard,file1);

//Read text from file
        StringBuilder text = new StringBuilder();

        try {
            BufferedReader br = new BufferedReader(new FileReader(file));
            String line;

            while ((line = br.readLine()) != null) {
                text.append(line);
                text.append('\n');
            }
            br.close();
        }
        catch (IOException e) {
            //You'll need to add proper error handling here
        }

//Find the view by its id
        Log.e("text",text.toString());
    }

编辑 这是我的堆栈跟踪:

W/System.err: java.io.FileNotFoundException: /storage/emulated/0/CrowdSensingData: open failed: EACCES (Permission denied)
W/System.err:     at libcore.io.IoBridge.open(IoBridge.java:487)
W/System.err:     at java.io.FileOutputStream.<init>(FileOutputStream.java:87)
W/System.err:     at java.io.FileWriter.<init>(FileWriter.java:58)
W/System.err:     at enis.example.com.alarmmanagertest.FileAndServerHandlingActivity.generateNoteOnSD(FileAndServerHandlingActivity.java:128)
W/System.err:     at enis.example.com.alarmmanagertest.ActivityRecognitionService.handleDetectedActivities(ActivityRecognitionService.java:107)
W/System.err:     at enis.example.com.alarmmanagertest.ActivityRecognitionService.onStart(ActivityRecognitionService.java:122)
W/System.err:     at android.app.Service.onStartCommand(Service.java:459)
W/System.err:     at android.app.ActivityThread.handleServiceArgs(ActivityThread.java:3209)
W/System.err:     at android.app.ActivityThread.-wrap17(ActivityThread.java)
W/System.err:     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1586)
W/System.err:     at android.os.Handler.dispatchMessage(Handler.java:111)
W/System.err:     at android.os.Looper.loop(Looper.java:207)
W/System.err:     at android.app.ActivityThread.main(ActivityThread.java:5728)
W/System.err:     at java.lang.reflect.Method.invoke(Native Method)
W/System.err:     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:789)
W/System.err:     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:679)
W/System.err: Caused by: android.system.ErrnoException: open failed: EACCES (Permission denied)
W/System.err:     at libcore.io.Posix.open(Native Method)
W/System.err:     at libcore.io.BlockGuardOs.open(BlockGuardOs.java:186)
W/System.err:     at libcore.io.IoBridge.open(IoBridge.java:473)

【问题讨论】:

  • requestPermissions() 是异步的。当它返回时,用户甚至还没有被请求权限,更不用说授予它了。
  • @CommonsWare 我明白你的意思,但我不明白如何更改代码。我是android的初学者
  • 任何关于 Android 应用程序开发的最新书籍或课程都涵盖了这些内容,the documentation 中也有少量介绍。在this sample project 中,我使用an AbstractPermissionActivity 隔离在活动启动时请求运行时权限所需的代码。

标签: android android-manifest android-permissions android-file runtime-permissions


【解决方案1】:

如果尚未授予权限,则不应立即调用需要它们的方法。而是在 onRequestPermissionsResult 中调用它们。这保证了它们只有在用户授予权限后才会被执行。

if (shouldAskPermissions()) {
    askPermissions();
} else {
    generateNoteOnSD(this,fileName,msg);
    readFile(fileName);
}

@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {
        case 1: {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                generateNoteOnSD(this,fileName,msg);
                readFile(fileName);
            } 
        }
    }
}

编辑: 此外,在您的 shouldAskPermissions() 方法中包含 ContextCompat.checkSelfPermission

编辑 2: 这是 shouldAskPermissions() 的实现

protected boolean shouldAskPermissions() {
    if(Build.VERSION.SDK_INT <= Build.VERSION_CODES.LOLLIPOP_MR1) {
        return false;
    }
    int permission = ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE);
    return  (permission != PackageManager.PERMISSION_GRANTED);
}

【讨论】:

  • 恐怕这并没有解决问题,我仍然收到 W/System.err: Caused by: android.system.ErrnoException: open failed: EACCES (Permission denied)跨度>
  • @futureengineer 请发布完整的堆栈跟踪和您的新 shouldAskPermissions() 方法
  • 只是为了在发布堆栈跟踪之前确保一些事情,这样您就不会浪费时间。我应该如何更改方法 shouldAskPermissions() ?它只是验证版本以检查它是否需要运行时权限。如果你的意思是方法 askPermissions(),我应该改变它吗?它只是请求权限,触发您在回复中发布的 onRequestPermissionsResult 方法,我误解了什么吗?
  • 即使尝试了您编辑的帖子中的代码,它仍然会给出相同的错误,我会将堆栈跟踪添加到我的原始帖子中
  • @futureengineer 我已经复制了代码,它工作得很好。确保您正确地进行了所有更改。
猜你喜欢
  • 1970-01-01
  • 2012-09-07
  • 2020-11-29
  • 1970-01-01
  • 2019-03-17
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-12-23
相关资源
最近更新 更多