Android没有这样的文件或目录异常

时间:2017-12-14 14:17:22

标签: java android file

我尝试按照本教程https://developer.android.com/training/camera/photobasics.html#TaskPath拍摄图像并保存文件。它打开相机,我拍照,点击复选标记,然后崩溃。我在StackOverflow上找到了几篇文章,并尝试了我能理解的所有解决方案(file.mkdirs()file.getParentFile().mkdirs())。

似乎即使它是RuntimeException,根问题是IOException(我捕获并打印消息)。我使用的是Genymotion,不确定它是否相关。我错过了什么让我无法创建文件?

我已为我的清单添加了权限:

<uses-feature android:name="android.hardware.camera"
    android:required="true" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

点击我的按钮:

public void onClick(View v) {
    Log.d("DebugMySocks", "Change photo button clicked");
    Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    try {
        // CHANGED THIS
        mImageUri = FileProvider.getUriForFile(getApplicationContext(), getApplicationContext().getPackageName() + "socks", createImageFile());
        takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, mImageUri);
    } catch (Exception e) {
        Log.d("DebugMySocks", "Error creating file " + e.getMessage());
    }
    if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
        startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
    }
}

和问题代码(我认为):

  private File createImageFile() throws IOException {
    //ADDED THIS
      int permissionCheck = ContextCompat.checkSelfPermission(this,
                Manifest.permission.WRITE_EXTERNAL_STORAGE);
      if (permissionCheck != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                    5);
      }
      // Create an image file name
      String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
      String imageFileName = "JPEG_" + timeStamp + "_";
      File storageDir = Environment.getExternalStorageDirectory();
      Log.d("DebugMySocks", "Storage directory is: " + storageDir);
      File image = new File(Environment.getExternalStorageDirectory() + File.separator + "socks"+ File.separator + imageFileName);
      Log.d("DebugMySocks", "getAbsoluteFile: " + image.getAbsoluteFile());
      image.getParentFile().mkdirs();
      image.mkdirs();
      image.createNewFile();
      //File image = File.createTempFile(
      //        imageFileName,  /* prefix */
      //        ".jpg",         /* suffix */
      //        storageDir      /* directory */
      //);

      // Save a file: path for use with ACTION_VIEW intents
      mCurrentPhotoPath = image.getAbsolutePath();
      return image;
  }

LogCat的输出:

12-14 08:05:01.411 4967-4967/com.example.kevin.moresocksplease D/DebugMySocks: Change photo button clicked
12-14 08:05:01.412 4967-4967/com.example.kevin.moresocksplease D/DebugMySocks: Storage directory is: /storage/emulated/0
12-14 08:05:01.413 4967-4967/com.example.kevin.moresocksplease D/DebugMySocks: getAbsoluteFile: /storage/emulated/0/socks/JPEG_20171214_080501_
12-14 08:05:01.416 4967-4967/com.example.kevin.moresocksplease D/DebugMySocks: Error creating file No such file or directory

例外:

FATAL EXCEPTION: main
Process: com.example.kevin.moresocksplease, PID: 4967
java.lang.RuntimeException: Failure delivering result ResultInfo{who=null, request=111, result=-1, data=Intent { act=inline-data (has extras) }} to activity {com.example.kevin.moresocksplease/com.example.kevin.moresocksplease.EditProfileActivity}: java.lang.NullPointerException: uri
 at android.app.ActivityThread.deliverResults(ActivityThread.java:4089)
 at android.app.ActivityThread.handleSendResult(ActivityThread.java:4132)
 at android.app.ActivityThread.-wrap20(ActivityThread.java)
 at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1533)
 at android.os.Handler.dispatchMessage(Handler.java:102)
 at android.os.Looper.loop(Looper.java:154)
 at android.app.ActivityThread.main(ActivityThread.java:6119)
 at java.lang.reflect.Method.invoke(Native Method)
 at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:886)
 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:776)
Caused by: java.lang.NullPointerException: uri
 at com.android.internal.util.Preconditions.checkNotNull(Preconditions.java:111)
 at android.content.ContentResolver.notifyChange(ContentResolver.java:1714)
 at android.content.ContentResolver.notifyChange(ContentResolver.java:1693)
 at com.example.kevin.moresocksplease.EditProfileActivity.onActivityResult(EditProfileActivity.java:254)
 at android.app.Activity.dispatchActivityResult(Activity.java:6932)
 at android.app.ActivityThread.deliverResults(ActivityThread.java:4085)
 at android.app.ActivityThread.handleSendResult(ActivityThread.java:4132) 
 at android.app.ActivityThread.-wrap20(ActivityThread.java) 
 at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1533) 
 at android.os.Handler.dispatchMessage(Handler.java:102) 
 at android.os.Looper.loop(Looper.java:154) 
 at android.app.ActivityThread.main(ActivityThread.java:6119) 
 at java.lang.reflect.Method.invoke(Native Method) 
 at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:886) 
 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:776) 

最后,包括因为它在异常堆栈中:

public void onActivityResult(int requestCode, int resultCode, Intent data) {
    Log.d("DebugMySocks", "onActivityResult in EditProfileActivity");
    if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == this.RESULT_OK) {
        this.getContentResolver().notifyChange(mImageUri, null); // <-- THIS IS LINE 254!
        ContentResolver cr = this.getContentResolver();
        Bitmap bitmap;
        try
        {
            bitmap = android.provider.MediaStore.Images.Media.getBitmap(cr, mImageUri);
            mProfileImage.setImageBitmap(bitmap);
            saveToFirebase(bitmap);
        }
        catch (Exception e)
        {
            Toast.makeText(this, "Failed to load", Toast.LENGTH_SHORT).show();
            Log.d("DebugMySocks", "Failed to load", e);
        }
    }
}

我从以下网站获取信息的其他链接:

File.createNewFile() thowing IOException No such file or directory

Android Camera Intent: how to get full sized photo?

编辑1:

查看此内容后,我将日志添加到mkdir请求中,并且它们都返回false。据这位先生说,这意味着目录已经存在,所以它只是创建了失败的文件?   Creating a directory in /sdcard fails

Log.d("DebugMySocks", "parentfile mkdirs: " + image.getParentFile().mkdirs());
Log.d("DebugMySocks", "image.mkdirs: " + image.mkdirs());
Log.d("DebugMySocks", "create file: " + image.createNewFile());

DebugMySocks: parentfile mkdirs: false
DebugMySocks: image.mkdirs: false
DebugMySocks: Error creating file No such file or directory

编辑2: 我现在有:

public void onClick(View v) {
                        Log.d("DebugMySocks", "Change photo button clicked");
                        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
                        try {
                            if (checkSelfPermission(android.Manifest.permission.WRITE_EXTERNAL_STORAGE)== PackageManager.PERMISSION_GRANTED) {
                                mImageUri = FileProvider.getUriForFile(getApplicationContext(), getApplicationContext().getPackageName() + "socks", createImageFile());
                            } else {
                                Log.v("DebugMySocks","Permission is revoked");
                                ActivityCompat.requestPermissions(getParent(), new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
                            }
                            takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, mImageUri);
                        } catch (Exception e) {
                            Log.d("DebugMySocks", "Error creating file " + e.getMessage());
                        }
                        if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
                            startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
                        }
                    }

    public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
        switch (requestCode) {
            case 0:
                boolean permissionsGranted = true;
                if (grantResults.length > 0 && permissions.length==grantResults.length) {
                    for (int i = 0; i < permissions.length; i++){
                        if (grantResults[i] != PackageManager.PERMISSION_GRANTED){
                            permissionsGranted=false;
                        }
                    }

                } else {
                    permissionsGranted=false;
                }
                if(permissionsGranted){
                    try {
                        mImageUri = FileProvider.getUriForFile(getApplicationContext(), getApplicationContext().getPackageName() + "socks", createImageFile());
                    } catch (Exception e) {
                        Log.d("DebugMySocks", "Permissions granted but exception happened " + e.getMessage());
                    }
                }
                break;
        }
    }

但我得到Error creating file Attempt to invoke virtual method 'android.content.res.XmlResourceParser android.content.pm.ProviderInfo.loadXmlMetaData(android.content.pm.PackageManager, java.lang.String)' on a null object reference

1 个答案:

答案 0 :(得分:2)

如果用户授予了权限,则需要检查权限。

private void checkPermission(){
    if (checkSelfPermission(android.Manifest.permission.WRITE_EXTERNAL_STORAGE)== PackageManager.PERMISSION_GRANTED) {        
        createFile();
    } else {    
        Log.v(TAG,"Permission is revoked");
        ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);  
    }
}


@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {
        case 0:
            boolean permissionsGranted = true;
            if (grantResults.length > 0 && permissions.length==grantResults.length) {
                for (int i = 0; i < permissions.length; i++){
                    if (grantResults[i] != PackageManager.PERMISSION_GRANTED){
                        permissionsGranted=false;
                    }
                }

            } else {
                permissionsGranted=false;
            }
            if(permissionsGranted){
                createFile();
            }
            break;
    }
}

private void createFile(){
   ContentResolver cr = this.getContentResolver();
   Bitmap bitmap;
   try{
       bitmap = android.provider.MediaStore.Images.Media.getBitmap(cr, mImageUri);
       mProfileImage.setImageBitmap(bitmap);
       saveToFirebase(bitmap);
   }catch (Exception e){
     Toast.makeText(this, "Failed to load", Toast.LENGTH_SHORT).show();
     Log.d("DebugMySocks", "Failed to load", e);
   }
}

public void onActivityResult(int requestCode, int resultCode, Intent data) {
    Log.d("DebugMySocks", "onActivityResult in EditProfileActivity");
    if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == this.RESULT_OK) {
        this.getContentResolver().notifyChange(mImageUri, null); // <-- THIS IS LINE 254!
        checkPermission();
    }
}