将 cordova writeFile 用于 20MB 的 blob 崩溃 (Android)

Using cordova writeFile for a 20MB blob crashes (Android)

我正在尝试将 20MB 的 blob 写入 Android 设备上的文件。此代码适用于 4-5MB 左右,但对于更大的文件会崩溃。

tp = cordova.file.dataDirectory; // for Android $cordovaFile.writeFile(tp, "temp-file.gif", blob,true);

有没有办法处理更大的斑点?谢谢。我已经在清单中有 android:largeHeap="true"

logcat 错误:

E/JavaBinder(  809): !!! FAILED BINDER TRANSACTION !!!
I/art     (  809): Background sticky concurrent mark sweep GC freed 62202(4MB) AllocSpace objects, 96(3MB) LOS objects, 20% free, 25MB/31MB, paused 20.488ms total 176.243ms
E/JavaBinder(  809): !!! FAILED BINDER TRANSACTION !!!
W/ActivityManager(  809): Exception thrown during pause
W/ActivityManager(  809): android.os.TransactionTooLargeException
W/ActivityManager(  809):   at android.os.BinderProxy.transactNative(Native Method)
W/ActivityManager(  809):   at android.os.BinderProxy.transact(Binder.java:496)
W/ActivityManager(  809):   at android.app.ApplicationThreadProxy.schedulePauseActivity(ApplicationThreadNative.java:711)
W/ActivityManager(  809):   at com.android.server.am.ActivityStack.startPausingLocked(ActivityStack.java:829)
W/ActivityManager(  809):   at com.android.server.am.ActivityStack.finishActivityLocked(ActivityStack.java:2749)
W/ActivityManager(  809):   at com.android.server.am.ActivityStack.finishTopRunningActivityLocked(ActivityStack.java:2606)
W/ActivityManager(  809):   at com.android.server.am.ActivityStackSupervisor.finishTopRunningActivityLocked(ActivityStackSupervisor.java:2544)
W/ActivityManager(  809):   at com.android.server.am.ActivityManagerService.handleAppCrashLocked(ActivityManagerService.java:11721)
W/ActivityManager(  809):   at com.android.server.am.ActivityManagerService.makeAppCrashingLocked(ActivityManagerService.java:11618)
W/ActivityManager(  809):   at com.android.server.am.ActivityManagerService.crashApplication(ActivityManagerService.java:12330)
W/ActivityManager(  809):   at com.android.server.am.ActivityManagerService.handleApplicationCrashInner(ActivityManagerService.java:11819)
W/ActivityManager(  809):   at com.android.server.am.NativeCrashListener$NativeCrashReporter.run(NativeCrashListener.java:86)

这是我用的,我没有限制或没有达到

function writeFile(__filename, __data){
        window.resolveLocalFileSystemURL(cordova.file.dataDirectory, function(dir){
            dir.getFile(__filename, {create:true}, function(file){            
                file.createWriter(function(fileWriter){
                    var blob = new Blob([__data], {type:'text/plain'});
                    fileWriter.write(blob);

                });                     
            });
        });   
    };

实验后的解决方案是将写入分成块 - 这似乎适用于我目前测试过的 20-30MB 文件。

    function writeFile2( path, file, blob, isAppend)
    {
        var csize = 4 * 1024 * 1024; // 4MB
        var d = $q.defer();
        NVRDataModel.debug ("Inside writeFile2 with blob size="+blob.size);

        // nothing more to write, so all good?
        if (!blob.size)
        {
            NVRDataModel.debug ("writeFile2 all done");
            d.resolve(true);
            return $q.resolve(true); 
        }


        if (!isAppend)
           {
               // return the delegated promise, even if it fails
               return $cordovaFile.writeFile(path, file, blob.slice(0,csize), true)
                   .then (function (succ) {
                       return writeFile2(path,file,blob.slice(csize),true);
                   });
           }
           else
           {
               // return the delegated promise, even if it fails
               return $cordovaFile.writeExistingFile(path, file, blob.slice(0,csize))
                   .then (function (succ) {
                       return writeFile2(path,file,blob.slice(csize),true);
                   });   
           }


    }

来源:@Andre Werlang 在这里帮助我: