0

Android File API が正常に動作しません。

これが私が達成したいことです。完全にダウンロードされた後にローカルに保存する必要があるAmazonからビデオをダウンロードしています。私はこれを非常に簡単な方法で行っています (コードは短縮されており、重要な行のみを示しています)。

inputStream = connection.getInputStream();
fileOutputStream = context.openFileOutput("my_video", Context.MODE_PRIVATE);
// Read bytes (and store them) until there is nothing more to read(-1)
do {
    int numread = inputStream.read(buffer);
    if (numread <= 0){
        break;                  
    }
    fileOutputStream.write(buffer, 0, numread);
} while (true);

inputStreamfileOutputStreamインスタンス変数です。

ビデオが完全にダウンロードされた場合、これは実際にはかなりうまく機能します。その場合、すべて問題なく、後でビデオにローカルでアクセスできます。

ただし、アプリでビデオのダウンロードが中断され、キャンセルする必要がある場合があります。この場合、そこにあるが明らかに完全ではないファイルを削除したいと思います。

ファイルを削除するコードは次のとおりです。

FileProvider fileProvider = new FileProvider();
File newFile = new File(context.getFilesDir(), "my_video");     
Uri contentUri = FileProvider.getUriForFile(context, FILE_PROVIDER, newFile);
fileProvider.delete(fileToDelete, null, null);

最後の行fileProvider.delete(fileToDelete, null, null);で がスローNullPointerExceptionされています。デバッグしたところ、fileProviderが初期化されていることがわかりました。そのため、 を呼び出すために使用している URI に問題があると強く思いますがdelete、何が問題なのかわかりません。ファイルプロバイダーを使用して適切な削除を行う方法を知っている人はいますか?

更新:これがやり過ぎではないことを願っています。現在、VideoDownloaderクラス全体を投稿しています:

public class VideoDownloader {

  private final int TIMEOUT_CONNECTION = 5000;//5sec
  private final int TIMEOUT_SOCKET = 30000;//30sec

  private static final String FILE_PROVIDER = "com.orangewise.fileprovider";

  private Context context;
  private String videoURL;
  private String targetFileName;

  private HttpURLConnection connection;
  private InputStream inputStream;
  private FileOutputStream fileOutputStream;

  private boolean downloadFinished;

  public VideoDownloader(Context context, String videoURL, String targetFileName) {
    this.context = context;
    this.videoURL = videoURL;
    this.targetFileName = targetFileName;
    this.downloadFinished = false;
  }

  public boolean isDownloadFinished(){
    return this.downloadFinished;
  }

  public void startDownload(){
    downloadVideoFile(this.context, this.videoURL, this.targetFileName);    
  }

  private void downloadVideoFile(Context context, String videoURL, String targetFileName) {
    URL url = null;
    try {
        url = new URL(videoURL);

        // Open a connection to that URL.
        connection = (HttpURLConnection) url.openConnection();

        connection.setReadTimeout(TIMEOUT_CONNECTION);
        connection.setConnectTimeout(TIMEOUT_SOCKET);r

        inputStream = connection.getInputStream();
        fileOutputStream = context.openFileOutput(targetFileName, Context.MODE_PRIVATE);

        byte[] buffer = new byte[3 * 1024];
        int counter = 0;

        // Read bytes (and store them) until there is nothing more to read(-1)
        do {
            int numread = inputStream.read(buffer);
            if (numread <= 0){
                break;                  
            }
            fileOutputStream.write(buffer, 0, numread);
        } while (true);
        downloadFinished = true;

        // Clean up
        closeStreams();
    } catch (Exception e) {
        Log.d(Constants.ERROR, "ERROR [" + getClass().getName() + "]: Caught exception (" + e + ") when trying to download video: " + e.getMessage());
    }
  }

  public Uri getUriForFile(){
    return getUriForFile(context, targetFileName);
  }

  private Uri getUriForFile(Context context, String fileName){

    File newFile = new File(context.getFilesDir(), fileName);       
    Uri contentUri = FileProvider.getUriForFile(context, FILE_PROVIDER, newFile);

    return contentUri;
  }

  public void cancel(){
    // 1. cancel the connection
    Log.d(Constants.LOG, "DEBUG [" + getClass().getName() + "]: Cancel connection");

    try {
        connection.disconnect();
        closeStreams();

        if(!isDownloadFinished()){              
            // Remove the file if it has not been fully downloaded
            FileProvider fileProvider = new FileProvider();
            Uri fileToDeleteUri = getUriForFile();
            fileProvider.delete(fileToDeleteUri, null, null); // returns 1 if the delete succeeds; otherwise, 0.
        }
        else{
            Log.d(Constants.LOG, "DEBUG [" + getClass().getName() + "]: Leave the file, it has been completely download");
        }
    } 
    catch (Exception e) {
        Log.d(Constants.ERROR, "Exception ( " + e + " ) caught: " +  e.getMessage() + "; ");
    }
}

private void closeStreams() throws IOException{     
    // Close the streams
    try {           
        fileOutputStream.flush();
        fileOutputStream.close();
        inputStream.close();
    } catch (NullPointerException e) {
        Log.d(Constants.ERROR, "Null pointer exception caught: " +  e.getMessage());
    }
    Log.d(Constants.LOG, "DEBUG [" + getClass().getName() + "]: Clean up performed");
  }
} 

別の更新:これが私のスタック トレースです。

07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152): java.lang.NullPointerException
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.support.v4.content.FileProvider.delete(FileProvider.java:497)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at com.orangewise.just4kidstv.util.VideoDownloader.cancel(VideoDownloader.java:134)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at com.orangewise.just4kidstv.util.VideoDownloadTask.cancel(VideoDownloadTask.java:20)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at com.orangewise.just4kidstv.activities.VideoPlayerActivity.onStop(VideoPlayerActivity.java:64)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.Instrumentation.callActivityOnStop(Instrumentation.java:1170)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.Activity.performStop(Activity.java:3873)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.ActivityThread.performDestroyActivity(ActivityThread.java:2623)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:2694)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.ActivityThread.access$2100(ActivityThread.java:117)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:968)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.os.Handler.dispatchMessage(Handler.java:99)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.os.Looper.loop(Looper.java:130)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at android.app.ActivityThread.main(ActivityThread.java:3687)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at java.lang.reflect.Method.invokeNative(Native Method)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at java.lang.reflect.Method.invoke(Method.java:507)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:867)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:625)
07-23 17:31:29.114: E/com.organgewise.just4kidstv.LOG(6152):    at dalvik.system.NativeStart.main(Native Method)
4

2 に答える 2

2

SDK エクストラに同梱されている v4 サポート ライブラリの FileProvider.java のソースを調べると、次のことがわかります。

@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
   // ContentProvider has already checked granted permissions
   final File file = mStrategy.getFileForUri(uri);   /* line 497 */
   return file.delete() ? 1 : 0;
}

したがって、mStrategy は null です。

さらに検索すると、 が 1 か所だけで初期化されていることがわかります。

/**
 * After the FileProvider is instantiated, this method is called to provide the system with
 * information about the provider.
 *
 * @param context A {@link Context} for the current component.
 * @param info A {@link ProviderInfo} for the new provider.
 */
@Override
public void attachInfo(Context context, ProviderInfo info) {
    super.attachInfo(context, info);

    // Sanity check our security
    if (info.exported) {
        throw new SecurityException("Provider must not be exported");
    }
    if (!info.grantUriPermissions) {
        throw new SecurityException("Provider must grant uri permissions");
    }

    mStrategy = getPathStrategy(context, info.authority);
}

したがって、FileProviderがこのメソッドの呼び出しで適切に設定されていないことは明らかです。

FileProvider のドキュメントは完全に明確ではありませんが、単に "new FileProvider()" を実行するのではなく、マニフェストで関連するセットアップを実行する必要があるように見えます。

于 2014-07-23T16:00:30.633 に答える
2

@クリス・ストラットンは正しいです。そのコンストラクターでインスタンス化しないでください。FileProvider は ContentProvider のサブクラスです。AndroidManifest.xml で宣言したので、次を使用してハンドルを取得できます。

context.getContentResolver()

したがって、これを行うだけで NullPointerException を修正できます。

context.getContentResolver().delete(contentUri, null, null);

1 が返されることを確認して、機能したことを確認できます。

于 2015-10-30T21:41:54.603 に答える