Android自定義View實(shí)現(xiàn)內(nèi)存清理加速球效果
前言
用過(guò)獵豹清理大師或者相類似的安全軟件,大家都知道它們都會(huì)有一個(gè)功能,那就是內(nèi)存清理,而展現(xiàn)的形式是通過(guò)一個(gè)圓形的小球來(lái)顯示內(nèi)存大小,通過(guò)百分比數(shù)字以及進(jìn)度條的形式來(lái)顯示清理的進(jìn)度。本文將對(duì)該效果的實(shí)現(xiàn)過(guò)程進(jìn)行詳細(xì)講述,但不涉及內(nèi)存清理的實(shí)現(xiàn)。
預(yù)覽
我們先來(lái)看看最終實(shí)現(xiàn)的效果是怎樣的(gif效果有點(diǎn)差):
從上面的圖片,我們可以看出:
①當(dāng)加速球View顯示的時(shí)候,進(jìn)度條以及百分比數(shù)字會(huì)從0%開(kāi)始增加到某一數(shù)值(60%)。
②進(jìn)度條停止增加后,中間的圓沿著Y軸開(kāi)始翻轉(zhuǎn),會(huì)翻轉(zhuǎn)180度,上面的百分比數(shù)字并不會(huì)出現(xiàn)鏡像效果(下面會(huì)提到)。
③用戶點(diǎn)擊該小球后,開(kāi)始清理內(nèi)存,進(jìn)度條和百分比數(shù)字會(huì)經(jīng)歷減小至0,再由0增加到某一數(shù)值的過(guò)程。
實(shí)現(xiàn)過(guò)程詳解
其實(shí)上面的效果,筆者是仿照獵豹清理大師的加速球所實(shí)現(xiàn)的,略有不同,但大致形式相同。如果讀者對(duì)上面的效果感興趣,那么請(qǐng)繼續(xù)讀下去吧,接下來(lái)是正文部分。
Step 1.初始化
我們首先要新建一個(gè)LieBaoView.java,繼承自View,我們重寫(xiě)它的構(gòu)造函數(shù)如下:
public LieBaoView(Context context) { super(context); init(); } public LieBaoView(Context context, AttributeSet attrs) { super(context, attrs); init(); } public LieBaoView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); }
無(wú)論通過(guò)哪種方式實(shí)例化該View,都會(huì)調(diào)用init()方法,該方法主要用于處理初始化各種成員變量,那么我們又需要哪些成員變量或者哪些實(shí)例來(lái)幫助我們呢?
筆者的思路是這樣的:通過(guò)一個(gè)空白的bitmap,我們?cè)谏厦胬L制圓形、文字等,這樣最后再將這個(gè)bitmap繪制到我們的view上面。
因此,我們?cè)诔跏蓟臅r(shí)候,需要獲取到各種Paint(畫(huà)筆)、Bitmap(空白圖片)、Canvas(畫(huà)布)等的實(shí)例。我們?cè)傧胍幌拢褐虚g的圓是可以旋轉(zhuǎn)的,那么中間的旋轉(zhuǎn)圓就不能和別的圓放到同一個(gè)bitmap上,否則會(huì)給后面旋轉(zhuǎn)的實(shí)現(xiàn)帶來(lái)麻煩,因此我們可以準(zhǔn)備兩張空白的bitmap。那么,我們可以先這樣:
public void init(){ //繪制背景圓的畫(huà)筆 mBackgroundCirclePaint = new Paint(); mBackgroundCirclePaint.setAntiAlias(true); mBackgroundCirclePaint.setColor(Color.argb(0xff, 0x10, 0x53, 0xff)); //繪制旋轉(zhuǎn)圓的畫(huà)筆 mFrontCirclePaint = new Paint(); mFrontCirclePaint.setAntiAlias(true); mFrontCirclePaint.setColor(Color.argb(0xff, 0x5e, 0xae, 0xff)); //繪制文字的畫(huà)筆 mTextPaint = new Paint(); mTextPaint.setAntiAlias(true); mTextPaint.setTextSize(80); mTextPaint.setColor(Color.WHITE); //繪制進(jìn)度條的畫(huà)筆 mArcPaint = new Paint(); mArcPaint.setAntiAlias(true); mArcPaint.setColor(Color.WHITE); mArcPaint.setStrokeWidth(12); mArcPaint.setStyle(Paint.Style.STROKE); mBitmap = Bitmap.createBitmap(mWidth,mHeight, Bitmap.Config.ARGB_8888); mBitmapCanvas = new Canvas(mBitmap); //將畫(huà)布和Bitmap關(guān)聯(lián) //旋轉(zhuǎn)bitmap與畫(huà)布 mOverturnBitmap = Bitmap.createBitmap(mWidth,mHeight, Bitmap.Config.ARGB_8888); mOverturnBitmapCanvas = new Canvas(mOverturnBitmap); //省略了一部分... //Camera、Matrix、Runnable等下面會(huì)講述 mMatrix = new Matrix(); mCamera = new Camera(); }
上面主要是初始化了各種不同的畫(huà)筆類型,以及準(zhǔn)備了兩個(gè)Bitmap及其相關(guān)聯(lián)的畫(huà)布,我們?cè)谄潢P(guān)聯(lián)的畫(huà)布上進(jìn)行繪制即可,這樣就能得到有著內(nèi)容的兩個(gè)Bitmap了。
我們接著往下思考:如果要實(shí)現(xiàn)翻轉(zhuǎn)效果,我們還需要些什么?Android SDK為我們準(zhǔn)備好了一套工具:Camera和Matrix,利用這兩個(gè)工具,我們可以很方便地實(shí)現(xiàn)對(duì)Bitmap的各種變換,比如縮放、平移、翻轉(zhuǎn)等。關(guān)于Camera和Matrix,讀者可以去搜索更詳細(xì)的相關(guān)知識(shí),這里就不展開(kāi)來(lái)詳談了。最后,我們還需要Runnable,因?yàn)樾枰獙?shí)現(xiàn)自動(dòng)翻轉(zhuǎn)以及進(jìn)度條的自動(dòng)增加與減少的,Runnable下面會(huì)詳細(xì)講述,先不用著急,當(dāng)然了,還需要設(shè)置一個(gè)點(diǎn)擊監(jiān)聽(tīng)器。
Step 2.繪制圖像
上面已經(jīng)為我們準(zhǔn)備好了畫(huà)筆、畫(huà)布等,我們接下來(lái)就來(lái)繪制所需的圖像。通過(guò)重寫(xiě)View的onDraw()方法即可。
①繪制背景圓,也即上圖中最外層深藍(lán)色的圓:
mBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2, mBackgroundCirclePaint);
②繪制中間的白色背景圓,也即旋轉(zhuǎn)圓進(jìn)行翻轉(zhuǎn)的過(guò)程中,背景的白色部分:
mBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2 - mPadding, mTextPaint);
③繪制進(jìn)度條,弧形進(jìn)度條該怎么實(shí)現(xiàn)呢?這里給出筆者的一個(gè)思路:通過(guò)canvas的drawArc()方法來(lái)實(shí)現(xiàn),該方法能在一個(gè)矩形內(nèi)繪制一個(gè)最大的圓(或者橢圓),設(shè)置畫(huà)筆為空心以及畫(huà)筆線條寬度為12左右即可,這樣就能實(shí)現(xiàn)一個(gè)粗弧線了,然后通過(guò)不斷地調(diào)用onDraw()方法,修改drawArc()的角度來(lái)實(shí)現(xiàn)進(jìn)度條效果。如果大家還有什么別的實(shí)現(xiàn)方法,歡迎交流。
mBitmapCanvas.save(); //實(shí)例化一個(gè)矩形,該矩形的左上角和右下角坐標(biāo)與原Bitmap并不重合,這是因?yàn)橐? //進(jìn)度條與最外面的圓有一定的間隙 RectF rectF = new RectF(10,10,mWidth-10,mHeight-10); //先將畫(huà)布逆時(shí)針旋轉(zhuǎn)90度,這樣drawArc的起始角度就能從0度開(kāi)始,省去不必要的麻煩 mBitmapCanvas.rotate(-90, mWidth / 2, mHeight / 2); mBitmapCanvas.drawArc(rectF, 0, ((float)mProgress/mMaxProgress)*360, false, mArcPaint); mBitmapCanvas.restore(); canvas.drawBitmap(mBitmap, 0, 0, null);
④繪制中間的旋轉(zhuǎn)圓。上面說(shuō)到,由于要實(shí)現(xiàn)翻轉(zhuǎn)效果,那么不能再同一張Bitmap上繪制了,所以我們用另一張空白的Bitmap。旋轉(zhuǎn)圓的繪制很簡(jiǎn)單,只要它的半徑比外圓半徑以及進(jìn)度條寬度相加之和還要小即可:
mOverturnBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2 - mPadding, mFrontCirclePaint);
⑤最后一步,在旋轉(zhuǎn)圓上繪制百分比數(shù)字。繪制文字,要用到Canvas的drawText方法,我們重點(diǎn)來(lái)看看這個(gè)方法:
/** * Draw the text, with origin at (x,y), using the specified paint. The * origin is interpreted based on the Align setting in the paint. * * @param text The text to be drawn * @param x The x-coordinate of the origin of the text being drawn * @param y The y-coordinate of the baseline of the text being drawn * @param paint The paint used for the text (e.g. color, size, style) */ public void drawText(@NonNull String text, float x, float y, @NonNull Paint paint) { //... }
第一個(gè)和第四個(gè)參數(shù)沒(méi)什么好說(shuō)的,第二個(gè)參數(shù)表示文字開(kāi)始的x坐標(biāo),第三個(gè)參數(shù)表示文字的baseline的y坐標(biāo)。要使文字居中顯示,我們只需要設(shè)置適當(dāng)?shù)膞、y坐標(biāo)即可,那么baseline又是什么呢?它其實(shí)代表著文本的基準(zhǔn)點(diǎn),我們來(lái)看一幅圖:
從圖中可以看出,baseline以上至文本最高點(diǎn)為Ascent,為負(fù)值;baseline以下至文本最低點(diǎn)為Descent,為正值。因此,如果我們要使文本在控件內(nèi)居中顯示,那么我們可以利用-(ascent-descent)/2計(jì)算出文本的高度的一半,此時(shí)再利用mHeight/2(控件高度的一半)加上該值,即可得出在控件中的baseline值,此時(shí)也就實(shí)現(xiàn)了居中顯示,代碼如下:
String text = (int) (((float)mProgress / mMaxProgress) *100) + "%"; //獲取文本的寬度 float textWidth = mTextPaint.measureText(text); //獲取文本規(guī)格 Paint.FontMetrics metrics = mTextPaint.getFontMetrics(); float baseLine = mHeight / 2 - (metrics.ascent + metrics.descent) /2; mOverturnBitmapCanvas.drawText(text, mWidth / 2 - textWidth / 2, baseLine, mTextPaint);
最后,再將bitmap繪制到view上:
canvas.drawBitmap(mOverturnBitmap, mMatrix, null);
經(jīng)過(guò)以上的繪制,我們先看看效果如何:
那么基本效果都已經(jīng)實(shí)現(xiàn)了。接下來(lái),我們將會(huì)實(shí)現(xiàn)動(dòng)態(tài)效果。
Step 3.實(shí)現(xiàn)自動(dòng)翻轉(zhuǎn)的效果
從上面的動(dòng)畫(huà)效果來(lái)看,我們首先讓進(jìn)度條從0增加到某個(gè)數(shù)值,接著再自動(dòng)翻轉(zhuǎn)。增加數(shù)值的實(shí)現(xiàn)很簡(jiǎn)單,只需要啟用一個(gè)Runnable,在Runnable內(nèi)把mProgress值不斷增加,再調(diào)用invalidate()方法刷新View即可。等進(jìn)度條增加完畢,那么就開(kāi)始翻轉(zhuǎn),翻轉(zhuǎn)的話利用Camera和Matrix對(duì)中間的bitmap進(jìn)行操作,不斷改變角度就能實(shí)現(xiàn),我們來(lái)看看代碼:
在onDraw()方法內(nèi):
@Override protected void onDraw(Canvas canvas) { //.... //如果當(dāng)前正在旋轉(zhuǎn) if(isRotating) { mCamera.save(); //旋轉(zhuǎn)角度 mCamera.rotateY(mRotateAngle); //如果旋轉(zhuǎn)角度大于或等于180度的時(shí)候,減去180度 if (mRotateAngle >= 180) { mRotateAngle -= 180; } //根據(jù)Camera的操作來(lái)獲得相應(yīng)的矩陣 mCamera.getMatrix(mMatrix); mCamera.restore(); mMatrix.preTranslate(-mWidth / 2, -mHeight / 2); mMatrix.postTranslate(mWidth / 2, mHeight / 2); } canvas.drawBitmap(mOverturnBitmap, mMatrix, null); //如果當(dāng)前控件尚未進(jìn)行翻轉(zhuǎn)過(guò)程 if(!isRotating && !isInital){ //設(shè)置isIncreasing,表示先開(kāi)始進(jìn)度條的增加過(guò)程 isIncreasing = true; isRotating = true; postDelayed(mRotateRunnable,10); }
接著,我們來(lái)寫(xiě)mRotateRunnable,Runnable的初始化在init()方法內(nèi):
mRotateRunnable = new Runnable() { @Override public void run() { //如果當(dāng)前是正在增加過(guò)程 if(isIncreasing){ Log.d("cylog","mProgress:"+mProgress); //當(dāng)進(jìn)度增加到某一個(gè)數(shù)值的時(shí)候,停止增加 if(mProgress >= 59){ isIncreasing = false; } mProgress++; }else { //如果增加過(guò)程結(jié)束,那么開(kāi)始翻轉(zhuǎn) //如果mRotateAngle是大于90度的,表示bitmap已經(jīng)翻轉(zhuǎn)了90度, //此時(shí)bitmap的內(nèi)容變成鏡像內(nèi)容,為了不出現(xiàn)鏡像效果,我們需要再轉(zhuǎn)過(guò)180度, //此時(shí)就變?yōu)檎5娘@示了,而這多轉(zhuǎn)的180度在onDraw內(nèi)會(huì)減去。 if (mRotateAngle > 90 && mRotateAngle < 180) mRotateAngle = mRotateAngle + 3 + 180; //如果mRotateAngle超過(guò)了180度,翻轉(zhuǎn)過(guò)程完成 else if (mRotateAngle >= 180) { isRotating = false; isInital = true; mRotateAngle = 0; return; } else //每次角度增加3,這個(gè)可以微調(diào),適當(dāng)即可 mRotateAngle += 3; } invalidate(); //25ms后再次調(diào)用該方法 postDelayed(this,25); } };
經(jīng)過(guò)以上的Runnable以及在onDraw()方法的配合,已經(jīng)可以實(shí)現(xiàn)自動(dòng)翻轉(zhuǎn)的效果了。
Step 4.實(shí)現(xiàn)點(diǎn)擊清理的效果
好了,我們來(lái)實(shí)現(xiàn)最后的效果,同樣,我們利用一個(gè)Runnable來(lái)實(shí)現(xiàn),由于該清理效果是需要用戶點(diǎn)擊小球后才開(kāi)始清理的,所以我們需要一個(gè)事件監(jiān)聽(tīng)器,每當(dāng)用戶點(diǎn)擊后,在onClick方法內(nèi)post一個(gè)Runnable即可。
先實(shí)現(xiàn)mCleaningRunnable,在init()方法內(nèi):
mCleaningRunnable = new Runnable() { @Override public void run() { //如果當(dāng)前進(jìn)度超過(guò)某一數(shù)值,那么停止清理 if (mProgress >= 60) { isCleaning = false; return; } //如果當(dāng)前處于下降過(guò)程,mProgress不斷減少,直到為0 if (isDescending) { mProgress--; if (mProgress <= 0) isDescending = false; } else { mProgress++; } invalidate(); postDelayed(this,40); } }; setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if(isCleaning) return; //如果當(dāng)前正在清理過(guò)程,那么直接return,防止post過(guò)多 //設(shè)置flag,來(lái)進(jìn)行清理 isDescending = true; isCleaning = true; mProgress--; postDelayed(mCleaningRunnable, 40); } });
上面的邏輯實(shí)現(xiàn)了,每當(dāng)點(diǎn)擊后,先把進(jìn)度值不斷減少直到0,接著又不斷增加直到某個(gè)固定的值,通過(guò)每一個(gè)調(diào)用invalidate()方法來(lái)通知組件刷新,這樣就實(shí)現(xiàn)了動(dòng)態(tài)效果。
好了,到目前為止,所有的效果已經(jīng)實(shí)現(xiàn)了,全部代碼在下面貼上。謝謝大家的閱讀~
public class LieBaoView extends View { private Paint mBackgroundCirclePaint; private Paint mFrontCirclePaint; private Paint mTextPaint; private Paint mArcPaint; private Bitmap mBitmap; private Bitmap mOverturnBitmap; private Canvas mBitmapCanvas; private Canvas mOverturnBitmapCanvas; private Matrix mMatrix; private Camera mCamera; private int mWidth = 400; private int mHeight = 400; private int mPadding = 20; private int mProgress = 0; private int mMaxProgress = 100; private int mRotateAngle = 0; private Runnable mRotateRunnable; private Runnable mCleaningRunnable; private boolean isRotating; private boolean isInital = false; private boolean isDescending; private boolean isIncreasing; private boolean isCleaning; public LieBaoView(Context context) { super(context); init(); } public LieBaoView(Context context, AttributeSet attrs) { super(context, attrs); init(); } public LieBaoView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { setMeasuredDimension(mWidth,mHeight); } public void init(){ //繪制背景圓的畫(huà)筆 mBackgroundCirclePaint = new Paint(); mBackgroundCirclePaint.setAntiAlias(true); mBackgroundCirclePaint.setColor(Color.argb(0xff, 0x10, 0x53, 0xff)); //繪制旋轉(zhuǎn)圓的畫(huà)筆 mFrontCirclePaint = new Paint(); mFrontCirclePaint.setAntiAlias(true); mFrontCirclePaint.setColor(Color.argb(0xff, 0x5e, 0xae, 0xff)); //繪制文字的畫(huà)筆 mTextPaint = new Paint(); mTextPaint.setAntiAlias(true); mTextPaint.setTextSize(80); mTextPaint.setColor(Color.WHITE); //繪制進(jìn)度條的畫(huà)筆 mArcPaint = new Paint(); mArcPaint.setAntiAlias(true); mArcPaint.setColor(Color.WHITE); mArcPaint.setStrokeWidth(12); mArcPaint.setStyle(Paint.Style.STROKE); mBitmap = Bitmap.createBitmap(mWidth,mHeight, Bitmap.Config.ARGB_8888); mBitmapCanvas = new Canvas(mBitmap); //將畫(huà)布和Bitmap關(guān)聯(lián) //旋轉(zhuǎn)bitmap與畫(huà)布 mOverturnBitmap = Bitmap.createBitmap(mWidth,mHeight, Bitmap.Config.ARGB_8888); mOverturnBitmapCanvas = new Canvas(mOverturnBitmap); mMatrix = new Matrix(); mCamera = new Camera(); mRotateRunnable = new Runnable() { @Override public void run() { //如果當(dāng)前是正在增加過(guò)程 if(isIncreasing){ Log.d("cylog","mProgress:"+mProgress); //當(dāng)進(jìn)度增加到某一個(gè)數(shù)值的時(shí)候,停止增加 if(mProgress >= 59){ isIncreasing = false; } mProgress++; }else { //如果增加過(guò)程結(jié)束,那么開(kāi)始翻轉(zhuǎn) //如果mRotateAngle是大于90度的,表示bitmap已經(jīng)翻轉(zhuǎn)了90度, //此時(shí)bitmap的內(nèi)容變成鏡像內(nèi)容,為了不出現(xiàn)鏡像效果,我們需要再轉(zhuǎn)過(guò)180度, //此時(shí)就變?yōu)檎5娘@示了,而這多轉(zhuǎn)的180度在onDraw內(nèi)會(huì)減去。 if (mRotateAngle > 90 && mRotateAngle < 180) mRotateAngle = mRotateAngle + 3 + 180; //如果mRotateAngle超過(guò)了180度,翻轉(zhuǎn)過(guò)程完成 else if (mRotateAngle >= 180) { isRotating = false; isInital = true; mRotateAngle = 0; return; } else //每次角度增加3,這個(gè)可以微調(diào),適當(dāng)即可 mRotateAngle += 3; } invalidate(); //25ms后再次調(diào)用該方法 postDelayed(this,25); } }; mCleaningRunnable = new Runnable() { @Override public void run() { //如果當(dāng)前進(jìn)度超過(guò)某一數(shù)值,那么停止清理 if (mProgress >= 60) { isCleaning = false; return; } //如果當(dāng)前處于下降過(guò)程,mProgress不斷減少,直到為0 if (isDescending) { mProgress--; if (mProgress <= 0) isDescending = false; } else { mProgress++; } invalidate(); postDelayed(this,40); } }; setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if(isCleaning) return; isDescending = true; isCleaning = true; mProgress--; postDelayed(mCleaningRunnable, 40); } }); } @Override protected void onDraw(Canvas canvas) { mBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2, mBackgroundCirclePaint); mBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2 - mPadding, mTextPaint); mBitmapCanvas.save(); //實(shí)例化一個(gè)矩形,該矩形的左上角和右下角坐標(biāo)與原Bitmap并不重合,這是因?yàn)橐? //進(jìn)度條與最外面的圓有一定的間隙 RectF rectF = new RectF(10,10,mWidth-10,mHeight-10); //先將畫(huà)布逆時(shí)針旋轉(zhuǎn)90度,這樣drawArc的起始角度就能從0度開(kāi)始,省去不必要的麻煩 mBitmapCanvas.rotate(-90, mWidth / 2, mHeight / 2); mBitmapCanvas.drawArc(rectF, 0, ((float)mProgress/mMaxProgress)*360, false, mArcPaint); mBitmapCanvas.restore(); canvas.drawBitmap(mBitmap, 0, 0, null); mOverturnBitmapCanvas.drawCircle(mWidth / 2, mHeight / 2, mWidth / 2 - mPadding, mFrontCirclePaint); String text = (int) (((float)mProgress / mMaxProgress) *100) + "%"; //獲取文本的寬度 float textWidth = mTextPaint.measureText(text); //獲取文本規(guī)格 Paint.FontMetrics metrics = mTextPaint.getFontMetrics(); float baseLine = mHeight / 2 - (metrics.ascent + metrics.descent) /2; mOverturnBitmapCanvas.drawText(text, mWidth / 2 - textWidth / 2, baseLine, mTextPaint); //如果當(dāng)前正在旋轉(zhuǎn) if(isRotating) { mCamera.save(); //旋轉(zhuǎn)角度 mCamera.rotateY(mRotateAngle); //如果旋轉(zhuǎn)角度大于或等于180度的時(shí)候,減去180度 if (mRotateAngle >= 180) { mRotateAngle -= 180; } //根據(jù)Camera的操作來(lái)獲得相應(yīng)的矩陣 mCamera.getMatrix(mMatrix); mCamera.restore(); mMatrix.preTranslate(-mWidth / 2, -mHeight / 2); mMatrix.postTranslate(mWidth / 2, mHeight / 2); } canvas.drawBitmap(mOverturnBitmap, mMatrix, null); //如果當(dāng)前控件尚未進(jìn)行翻轉(zhuǎn)過(guò)程 if(!isRotating && !isInital){ //設(shè)置isIncreasing,表示先開(kāi)始進(jìn)度條的增加過(guò)程 isIncreasing = true; isRotating = true; postDelayed(mRotateRunnable,10); } } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
RecyclerView實(shí)現(xiàn)查看更多及收起
這篇文章主要為大家詳細(xì)介紹了RecyclerView實(shí)現(xiàn)查看更多及收起,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01Android 加載大圖及多圖避免程序出現(xiàn)OOM(OutOfMemory)異常
這篇文章主要介紹了Android 加載大圖及多圖避免程序出現(xiàn)OOM(OutOfMemory)異常的相關(guān)資料,需要的朋友可以參考下2017-03-03Android實(shí)現(xiàn)定時(shí)任務(wù)功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)定時(shí)任務(wù)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01Android拍照保存在系統(tǒng)相冊(cè)不顯示的問(wèn)題解決方法
我們保存相冊(cè)到Android手機(jī)的時(shí)候,然后去打開(kāi)系統(tǒng)圖庫(kù)找不到我們想要的那張圖片,那是因?yàn)槲覀儾迦氲膱D片還沒(méi)有更新的緣故,下面與大家分享下此問(wèn)題的解決方法2013-06-06Android中使用TabHost 與 Fragment 制作頁(yè)面切換效果
這篇文章主要介紹了Android中使用TabHost 與 Fragment 制作頁(yè)面切換效果的相關(guān)資料,需要的朋友可以參考下2016-03-03Android程序開(kāi)發(fā)仿新版QQ鎖屏下彈窗功能
最近做了一個(gè)項(xiàng)目,其中涉及到這樣一個(gè)功能:新版的qq能在鎖屏下彈窗顯示qq消息,下面小編抽時(shí)間把實(shí)現(xiàn)代碼分享給大家感興趣的朋友參考下吧2016-09-09Android采用GET方法進(jìn)行網(wǎng)絡(luò)傳值
這篇文章主要為大家詳細(xì)介紹了Android采用GET方法進(jìn)行網(wǎng)絡(luò)傳值的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12