Why is AssetManger.list() so slow?

后端 未结 4 693
遥遥无期
遥遥无期 2021-01-18 03:25

I\'m trying to populate a ListView with a mixture of files stored on the SDcard AND stored as assets in the APK. Using TraceView, I can see that the performanc

相关标签:
4条回答
  • 2021-01-18 04:04

    You can approach APK package as it's a ZIP file and read all the entries using Java's builtin ZipFile. It will give you all the file names with their full paths. Perhaps it shouldn't be hard to find which directories you have.

    So far this is the fastest approach I've tested.

    credit goes to @obastemur's commit on jxcore-android-basics sample project

    0 讨论(0)
  • 2021-01-18 04:10

    Coverdriven's comment "stored in a table somewhere" inspired me to solve my own problem which I've been putting off for a while.

    This doesn't answer the OP but does offer a different approach and it handles subfolders which CommonsWare's solution doesn't unless you go recursive (which of course is another possible solution). It's specifically aimed at apps which have a large number of assets in subfolders.

    I added an ANT pre-build target to run this command (I'm on Windows)

    dir assets /b /s /A-d > res\raw\assetfiles
    

    This creates a recursive (/s), barebones (/b) listing of all files, excluding directory entries (/A-d) in my assets folder.

    I then created this class to statically load the contents of assetfiles into a hashmap, the key of which is the filename and the value the full path

    public class AssetFiles {
    
    // create a hashmap of all files referenced in res/raw/assetfiles
    
    /*map of all the contents of assets located in the subfolder with the name specified in FILES_ROOT
    the key is the filename without path, the value is the full path relative to FILES_ROOT
    includes the root, e.g. harmonics_data/subfolder/file.extension - this can be passed
    directly to AssetManager.open()*/
    public static HashMap<String, String> assetFiles = new HashMap<String, String>();
    public static final String FILES_ROOT = "harmonics_data";
    
    static {
    
        String line;
        String filename;
        String path;
    
        try {
    
            BufferedReader reader = new BufferedReader(new InputStreamReader(TidesPlannerApplication.getContext().getResources().openRawResource(R.raw.assetfiles)));
    
            while ((line = reader.readLine()) != null) {
                // NB backlash (note the escape) is specific to Windows
                filename = line.substring(line.lastIndexOf("\\")+1);
                path = line.substring(line.lastIndexOf(FILES_ROOT)).replaceAll("\\\\","/");;
                assetFiles.put(filename, path);
            }
    
        } catch (IOException e) {
            e.printStackTrace();
        }
    
    }
    
    public static boolean exists(String filename){
        return assetFiles.containsKey(filename);
    }
    
    public static String getFilename(String filename){
        if (exists(filename)){
            return assetFiles.get(filename);
        } else {
            return "";
        }
    
    }
    

    }

    To use it, I simply call AssetFiles.getFilename(filename) which returns the full path which I can pass to AssetManager.open(). Much much faster!

    NB. I haven't finished this class and it's not hardened yet so you'll need to add appropriate exception catches and actions. It's also quite specific to my app in that all of my assets are in subfolders which are in turn located in a subfolder of the assets folder (see FILES_ROOT) but easy to adapt to your situation.

    Note also the need to replace backslashes, since Windows generates the assetfiles listing, with forward slashes. You could eliminate this on OSX and *nix platforms.

    0 讨论(0)
  • 2021-01-18 04:13

    Can anyone explain why the performance would be so poor?

    Reading the contents of a ZIP archive (the APK where the assets are located) is slower than reading the contents of a directory on the filesystem, apparently. In the abstract, this is not especially surprising, as I suspect that this would be true for all major operating systems.

    Read in that list() data once, then save it somewhere else for faster access (e.g., database), particularly in a form that is optimized for future lookups (e.g., where a simple database query could give you what you want, vs. having to load and "recursively search it" again).

    0 讨论(0)
  • 2021-01-18 04:20

    If you have a deep tree of directories in the assets you can detect firstly if an item is file or directory and then call .list() on it (really accelerates the walking through the tree). This is my solution I've discovered for this:

    try {
        AssetFileDescriptor desc = getAssets().openFd(path);  // Always throws exception: for directories and for files
        desc.close();  // Never executes
    } catch (Exception e) {
        exception_message = e.toString();
    }
    
    if (exception_message.endsWith(path)) {  // Exception for directory and for file has different message
        // Directory
    } else {
        // File
    }
    
    0 讨论(0)
提交回复
热议问题