編輯:關於Android編程
自學了android有幾個月了,跟著網上的節奏,應該早點寫些博客來提高自己的水准的。但苦於技術水准始終不自信(也是不過關的結果吧),就一直只是將自己學習過程中的問題和重要的知識點寫在自己的筆記文檔中。
但,總感覺一個人寫下來成就感還是欠缺了那麼一些,而且有些問題及解答方法拋出來,是有可能得到更多好的反饋及解決方案的。於是,本著不作不會死的心態,一步一步在技術成長的道路前行——>這篇博客就是其中一步!
若博客中有些技術知識點有誤或者有更優化的解答方案,還望各位小伙伴多多指出。
以下是正題了:
目標:利用SurfaceView實現一個簡單的計時器
圖示:
描述:1.利用SurfaceView來實現計時功能,同時不斷將圓弧畫滿;2.點擊按鈕可以停止計時;
重點:
1.自定義SurfaceView中針對SurfaceHolder.CallBack的三個方法進行覆寫;
2.通過surfaceHolder.lockCanvas()在新開的線程中得到canvas對象,從而進行圖形和時間文字的繪制;
3.通過設置flag值,從而控制在線程run()方法中邏輯代碼的執行;
重要部分代碼:
public TestView(Context context) { super(context); surfaceHolder = getHolder(); surfaceHolder.addCallback(this); countThread = new CountThread(surfaceHolder); } public TestView(Context context, AttributeSet attributeSet) { super(context, attributeSet); surfaceHolder = getHolder(); surfaceHolder.addCallback(this); countThread = new CountThread(surfaceHolder); }
以上為自定義的SurfaceView(TestView)的構造函數,做相應的初始化工作。(第二個構造方法在實現過程中沒有覆寫,導致如果是通過布局文件引入進Activity中時,則顯示不出View——>也就是一般自定義View一定要覆寫的構造方法)。
在構造方法中初始化了自定義的內部線程類CountThread,用來執行繪制工作。
以下為CountThread類的run()方法執行邏輯:
@Override public void run() { Canvas canvas = null; int pivotX = getResources().getDisplayMetrics().widthPixels / 2; RectF rectF = new RectF(pivotX - 300, pivotX - 300, pivotX + 300, pivotX + 300); while (!isStop) { try { canvas = surfaceHolder.lockCanvas(); canvas.drawColor(Color.WHITE);//設置畫布背景為白色 // canvas.drawRoundRect(300, 300, 600, 600, 150, 150, paint);//直接使用該行代碼來畫圓是行不通的,因為這個方法要求版本21,我的手機運行android版本是19 canvas.drawArc(rectF, -90, endAngle++, false, paint);//-90在這裡不等於270,所以要想從最上方開始畫弧,就得用-90 canvas.drawText(countTime(endAngle), pivotX, pivotX, paintText);//顯示計算的時間 Thread.sleep(1000); } catch (Exception e) { e.printStackTrace(); } finally { if (canvas != null) {//需要對canvas進行非空判斷 surfaceHolder.unlockCanvasAndPost(canvas); } } } }
主要就是通過Thread.sleep(1000)來停頓一秒,從而通過endAngle來計數。
至此,基本的程序邏輯已經實現。再講該自定義View引入進Activity指定的layout布局文件中即可進行顯示。
但,這其中也會涉及到相應的問題。譬如:
1.圖示中的按鈕無法顯示出來,只能看到自定義的SurfaceView的視圖;
我的做法是:將布局設定為FrameLayout即可。因為SurfaceView是浮在窗口的一層,那麼就可以把它看做一個圖層。
2.java.lang.IllegalThreadStateException:Thread already started
該錯誤表明,線程已經存在了。這種錯誤的操作重現是:按下home鍵或者menu鍵會導致該自定的SurfaceView銷毀,但線程並沒有被銷毀,再次啟動該自定義SurfaceView的時候又去重新啟動該線程。解決的方法是:
@Override public void surfaceCreated(SurfaceHolder holder) { if (!countThread.isAlive()) {//如果線程不存在,則啟動線程——>當應用掛起的時候Thread是存在的,如果不做這個判斷,會報“Thread already started ”錯誤 countThread.start();//SurfaceView創建時開啟線程 } }
在surfaceCreated()方法中對該線程是否是在存活中進行判斷。
當然,這裡面還有最重要的一個問題:
當按下home鍵或者menu鍵時,程序是沒有在計時的。那麼這種情況下,我的一個解決方案是,通過開啟一個service來接收程序停止( onStop() )時已經計時的數值,然後傳遞給service記下並計時,當應用程序界面重新回歸屏幕時( onRestart() )則將數據取出並回傳到自定義的SurfaceView的邏輯run()方法中繼續計時。
整個小程序的代碼可以通過以下鏈接下載:
點擊進入下載頁面:http://xiazai.jb51.net/201701/yuanma/AndroidSurfaceView(jb51.net).rar
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。
我們都知道取消標題欄有兩種方式,一種是在Java代碼中取消,另一種通過設置styles.xml文件中的Theme即可;如下圖:第一種:第二種:但是運行在Android 5
通過內部存儲方式實現了在兩個Activity之間傳遞Bitmap對象以及其它支持串行化的Java對象,關鍵點有如下:1. HTTP客戶端下載圖片,通過Imag
事件分發在Android中非常重要,在滑動沖突,下拉刷新,嵌套滑動的時候都需要非常清楚事件分發的機制,才能寫好對應的處理代碼。曾經以為我對事件分發已經很清楚了,也寫過幾篇
從意圖返回結果 startActivity()方法調用另一個活動,但並沒有返回結果給當前活動。此時如想從一個活動中回傳數據,就要使用startActivityFo