Android應用開發中如何使用隱藏的API
阿新 • • 發佈:2019-01-23
Android應用中有時為了實現某些特殊的功能或效果,往往需要使用一些谷歌隱藏的API(加有@hide標記的public類、方法或常量),例如PolicyManager。
使用Android隱藏的API主要有兩種辦法:1.利用Java反射機制,使用反射的方法得到隱藏API;2.使用原始碼編譯時生成的全編譯過的classes.jar包。
舉例說明,下面的程式碼如果直接使用會報錯。
mWindowManager = (WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE); mWindow = PolicyManager.makeNewWindow(mContext); mWindow.setWindowManager(mWindowManager, null, null);
1,使用Java反射機制,改為以下程式碼,就可以正常使用了。
2,如果不想修改程式碼,就需要使用第二種方法:匯入全編譯的classes.jar包。編譯Android框架原始碼後,將生成全編譯的classes.jar匯入到工程中。使用Eclipse,Android工程新增library(BuildPath -> Add Libraries->User Library->New User Library),將.jar檔案加入新增到library,同時勾選“SystemLibrary”選項,以避免產生“java.lang.OutOfMemoryError:Java Heap Space”錯誤。如果已經正確匯入了jar庫,卻仍然找不到隱藏的API。原因可能是Buildclass path order不正確,即android.jar和classes.jar的匯入順序不對,具體調節Buildclass path order,選擇Build Path-> Config Build Path->Order and Export,調整自定義的library與android.jar的順序。mWindowManager = (WindowManager) mContext .getSystemService(Context.WINDOW_SERVICE); // using reflect mechanism to invoke hide api String POLICYMANAGER_CLASS_NAME = "com.android.internal.policy.PolicyManager"; try { Class policyClass; policyClass = Class.forName(POLICYMANAGER_CLASS_NAME); Log.i(TAG, policyClass.toString()); Method meths[] = policyClass.getMethods(); Method makenewwindow = null; // Method makenewwindow = policyClass.getMethod("makeNewWindow"); for (int i = 0; i < meths.length; i++) { if (meths[i].getName().endsWith("makeNewWindow")) makenewwindow = meths[i]; } Log.i(TAG, makenewwindow.toString()); mWindow = (Window) makenewwindow.invoke(null, mContext); } catch (Exception e) { e.printStackTrace(); } mWindow.setWindowManager(mWindowManager, null, null);
此時,本文最開始的那段程式碼不需要再做改動就可以生效了。
利用反射機制使用隱藏API的方法優點是靈活,能夠在不相容的系統捕獲異常,而使程式不至於崩潰;缺點是過程太繁雜,而且對於隱藏類的繼承等很難實現。如果需要大量使用隱藏API,使用反射無疑會讓人厭煩。
通過匯入全編譯的classes.jar包來實現對隱藏API的使用方法簡單,容易程式設計,就像是那些隱藏的API在SDK中變得可見了一樣,其缺點就是相容性問題,希望使用者引起重視。
兩種方法各有利弊,請大家斟酌使用。一般來說,谷歌設定隱藏API的主要原因在於Android系統本身還在不斷的進化發展中,這些API可能還不夠成熟、穩定。因此除非特殊情況,應儘量減少對隱藏API的使用。