Android面試常見問題

一般的面試流程是筆試完就接着是面試了,面試時技術經理會問你一些你工作中遇到的Android方面的問題,談談你所做的項目,和在項目中所扮演的角色。今天我就給大家整理一些,面試中常見的面試官提的一些問題?

1.要做一個儘可能流暢的ListView,你平時在工作中如何進行優化的?
①Item佈局,層級越少越好,使用hierarchyview工具查看優化。
②複用convertView
③使用ViewHolder
④item中有圖片時,異步加載
⑤快速滑動時,不加載圖片
⑥item中有圖片時,應對圖片進行適當壓縮
⑦實現數據的分頁加載

2.對於Android 的安全問題,你知道多少
①錯誤導出組件
② 參數校驗不嚴
③WebView引入各種安全問題,webview中的js注入
④不混淆、不防二次打包
⑤明文存儲關鍵信息
⑦ 錯誤使用HTTPS
⑧山寨加密方法
⑨濫用權限、內存泄露、使用debug簽名

3. 如何縮減APK包大小?
代碼
保持良好的編程習慣,不要重複或者不用的代碼,謹慎添加libs,移除使用不到的libs。
使用proguard混淆代碼,它會對不用的代碼做優化,並且混淆後也能夠減少安裝包的大小。
native code的部分,大多數情況下只需要支持armabi與x86的架構即可。如果非必須,可以考慮拿掉x86的部分。
資源
使用Lint工具查找沒有使用到的資源。去除不使用的圖片,String,XML等等。 assets目錄下的資源請確保沒有用不上的文件。
生成APK的時候,aapt工具本身會對png做優化,但是在此之前還可以使用其他工具如tinypng對圖片進行進一步的壓縮預處理。
jpeg還是png,根據需要做選擇,在某些時候jpeg可以減少圖片的體積。 對於9.png的圖片,可拉伸區域儘量切小,另外可以通過使用9.png拉伸達到大圖效果的時候儘量不要使用整張大圖。
策略
有選擇性的提供hdpi,xhdpi,xxhdpi的圖片資源。建議優先提供xhdpi的圖片,對於mdpi,ldpi與xxxhdpi根據需要提供有差異的部分即可。
儘可能的重用已有的圖片資源。例如對稱的圖片,只需要提供一張,另外一張圖片可以通過代碼旋轉的方式實現。
能用代碼繪製實現的功能,儘量不要使用大量的圖片。例如減少使用多張圖片組成animate-list的AnimationDrawable,這種方式提供了多張圖片很佔空間。

4.Android與服務器交互的方式中的對稱加密和非對稱加密是什麼?
對稱加密,就是加密和解密數據都是使用同一個key,這方面的算法有DES。
非對稱加密,加密和解密是使用不同的key。發送數據之前要先和服務端約定生成公鑰和私鑰,使用公鑰加密的數據可以用私鑰解密,反之。這方面的算法有RSA。ssh 和 ssl都是典型的非對稱加密。

5.設備橫豎屏切換的時候,接下來會發生什麼?
1、不設置Activity的android:configChanges時,切屏會重新調用各個生命週期,切橫屏時會執行一次,切豎屏時會執行兩次
2、設置Activity的android:configChanges=”orientation”時,切屏還是會重新調用各個生命週期,切橫、豎屏時只會執行一次
3、設置Activity的android:configChanges=”orientation|keyboardHidden”時,切屏不會重新調用各個生命週期,只會執行onConfigurationChanged方法

6.Android啓動Service的兩種方式是什麼? 它們的適用情況是什麼?
如果後臺服務開始後基本可以獨立運行的話,可以用startService。音樂播放器就可以這樣用。它們會一直運行直到你調用 stopSelf或者stopService。你可以通過發送Intent或者接收Intent來與正在運行的後臺服務通信,但大部分時間,你只是啓動服務並讓它獨立運行。如果你需要與後臺服務通過一個持續的連接來比較頻繁地通信,建議使用bind()。比如你需要定位服務不停地把更新後的地理位置傳給UI。Binder比Intent開發起來複雜一些,但如果真的需要,你也只能使用它。
startService:生命週期與調用者不同。啓動後若調用者未調用stopService而直接退出,Service仍會運行
bindService:生命週期與調用者綁定,調用者一旦退出,Service就會調用unBind->onDestroy

7.談談你對Android中Context的理解?
Context:包含上下文信息(外部值) 的一個參數. Android 中的 Context 分三種,Application Context ,Activity Context ,Service Context.
它描述的是一個應用程序環境的信息,通過它我們可以獲取應用程序的資源和類,也包括一些應用級別操作,例如:啓動一個Activity,發送廣播,接受Intent信息等

8.Service的onCreate回調在UI線程中嗎?
Service生命週期的各個回調和其他的應用組件一樣,是跑在主線程中,會影響到你的UI操作或者阻塞主線程中的其他事情

9.請介紹下AsyncTask的內部實現,適用的場景是?
AsyncTask內部也是Handler機制來完成的,只不過Android提供了執行框架來提供線程池來執行相應地任務,因爲線程池的大小問題,所以AsyncTask只應該用來執行耗時時間較短的任務,比如HTTP請求,大規模的下載和數據庫的更改不適用於AsyncTask,因爲會導致線程池堵塞,沒有線程來執行其他的任務,導致的情形是會發生AsyncTask根本執行不了的問題。

10.談談你對binder機制的理解?
binder是一種IPC機制,進程間通訊的一種工具.
Java層可以利用aidl工具來實現相應的接口.

11.Android中進程間通信有哪些實現方式?
Intent,Binder(AIDL),Messenger,BroadcastReceiver

12.介紹下實現一個自定義view的基本流程
1、自定義View的屬性 編寫attr.xml文件
2、在layout佈局文件中引用,同時引用命名空間
3、在View的構造方法中獲得我們自定義的屬性 ,在自定義控件中進行讀取(構造方法拿到attr.xml文件值)
4、重寫onMesure
5、重寫onDraw

13.Android中touch事件的傳遞機制是怎樣的?
1.Touch事件傳遞的相關API有dispatchTouchEvent、onTouchEvent、onInterceptTouchEvent
2.Touch事件相關的類有View、ViewGroup、Activity
3.Touch事件會被封裝成MotionEvent對象,該對象封裝了手勢按下、移動、鬆開等動作
4.Touch事件通常從Activity#dispatchTouchEvent發出,只要沒有被消費,會一直往下傳遞,到最底層的View。
5.如果Touch事件傳遞到的每個View都不消費事件,那麼Touch事件會反向向上傳遞,最終交由Activity#onTouchEvent處理.
6.onInterceptTouchEvent爲ViewGroup特有,可以攔截事件.
7.Down事件到來時,如果一個View沒有消費該事件,那麼後續的MOVE/UP事件都不會再給它

14.Android多線程的實現方式有哪些?
Thread & AsyncTask
Thread 可以與Loop 和 Handler 共用建立消息處理隊列
AsyncTask 可以作爲線程池並行處理多任務

15.Android開發中何時使用多進程?使用多進程的好處是什麼?
要想知道如何使用多進程,先要知道Android裏的多進程概念。一般情況下,一個應用程序就是一個進程,這個進程名稱就是應用程序包名。我們知道進程是系統分配資源和調度的基本單位,所以每個進程都有自己獨立的資源和內存空間,別的進程是不能任意訪問其他進程的內存和資源的。
那如何讓自己的應用擁有多個進程?
很簡單,我們的四大組件在AndroidManifest文件中註冊的時候,有個屬性是android:process,1.這裏可以指定組件的所處的進程。默認就是應用的主進程。指定爲別的進程之後,系統在啓動這個組件的時候,就先創建(如果還沒創建的話)這個進程,然後再創建該組件。你可以重載Application類的onCreate方法,打印出它的進程名稱,就可以清楚的看見了。再設置android:process屬性時候,有個地方需要注意:如果是android:process=”:deamon”,以:開頭的名字,則表示這是一個應用程序的私有進程,否則它是一個全局進程。私有進程的進程名稱是會在冒號前自動加上包名,而全局進程則不會。一般我們都是有私有進程,很少使用全局進程。他們的具體區別不知道有沒有誰能補充一下。

2.使用多進程顯而易見的好處就是分擔主進程的內存壓力。我們的應用越做越大,內存越來越多,將一些獨立的組件放到不同的進程,它就不佔用主進程的內存空間了。當然還有其他好處,有心人會發現Android後臺進程裏有很多應用是多個進程的,因爲它們要常駐後臺,特別是即時通訊或者社交應用,不過現在多進程已經被用爛了。典型用法是在啓動一個不可見的輕量級私有進程,在後臺收發消息,或者做一些耗時的事情,或者開機啓動這個進程,然後做監聽等。還有就是防止主進程被殺守護進程,守護進程和主進程之間相互監視,有一方被殺就重新啓動它。應該還有還有其他好處,這裏就不多說了。

3.壞處的話,多佔用了系統的空間,大家都這麼用的話系統內存很容易佔滿而導致卡頓。消耗用戶的電量。應用程序架構會變複雜,應爲要處理多進程之間的通信。這裏又是另外一個問題了。

16.ANR是什麼?怎樣避免和解決ANR?
ANR:Application Not Responding,即應用無響應
ANR一般有三種類型:
1:KeyDispatchTimeout(5 seconds) –主要類型
按鍵或觸摸事件在特定時間內無響應

2:BroadcastTimeout(10 seconds)
BroadcastReceiver在特定時間內無法處理完成

3:ServiceTimeout(20 seconds) –小概率類型
Service在特定的時間內無法處理完成

超時的原因一般有兩種:
(1)當前的事件沒有機會得到處理(UI線程正在處理前一個事件沒有及時完成或者looper被某種原因阻塞住)
(2)當前的事件正在處理,但沒有及時完成

UI線程儘量只做跟UI相關的工作,耗時的工作(數據庫操作,I/O,連接網絡或者其他可能阻礙UI線程的操作)放入單獨的線程處理,儘量用Handler來處理UI thread和thread之間的交互。

UI線程主要包括如下:
Activity:onCreate(), onResume(), onDestroy(), onKeyDown(), onClick()
AsyncTask: onPreExecute(), onProgressUpdate(), onPostExecute(), onCancel()
Mainthread handler: handleMessage(), post(runnable r)
other

17.Android下解決滑動衝突的常見思路是什麼?
相關的滑動組件 重寫onInterceptTouchEvent,然後判斷根據xy值,來決定是否要攔截當前操作

18.如何把一個應用設置爲系統應用?
成爲系統應用,首先要在 對應設備的 Android 源碼 SDK 下編譯,編譯好之後:
此 Android 設備是 Debug 版本,並且已經 root,直接將此 apk 用 adb 工具 push 到 system/app 或 system/priv-app 下即可。
如果非 root 設備,需要編譯後重新燒寫設備鏡像即可。

有些權限(如 WRITE_SECURE_SETTINGS ),是不開放給第三方應用的,只能在對應設備源碼中編譯然後作爲系統 app 使用。

19、Android內存泄露研究
Android內存泄漏指的是進程中某些對象(垃圾對象)已經沒有使用價值了,但是它們卻可以直接或間接地引用到gc roots導致無法被GC回收。無用的對象佔據着內存空間,使得實際可使用內存變小,形象地說法就是內存泄漏了。
場景
類的靜態變量持有大數據對象
靜態變量長期維持到大數據對象的引用,阻止垃圾回收。
非靜態內部類的靜態實例
非靜態內部類會維持一個到外部類實例的引用,如果非靜態內部類的實例是靜態的,就會間接長期維持着外部類的引用,阻止被回收掉。
資源對象未關閉
資源性對象如Cursor、File、Socket,應該在使用後及時關閉。未在finally中關閉,會導致異常情況下資源對象未被釋放的隱患。
註冊對象未反註冊
未反註冊會導致觀察者列表裏維持着對象的引用,阻止垃圾回收。
Handler臨時性內存泄露
Handler通過發送Message與主線程交互,Message發出之後是存儲在MessageQueue中的,有些Message也不是馬上就被處理的。在Message中存在一個 target,是Handler的一個引用,如果Message在Queue中存在的時間越長,就會導致Handler無法被回收。如果Handler是非靜態的,則會導致Activity或者Service不會被回收。
由於AsyncTask內部也是Handler機制,同樣存在內存泄漏的風險。
此種內存泄露,一般是臨時性的。

20.內存泄露檢測有什麼好方法?
檢測:
1、DDMS Heap發現內存泄露
dataObject totalSize的大小,是否穩定在一個範圍內,如果操作程序,不斷增加,說明內存泄露
2、使用Heap Tool進行內存快照前後對比
BlankActivity手動觸發GC進行前後對比,對象是否被及時回收

定位:
1、MAT插件打開.hprof具體定位內存泄露:
查看histogram項,選中某一個對象,查看它的GC引用鏈,因爲存在GC引用鏈的,說明無法回收
2、AndroidStudio的Allocation Tracker:
觀測到期間的內存分配,哪些對象被創建,什麼時候創建,從而準確定位


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章