Android 馬賽克(Mosaics)效果
前幾天看見開源項(xiàng)目效果好贊,看了下代碼,實(shí)現(xiàn)大致就是在原界面之上覆蓋一成自定義的View,獲取到點(diǎn)擊的那個(gè)View的內(nèi)容(Bitmap),然后在覆蓋的那個(gè)自定義View的特定位置畫出來(lái),之后就是對(duì)這個(gè)Bitmap做一些列拆分,變化重繪的過(guò)程。在這里根據(jù)他對(duì)bitmap的拆分,感覺(jué)用來(lái)實(shí)現(xiàn)Bitmap的效果也是不錯(cuò)的,就試著做一做。
在這里介紹使用兩種方式實(shí)現(xiàn)馬賽克效果.開始之前先看看效果
感覺(jué)還不錯(cuò)吧!
1、直接繪制
public static Bitmap getMosaicsBitmap(Bitmap bmp, double precent) { long start = System.currentTimeMillis(); int bmpW = bmp.getWidth(); int bmpH = bmp.getHeight(); Bitmap resultBmp = Bitmap.createBitmap(bmpW, bmpH, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(resultBmp); Paint paint = new Paint(); double unit; if (precent == 0) { unit = bmpW; } else { unit = 1 / precent; } double resultBmpW = bmpW / unit; double resultBmpH = bmpH / unit; for (int i = 0; i < resultBmpH; i++) { for (int j = 0; j < resultBmpW; j++) { int pickPointX = (int) (unit * (j + 0.5)); int pickPointY = (int) (unit * (i + 0.5)); int color; if (pickPointX >= bmpW || pickPointY >= bmpH) { color = bmp.getPixel(bmpW / 2, bmpH / 2); } else { color = bmp.getPixel(pickPointX, pickPointY); } paint.setColor(color); canvas.drawRect((int) (unit * j), (int) (unit * i), (int) (unit * (j + 1)), (int) (unit * (i + 1)), paint); } } canvas.setBitmap(null); long end = System.currentTimeMillis(); Log.v(TAG, "DrawTime:" + (end - start)); return resultBmp; }
2、修改像素點(diǎn)
public static Bitmap getMosaicsBitmaps(Bitmap bmp, double precent) { long start = System.currentTimeMillis(); int bmpW = bmp.getWidth(); int bmpH = bmp.getHeight(); int[] pixels = new int[bmpH * bmpW]; bmp.getPixels(pixels, 0, bmpW, 0, 0, bmpW, bmpH); int raw = (int) (bmpW * precent); int unit; if (raw == 0) { unit = bmpW; } else { unit = bmpW / raw; //原來(lái)的unit*unit像素點(diǎn)合成一個(gè),使用原左上角的值 } if (unit >= bmpW || unit >= bmpH) { return getMosaicsBitmap(bmp, precent); } for (int i = 0; i < bmpH; ) { for (int j = 0; j < bmpW; ) { int leftTopPoint = i * bmpW + j; for (int k = 0; k < unit; k++) { for (int m = 0; m < unit; m++) { int point = (i + k) * bmpW + (j + m); if (point < pixels.length) { pixels[point] = pixels[leftTopPoint]; } } } j += unit; } i += unit; } long end = System.currentTimeMillis(); Log.v(TAG, "DrawTime:" + (end - start)); return Bitmap.createBitmap(pixels, bmpW, bmpH, Bitmap.Config.ARGB_8888); }
從效率上來(lái)看,第二中方式效率會(huì)高10倍,只要是因?yàn)榈谝环N方式繪制的次數(shù)太多了,而繪制是比較費(fèi)時(shí)間的。這里特別提示,不要在大量的循環(huán)語(yǔ)句內(nèi)部使用Log.v(...),這是一個(gè)很耗時(shí)間的操作。
是不是很有趣,大家可以親自動(dòng)手試驗(yàn)一下。
相關(guān)文章
基于Android應(yīng)用中如何反饋Crash報(bào)告的詳解
本篇文章是對(duì)在Android應(yīng)用中如何反饋Crash報(bào)告的詳細(xì)分析介紹。需要的朋友參考下2013-05-05Android SurfaceView基礎(chǔ)用法詳解
這篇文章主要介紹了Android SurfaceView基礎(chǔ)用法詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08Android ListView物流獲取追蹤功能實(shí)現(xiàn)
這篇文章主要介紹了Android ListView物流獲取追蹤功能實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2016-03-03詳解android 視頻圖片混合輪播實(shí)現(xiàn)
這篇文章主要介紹了android 視頻圖片混合輪播實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05Android ListView長(zhǎng)按彈出菜單二種實(shí)現(xiàn)方式示例
這篇文章主要介紹了Android ListView長(zhǎng)按彈出菜單的方法,大家參考實(shí)現(xiàn)2013-11-11Android開發(fā)實(shí)現(xiàn)的圖片點(diǎn)擊切換功能示例
這篇文章主要介紹了Android開發(fā)實(shí)現(xiàn)的圖片點(diǎn)擊切換功能,涉及Android ImageView組件創(chuàng)建、布局及實(shí)現(xiàn)圖形切換相關(guān)操作技巧,需要的朋友可以參考下2019-04-04