成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

wemall app商城源碼Android之ListView異步加載網(wǎng)絡(luò)圖片(優(yōu)化緩存機(jī)制)

luqiuwen / 2155人閱讀

摘要:本文分享商城源碼之異步加載網(wǎng)絡(luò)圖片優(yōu)化緩存機(jī)制代碼信息,供技術(shù)員參考學(xué)習(xí)。采用線程池內(nèi)存緩存文件緩存內(nèi)存緩存中網(wǎng)上很多是采用來(lái)防止堆溢出,這兒嚴(yán)格限制只能使用最大內(nèi)存的對(duì)下載的圖片進(jìn)行按比例縮放,以減少內(nèi)存的消耗具體的代碼里面說(shuō)明。

    wemall-mobile是基于WeMall的android app商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享wemall app商城源碼Android之ListView異步加載網(wǎng)絡(luò)圖片(優(yōu)化緩存機(jī)制)代碼信息,供技術(shù)員參考學(xué)習(xí)。

1、采用線程池
2、內(nèi)存緩存+文件緩存
3、內(nèi)存緩存中網(wǎng)上很多是采用SoftReference來(lái)防止堆溢出,這兒嚴(yán)格限制只能使用最大JVM內(nèi)存的1/4
4、對(duì)下載的圖片進(jìn)行按比例縮放,以減少內(nèi)存的消耗
具體的代碼里面說(shuō)明。先放上內(nèi)存緩存類的代碼MemoryCache.java:

public class MemoryCache {

    private static final String TAG = "MemoryCache";
    // 放入緩存時(shí)是個(gè)同步操作
    // LinkedHashMap構(gòu)造方法的最后一個(gè)參數(shù)true代表這個(gè)map里的元素將按照最近使用次數(shù)由少到多排列,即LRU
    // 這樣的好處是如果要將緩存中的元素替換,則先遍歷出最近最少使用的元素來(lái)替換以提高效率
    private Map cache = Collections
            .synchronizedMap(new LinkedHashMap(10, 1.5f, true));
    // 緩存中圖片所占用的字節(jié),初始0,將通過(guò)此變量嚴(yán)格控制緩存所占用的堆內(nèi)存
    private long size = 0;// current allocated size
    // 緩存只能占用的最大堆內(nèi)存
    private long limit = 1000000;// max memory in bytes

    public MemoryCache() {
        // use 25% of available heap size
        setLimit(Runtime.getRuntime().maxMemory() / 4);
    }

    public void setLimit(long new_limit) { 
        limit = new_limit;
        Log.i(TAG, "MemoryCache will use up to " + limit / 1024. / 1024. + "MB");
    }

    public Bitmap get(String id) {
        try {
            if (!cache.containsKey(id))
                return null;
            return cache.get(id);
        } catch (NullPointerException ex) {
            return null;
        }
    }

    public void put(String id, Bitmap bitmap) {
        try {
            if (cache.containsKey(id))
                size -= getSizeInBytes(cache.get(id));
            cache.put(id, bitmap);
            size += getSizeInBytes(bitmap);
            checkSize();
        } catch (Throwable th) {
            th.printStackTrace();
        }
    }

    /**
     * 嚴(yán)格控制堆內(nèi)存,如果超過(guò)將首先替換最近最少使用的那個(gè)圖片緩存
     * 
     */
    private void checkSize() {
        Log.i(TAG, "cache size=" + size + " length=" + cache.size());
        if (size > limit) {
            // 先遍歷最近最少使用的元素
            Iterator> iter = cache.entrySet().iterator();
            while (iter.hasNext()) {
                Entry entry = iter.next();
                size -= getSizeInBytes(entry.getValue());
                iter.remove();
                if (size <= limit)
                    break;
            }
            Log.i(TAG, "Clean cache. New size " + cache.size());
        }
    }

    public void clear() {
        cache.clear();
    }

    /**
     * 圖片占用的內(nèi)存
     * 
     * @param bitmap
     * @return
     */
    long getSizeInBytes(Bitmap bitmap) {
        if (bitmap == null)
            return 0;
        return bitmap.getRowBytes() * bitmap.getHeight();
    }
}

也可以使用SoftReference,代碼會(huì)簡(jiǎn)單很多,但是我推薦上面的方法。

public class MemoryCache {
    
    private Map> cache = Collections
            .synchronizedMap(new HashMap>());

    public Bitmap get(String id) {
        if (!cache.containsKey(id))
            return null;
        SoftReference ref = cache.get(id);
        return ref.get();
    }

    public void put(String id, Bitmap bitmap) {
        cache.put(id, new SoftReference(bitmap));
    }

    public void clear() {
        cache.clear();
    }

}

下面是文件緩存類的代碼FileCache.java:

public class FileCache {

    private File cacheDir;

    public FileCache(Context context) {
        // 如果有SD卡則在SD卡中建一個(gè)LazyList的目錄存放緩存的圖片
        // 沒(méi)有SD卡就放在系統(tǒng)的緩存目錄中
        if (android.os.Environment.getExternalStorageState().equals(
                android.os.Environment.MEDIA_MOUNTED))
            cacheDir = new File(
                    android.os.Environment.getExternalStorageDirectory(),
                    "LazyList");
        else
            cacheDir = context.getCacheDir();
        if (!cacheDir.exists())
            cacheDir.mkdirs();
    }

    public File getFile(String url) {
        // 將url的hashCode作為緩存的文件名
        String filename = String.valueOf(url.hashCode());
        // Another possible solution
        // String filename = URLEncoder.encode(url);
        File f = new File(cacheDir, filename);
        return f;

    }

    public void clear() {
        File[] files = cacheDir.listFiles();
        if (files == null)
            return;
        for (File f : files)
            f.delete();
    }

}

最后最重要的加載圖片的類,ImageLoader.java:

public class ImageLoader {

    MemoryCache memoryCache = new MemoryCache();
    FileCache fileCache;
    private Map imageViews = Collections
            .synchronizedMap(new WeakHashMap());
    // 線程池
    ExecutorService executorService;

    public ImageLoader(Context context) {
        fileCache = new FileCache(context);
        executorService = Executors.newFixedThreadPool(5);
    }

    // 當(dāng)進(jìn)入listview時(shí)默認(rèn)的圖片,可換成你自己的默認(rèn)圖片
    final int stub_id = R.drawable.stub;

    // 最主要的方法
    public void DisplayImage(String url, ImageView imageView) {
        imageViews.put(imageView, url);
        // 先從內(nèi)存緩存中查找

        Bitmap bitmap = memoryCache.get(url);
        if (bitmap != null)
            imageView.setImageBitmap(bitmap);
        else {
            // 若沒(méi)有的話則開(kāi)啟新線程加載圖片
            queuePhoto(url, imageView);
            imageView.setImageResource(stub_id);
        }
    }

    private void queuePhoto(String url, ImageView imageView) {
        PhotoToLoad p = new PhotoToLoad(url, imageView);
        executorService.submit(new PhotosLoader(p));
    }

    private Bitmap getBitmap(String url) {
        File f = fileCache.getFile(url);

        // 先從文件緩存中查找是否有
        Bitmap b = decodeFile(f);
        if (b != null)
            return b;

        // 最后從指定的url中下載圖片
        try {
            Bitmap bitmap = null;
            URL imageUrl = new URL(url);
            HttpURLConnection conn = (HttpURLConnection) imageUrl
                    .openConnection();
            conn.setConnectTimeout(30000);
            conn.setReadTimeout(30000);
            conn.setInstanceFollowRedirects(true);
            InputStream is = conn.getInputStream();
            OutputStream os = new FileOutputStream(f);
            CopyStream(is, os);
            os.close();
            bitmap = decodeFile(f);
            return bitmap;
        } catch (Exception ex) {
            ex.printStackTrace();
            return null;
        }
    }

    // decode這個(gè)圖片并且按比例縮放以減少內(nèi)存消耗,虛擬機(jī)對(duì)每張圖片的緩存大小也是有限制的
    private Bitmap decodeFile(File f) {
        try {
            // decode image size
            BitmapFactory.Options o = new BitmapFactory.Options();
            o.inJustDecodeBounds = true;
            BitmapFactory.decodeStream(new FileInputStream(f), null, o);

            // Find the correct scale value. It should be the power of 2.
            final int REQUIRED_SIZE = 70;
            int width_tmp = o.outWidth, height_tmp = o.outHeight;
            int scale = 1;
            while (true) {
                if (width_tmp / 2 < REQUIRED_SIZE
                        || height_tmp / 2 < REQUIRED_SIZE)
                    break;
                width_tmp /= 2;
                height_tmp /= 2;
                scale *= 2;
            }

            // decode with inSampleSize
            BitmapFactory.Options o2 = new BitmapFactory.Options();
            o2.inSampleSize = scale;
            return BitmapFactory.decodeStream(new FileInputStream(f), null, o2);
        } catch (FileNotFoundException e) {
        }
        return null;
    }

    // Task for the queue
    private class PhotoToLoad {
        public String url;
        public ImageView imageView;

        public PhotoToLoad(String u, ImageView i) {
            url = u;
            imageView = i;
        }
    }

    class PhotosLoader implements Runnable {
        PhotoToLoad photoToLoad;

        PhotosLoader(PhotoToLoad photoToLoad) {
            this.photoToLoad = photoToLoad;
        }

        @Override
        public void run() {
            if (imageViewReused(photoToLoad))
                return;
            Bitmap bmp = getBitmap(photoToLoad.url);
            memoryCache.put(photoToLoad.url, bmp);
            if (imageViewReused(photoToLoad))
                return;
            BitmapDisplayer bd = new BitmapDisplayer(bmp, photoToLoad);
            // 更新的操作放在UI線程中
            Activity a = (Activity) photoToLoad.imageView.getContext();
            a.runOnUiThread(bd);
        }
    }

    /**
     * 防止圖片錯(cuò)位
     * 
     * @param photoToLoad
     * @return
     */
    boolean imageViewReused(PhotoToLoad photoToLoad) {
        String tag = imageViews.get(photoToLoad.imageView);
        if (tag == null || !tag.equals(photoToLoad.url))
            return true;
        return false;
    }

    // 用于在UI線程中更新界面
    class BitmapDisplayer implements Runnable {
        Bitmap bitmap;
        PhotoToLoad photoToLoad;

        public BitmapDisplayer(Bitmap b, PhotoToLoad p) {
            bitmap = b;
            photoToLoad = p;
        }

        public void run() {
            if (imageViewReused(photoToLoad))
                return;
            if (bitmap != null)
                photoToLoad.imageView.setImageBitmap(bitmap);
            else
                photoToLoad.imageView.setImageResource(stub_id);
        }
    }

    public void clearCache() {
        memoryCache.clear();
        fileCache.clear();
    }

    public static void CopyStream(InputStream is, OutputStream os) {
        final int buffer_size = 1024;
        try {
            byte[] bytes = new byte[buffer_size];
            for (;;) {
                int count = is.read(bytes, 0, buffer_size);
                if (count == -1)
                    break;
                os.write(bytes, 0, count);
            }
        } catch (Exception ex) {
        }
    }
}

主要流程是先從內(nèi)存緩存中查找,若沒(méi)有再開(kāi)線程,從文件緩存中查找都沒(méi)有則從指定的url中查找,并對(duì)bitmap進(jìn)行處理,最后通過(guò)下面方法對(duì)UI進(jìn)行更新操作。

a.runOnUiThread(...);

在你的程序中的基本用法:

ImageLoader imageLoader=new ImageLoader(context);
...
imageLoader.DisplayImage(url, imageView);
 比如你的放在你的ListView的adapter的getView()方法中,當(dāng)然也適用于GridView。

wemall官網(wǎng)地址:http://www.wemallshop.com
原文詳情地址:http://Git.oschina.NET/zzunet...
wemall doraemonAndroid app商城詳情地址:http://www.koahub.com/home/pr...
wemall 開(kāi)源微商城 ,微信商城,商城源碼,三級(jí)分銷,微生鮮,微水果,微外賣(mài),微訂餐---專業(yè)的o2o系統(tǒng)

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/66267.html

相關(guān)文章

  • wemall app商城源碼Android 獲取XML網(wǎng)絡(luò)數(shù)據(jù)并綁定到ListView

    摘要:是基于的商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享獲取網(wǎng)絡(luò)數(shù)據(jù)并綁定到,對(duì)數(shù)據(jù)進(jìn)行獲取并顯示代碼信息,供技術(shù)員參考學(xué)習(xí)。 wemall-mobile是基于WeMall的android app商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享Android 獲取XML網(wǎng)絡(luò)數(shù)據(jù)并綁定到ListView,對(duì)數(shù)...

    Benedict Evans 評(píng)論0 收藏0
  • wemall app商城源碼Android支付寶通知處理類

    摘要:是基于的商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享商城源碼之處理支付寶各接口通知返回,供技術(shù)員參考學(xué)習(xí)。 wemall-mobile是基于WeMall的Android app商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享wemall app商城源碼Android之處理支付寶各接口通知返回,供技術(shù)員...

    Crazy_Coder 評(píng)論0 收藏0
  • wemall app商城源碼Android支付寶接口公用函數(shù)

    摘要:本文分享商城源碼之支付寶接口公用函數(shù),該類是請(qǐng)求通知返回兩個(gè)文件所調(diào)用的公用函數(shù)核心處理文件,供技術(shù)員參考學(xué)習(xí)。 wemall-mobile是基于WeMall的Android app商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享wemall app商城源碼Android之 支付寶接口公用函數(shù),該類是請(qǐng)求、通知返回兩個(gè)文件所調(diào)用的公用函數(shù)核心處理文...

    Lemon_95 評(píng)論0 收藏0
  • wemall app商城源碼AndroidNative(原生)支付模式一demo

    摘要:本文分享原生支付模式一,供技術(shù)員參考學(xué)習(xí)。官網(wǎng)地址模式一商戶按固定格式生成鏈接二維碼,用戶掃碼后調(diào)微信會(huì)將和用戶發(fā)送到商戶設(shè)置的鏈接上,商戶收到請(qǐng)求生成訂單,調(diào)用統(tǒng)一支付接口下單提交到微信,微信會(huì)返回給商戶。 wemall-mobile是基于WeMall的Android app商城,只需要在原商城目錄下上傳接口文件即可完成服務(wù)端的配置,客戶端可定制修改。本文分享Native(原生)支付...

    source 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<