android中的java
『壹』 如何在Android上編寫高效的java代碼
Java平台一般有三個版本:Java ME(微型版,用於某些手機)、Java SE(標准版,用於台式電腦)、Java EE(企業版,用於伺服器端應用)。在談到Java時,我們通常是指Java SE,因為只有這個版本包含虛擬機和編譯器。
首先,Java代碼會被編譯成稱為位元組碼的中間格式。當位元組碼在目標電腦上運行時,虛擬機會快速將它解析成目標電腦硬體和操作系統所需要的本機格式。
除了為開發者提供「一次編寫,到處運行」的優勢,Java還能通過垃圾回收器(GC)實現自動內存管理,開發者可免去手動在代碼中釋放無用對象的內存。雖然這個功能非常有用,且大大降低了在代碼中引入內存問題的風險,但是它會增加運行時的開銷,因為需要不停地執行垃圾回收進程。
本文開頭將比較Java SE和用於Android開發的Java之間的差異。首先我會介紹開發者習慣的Java
SE語言結構以及它們是如何在Android上運行的。其次,我會介紹如何優化Android中的Java代碼,如何優化內存分配,以及如何恰當地處理多線程。
比較Android上的Dalvik Java和Java SE
雖然遠在Android出現之前,開發者就能用Java編程語言為移動設備編寫應用程序,但它只是Java中功能極為有限的一個版本,稱為Java
ME(微型版)。不同的移動設備還需編寫不同的代碼,因此,寫一個應用程序就能在支持Java
ME的任何手機上運行是幾乎不可能的。此外,由於當時不存在很好的在線商店,應用發布過程極其復雜。
Android的問世為開發者提供了構建智能手機強大應用的機會,開發者只需用Java編程語言以及他們熟知的標准Java
API編寫代碼。然而,盡管Android開發者仍使用Java SE編譯器來編譯應用程序,你會發現,James
Gosling開發的Java和Android設備上的Java存在許多不同之處。
在Android設備上運行的VM(虛擬機)稱為Dalvik。它最初由谷歌的Dan
Bornstein開發,適用於CPU和內存受限的移動設備。Java SE和Dalvik Java存在一些差異,主要體現在虛擬機上。Java
SE使用了棧機設計,而Dalvik被設計成了基於寄存器的機器。Android SDK中有一個dx工具,它會把Java
SE棧機器的位元組碼轉換成基於寄存器的Dalvik機器位元組碼,該轉換步驟由IDE自動完成。
基於棧的虛擬機和基於寄存器的虛擬機的定義以及差異將不列入我們的討論范圍。由於歷史原因,Android使用基於寄存器的虛擬機。雖然基於寄存器的虛擬機最多可以比基於棧的虛擬機快32%,但這只限於執行時解釋位元組碼的虛擬機(也就是說,解釋型虛擬機)。在Android
2.2版本(也稱為Froyo)之前,Dalvik虛擬機都是純解釋型的。Froyo版本引入了JIT編譯器(即時編譯),這是Java
SE很早就有的一個優勢。
JIT編譯,也稱為動態翻譯。它在執行前把位元組碼翻譯成本機代碼(如圖1所示),這樣主要有兩個好處。首先,它消除了那些純解釋型虛擬機的開銷;其次,它能對本機代碼執行優化,這通常是靜態編譯代碼無法做到的。例如,JIT編譯器可以在它運行的CPU上選擇最合適的優化,也可以根據應用程序的輸入來分析代碼是如何運行的,以便進行下一步的優化。
圖1Android Java和Java SE翻譯步驟
雖然Android的Dalvik JIT編譯器有很大的發展前景,但要達到如Java SE的JIT編譯器般穩定、成熟度尚需很長一段時間。不過,Dalvik JIT的出現為Android提供了巨大的性能優勢,而且它也在不斷得以改善。
JAVA
SE虛擬機和Dalvik虛擬機的另一個區別是,後者進行了優化,可運行在同一個機器上的多個實例中。它在開機時會啟動一個叫做zygote的進程,該進程會創建第一個Dalvik實例,由這個實例創建所有其他的實例。當應用程序啟動時,zygote進程會收到一個創建新虛擬機實例的請求,並給該應用程序創建一個新進程(如圖2所示)。如果開發者已習慣於Java
SE開發,這樣的設計可能看起來不切實際,但它有一個很大的優勢,可以避免由一個應用程序運行失敗導致Dalvik虛擬機崩潰,繼而引發多應用程序崩潰。
圖2在Android中啟動新Dalvik虛擬機實例
Android和Java
SE除了運行的虛擬機不同之外,它們實現API的方式也不一樣。Android中屬於java和javax包中的API都來自Apache
Harmony(這是一個開源項目,旨在重新實現Java SE軟體棧,該項目從2011年11月不再維護)。在開發方面,這些API和Java
SE包中的類似,但也存在一些差別。例如,谷歌對HttpUrlConnection類進行了Java SE版本中所沒有的重大升級。
此外,Android平台移除了Java
SE中無關的API。例如,Swing/AWT包被完全移除,因為Android使用不同的UI框架。其他被移除的API還有RMI、CORBA、ImageIO和JMX。它們或者被替換為特定的Android版本(在android包空間內),或者因為一些實際原因根本不存在。
優化Android上的Java代碼
經過多年的改進,Java
SE具備了一些簡化編寫復雜代碼結構的新特性。其中的一些特性會讓整個流程變得更簡單,但開發者需要了解何時以及如何正確地使用它們。另外,由於Java
SE大多用於伺服器端開發(使用Java企業版的API),因而開發人員專門對伺服器端Java代碼進行了優化。註解和Java虛擬機對腳本語言的支持就是對伺服器端開發進行優化的例證。雖然這些工具在構建後端開發時很強大,但在開發Android客戶端代碼時,這些特性的作用很小,甚至起反作用。Java開發者已經習慣於無限量的RAM和CPU,而Android開發需要密切關注性能和內存分配。簡單地說,開發者需要使用稍微不同的方法對待Android和後端的開發。
然而,隨著Android的首次發布,情況有所改變。曾經一些在Android上盡量不用的Java規范重新被推薦,這主要因為Android目前的JIT編譯器解決了這些規范導致的性能問題。
本文將討論編寫Android應用程序需要了解的Java代碼。我們不會深究Java編程語言的細節,而是重點關注對Android開發重要的東西。不過,開發者仍需了解,大多數適用於Java SE的規則和建議同樣適用於Android和Dalvik虛擬機。
Android上的類型安全枚舉
Java SE 5.0新增了許多方便開發者的新特性。其中最值得期待的是引入了類型安全枚舉。枚舉在代碼中用來表示屬於某一組的幾個選擇。在早期版本的Java中,可以用多個整型常量解決這個問題。雖然這在技術上可行,但是很容易出錯。請看下面的代碼:
public class Machine {
public static final int STOPPED = 10;
public static final int INITIALIZING = 20;
public static final int STARTING = 30;
public static final int RUNNING = 40;
public static final int STOPPING = 50;
public static final int CRASHED = 60;
private int mState;
public Machine() {
mState = STOPPED;
}
public int getState() {
return mState;
}
public void setState(int state) {
mState = state;
}
}
問題是,雖然這些常量是期望的,但是沒有機制保證setState()方法接收不同的值。如果要在設置方法中添加檢查,那麼一旦得到的是非預期值,開發者就需要處理錯誤。開發者所需要的是在編譯時檢查非法賦值。類型安全的枚舉解決了這個問題,如下所示:
public class Machine {
public enum State {
STOPPED, INITIALIZING, STARTING, RUNNING, STOPPING, CRASHED
}
private State mState;
public Machine() {
mState = State.STOPPED;
}
public State getState() {
return mState;
}
public void setState(State state) {
mState = state;
}
}
注意在聲明不同類型安全值的地方新加的內部枚舉類。這在編譯時就會解決非法賦值的問題,所以代碼更不容易出錯。
如果Dalvik虛擬機還沒有JIT編譯器優化代碼,不建議在Android平台上使用枚舉類型,因為和使用整型常量相比,這種設計帶來的內存和性能損失更大。這就是為什麼在一些老版本的Android
API中還存在如此多的整型常量的原因。如今有了更強的JIT編譯器以及一個不斷改進的Dalvik虛擬機,開發者不必再擔心這個問題,放心大膽地使用類型安全枚舉即可。
然而,仍然存在一些情況使用整型常量是更好的選擇。像int這樣的Java基本類型,不會增加GC的開銷。此外,Android SDK中許多已有的API仍然依賴基本類型,比如Handler類——在這種情況下,你沒有太多的選擇。
Android中增強版的for循環
Java SE 5.0還引入了增強版的for循環,提供了一個通用的縮寫表達式來遍歷集合和數組。首先,比較以下五種方法:
void loopOne(String[] names) {
int size = names.length;
for (int i = 0; i < size; i++) {
printName(names[i]);
}
}
void loopTwo(String[] names) {
for (String name : names) {
printName(name);
}
}
void loopThree(Collection<String> names) {
for (String name : names) {
printName(name);
}
}
void loopFour(Collection<String> names) {
Iterator<String> iterator = names.iterator();
while (iterator.hasNext()) {
printName(iterator.next());
}
}
// 不要在ArrayList上使用增強版的for循環
void loopFive(ArrayList<String> names) {
int size = names.size();
for (int i = 0; i < size; i++) {
printName(names.get(i));
}
}
上面顯示了四種不同遍歷集合和數組的方式。前面兩種有著相同的性能,所以如果只是讀取元素的話,可以放心地對數組使用增強版for循環。對Collection對象來說,增強版for循環和使用迭代器遍歷元素有著相同的性能。ArrayList對象應避免使用增強版for循環。
如果不僅需要遍歷元素,而且需要元素的位置,就一定要使用數組或者ArrayList,因為所有其他Collection類在這些情況下會更慢。
一般情況下,如果在讀取元素幾乎不變的數據集時對性能要求很高,建議使用常規數組。然而,數組的大小固定,添加數據會影響性能,所以編寫代碼時要考慮所有因素。
隊列、同步和鎖
通常情況下,應用程序會在一個線程中生產數據,在另一個線程中使用它們。常見的例子是在一個線程中獲取網路上的數據,在另一個線程(操作UI的主線程)中把這些數據展現給用戶。這種模式稱為生產者/消費者模式,在面向對象編程課程中,開發者用演算法來實現該模式可能要花上幾個小時。下面會介紹一些簡化生產者/消費者模式實現的現成類。
1. 更智能的隊列
雖然已有現成的類並能用更少的代碼實現該功能,但許多Java開發者仍然選擇使用LinkedList以及同步塊實現隊列功能。開發者可在java.util.concurrent包中找到同步相關的類。此外,本包還包含信號量、鎖以及對單個變數進行原子操作的類。考慮下面使用標準的LinkedList實現線程安全隊列的代碼。
public class ThreadSafeQueue {
private LinkedList<String> mList = new LinkedList<String>();
private final Object mLock = new Object();
public void offer(String value) {
synchronized (mLock) {
mList.offer(value);
mLock.notifyAll();
}
}
public synchronized String poll() {
synchronized (mLock) {
while (mList.isEmpty()) {
try {
mLock.wait();
} catch (InterruptedException e) {
//簡潔起見忽略異常處理
}
}
return mList.poll();
}
}
}
雖然這段代碼是正確的,並有可能在考試中得滿分,但實現和測試這樣一段代碼只是在浪費時間。實際上,所有前面的代碼可用下面一行代替。
LinkedBlockingQueue<String> blockingQueue =
new LinkedBlockingQueue<String>();
上面的一行代碼能像前面的例子一樣提供相同類型的阻塞隊列,甚至能提供額外的線程安全操作。java.util.concurrent包含許多可選的隊列以及並發映射類,所以,一般情況下,建議使用它們,而不是像之前的示例那樣使用更多代碼。
2. 更智能的鎖
Java提供的synchronized關鍵字允許開發者創建線程安全的方法和代碼塊。synchronized關鍵字易於使用,也很容易濫用,對性能造成負面影響。當需要區分讀數據和寫數據時,synchronized關鍵字並不是最有效的。幸好,java.util.concurrent.locks包中的工具類對這種情況提供了很好的支持。
public class ReadWriteLockDemo {
private final ReentrantReadWriteLock mLock;
private String mName;
private int mAge;
private String mAddress;
public ReadWriteLockDemo() {
mLock = new ReentrantReadWriteLock();
}
public void setPersonData(String name, int age, String address) {
ReentrantReadWriteLock.WriteLock writeLock = mLock.writeLock();
try {
writeLock.lock();
mName = name;
mAge = age;
mAddress = address;
} finally {
writeLock.unlock();
}
}
public String getName() {
ReentrantReadWriteLock.ReadLock readLock = mLock.readLock();
try {
readLock.lock();
return mName;
} finally {
readLock.unlock();
}
}
// 重復代碼不再贅述
}
上面的代碼展示了在什麼地方使用ReentrantReadWriteLock,它允許多個並發線程對數據進行只讀訪問,並確保同一時間只有一個線程寫入相同的數據。
在代碼中使用synchronized關鍵字仍然是處理鎖問題的有效方法,但無論何種情況下,都要考慮ReentrantReadWriteLock是否是
『貳』 如何在Android中啟動JAVA程序
在Android中啟動JAVA程序其實有很多種方式,現總結如下
一、在Android應用程序中發送Intent啟動Android應用程序
這個方式最簡單,最常用。在此不在累述。關於Intent的更多內容請閱讀《Intent技術簡介》
二、在shell控制台通過am命令發送Intent來啟動Android應用程序
在Android的shell控制台通過am命令發送Intent來啟動Android應用程序
關於此的詳細內容請參考《Android命令am詳解》
三、在shell控制台直接通過davlikvm命令啟動一個JAVA程序。
該方式有個天生的缺點,即在其中,很多Android的JNI無法調用。因為Android的很多JNI其實是需要手動注冊的。
關於請參考《基本Dalvik VM調用》
四、在shell控制台直接通過運行app_process程序啟動一個JAVA程序
在app_process程序中,他會對Android的JNI進行手動注冊的,能很好的使用Android的API,因此通過運行app_process程序啟動一個JAVA程序,是一個比較完美的方式。app_process程序是一個C程序,它的源碼位於frameworks\base\cmds\app_process。
關於它的使用請參考《Android命令am詳解》以及shell腳本frameworks\base\cmds\am\am和frameworks\base\cmds\pm\pm
am腳本文件如下:
# Script to start "am" on the device, which has a very rudimentary# shell.#base=/systemexport CLASSPATH=$base/framework/am.jarexec app_process $base/bin com.android.commands.am.Am "$@"pm腳本文件如下:
# Script to start "pm" on the device, which has a very rudimentary# shell.#base=/systemexport CLASSPATH=$base/framework/pm.jarexec app_process $base/bin com.android.commands.pm.Pm "$@"CLASSPATH指定了你的程序的位置,com.android.commands.pm.Pm則說明了程序的入口為com.android.commands.pm.Pm,即入口函數main()所在的類,"$@"就是傳遞給main()函數的參數,只是這里"$@"本身又是個shell傳入的參數而已
需要注意的是CLASSPATH中的文件必須是dalvik文件格式的,關於此的轉換請參考《基本Dalvik VM調用》當然CLASSPATH中的文件可以是apk文件,只是你的apk中至少應該有個擁有main()入口函數的類。
轉載
『叄』 Android開發和JAVA開發有什麼區別
Android開發是java開發的一個分支,android種的類庫有一部分和java類庫基本一致,也就是類名一致,不過java種的swing這些類庫在android種就沒有了,也就是說android種只用了java種的常用的類庫。
『肆』 android和java的區別
android Java虛擬機和sun java虛擬機的主要區別體現在因為手機內存和硬體的限制,不可能直接將sun java虛擬機的機制搬過去,對於一些位元組、內存管理方面需要重新設計,一下是主要區別:
1、Dalvik 和標准 Java 虛擬機(JVM)
Dalvik 基於寄存器,而 JVM 基於棧。基於寄存器的虛擬機對於更大的程序來說,在它們編譯的時候,花費的時間更短。 JVM位元組碼中,局部變數會被放入局部變數表中,繼而被壓入堆棧供操作碼進行運算,當然JVM也可以只使用堆棧而不顯式地將局部變數存入變數表中。Dalvik位元組碼中,局部變數會被賦給65536個可用的寄存器中的任何一個,Dalvik指令直接操作這些寄存器,而不是訪問堆棧中的元素。
2、Dalvik 和 Java 位元組碼的區別
VM位元組碼由.class文件組成,每個文件一個class。JVM在運行的時候為每一個類裝載位元組碼。相反的,Dalvik程序只包含一個.dex文件,這個文件包含了程序中所有的類。Java編譯器創建了JVM位元組碼之後,Dalvik的dx編譯器刪除.class文件,重新把它們編譯成Dalvik位元組碼,然後把它們寫進一個.dex文件中。這個過程包括翻譯、重構、解釋程序的基本元素(常量池、類定義、數據段)。常量池描述了所有的常量,包括引用、方法名、數值常量等。類定義包括了訪問標志、類名等基本信息。數據段中包含各種被VM執行的函數代碼以及類和函數的相關信息(例如DVM所需要的寄存器數量、局部變數表、操作數堆棧大小),還有實例變數。
3、Dalvik 和 Java 運行環境的區別
Dalvik 經過優化,允許在有限的內存中同時運行多個虛擬機的實例,並且每一個Dalvik 應用作為一個獨立的Linux 進程執行。獨立的進程可以防止在虛擬機崩潰的時候所有程序都被關閉。
Dalvik虛擬機在android2.2之後使用JIT (Just-In-Time)技術,與傳統JVM的JIT並不完全相同,
Dalvik虛擬機有自己的 bytecode,並非使用 Java bytecode。
還有以下幾點:
1、Dalvik主要是完成對象生命周期管理,堆棧管理,線程管理,安全和異常管理,以及垃圾回收等等重要功能。
2、Dalvik負責進程隔離和線程管理,每一個Android應用在底層都會對應一個獨立的Dalvik虛擬機實例,其代碼在虛擬機的解釋下得以執行。
3、不同於Java虛擬機運行java位元組碼,Dalvik虛擬機運行的是其專有的文件格式Dex。
不過有一點是習慣了java語言開發的程序員,在android平台上同樣可以接著使用java的全部語法,只不過新增了很多知識點。
『伍』 安卓程序是用什麼語言編寫的
Android以Java為編程語言,使介面到功能,都有層出不窮的變化,其中Activity等同於J2ME的MIDlet,一個 Activity 類(class)負責創建視窗(window),一個活動中的Activity就是在 foreground(前景)模式,背景運行的程序叫做Service。
兩者之間通過由ServiceConnection和AIDL連結,達到復數程序同時運行的效果。如果運行中的 Activity 全部畫面被其他 Activity 取代時,該 Activity 便被停止(stopped),甚至被系統清除(kill)。
(5)android中的java擴展閱讀
Android應用程序是android系統智能手機的主要構成部分,實現了智能手機的多樣性、多功能性,結合了辦公功能、娛樂功能、生活實用功能等,廣受人們的喜愛。
Android應用程序有很多,其中讓人耳目一新的個性化實用性應用程序有:
MINT、CHROME、神奇阿萊西、SWYPE/SWiftKey鍵盤、SWYPE/SWiftKey鍵盤、Swype、POCKET閱讀器、SNAPSEED、EYE IN THE SKY天氣、FLICK NOTE、LIGHT FLOW、ANY.DO。
『陸』 在Android上怎樣實現JAVA和JS交互
Android中java與js交互是通過webView來交互的。
WebView(網路視圖)能載入顯示網頁,可以將其視為一個瀏覽器。它使用了WebKit渲染引擎載入顯示網頁,實現WebView有以下兩種不同的方法:
第一種方法的步驟:
1.在要Activity中實例化WebView組件:WebView webView = new WebView(this);
2.調用WebView的loadUrl()方法,設置WevView要顯示的網頁:
互聯網用:webView.loadUrl("http://www.google.com");
本地文件用:webView.loadUrl("file:///android_asset/XX.html"); 本地文件存放在:assets 文件中
3.調用Activity的setContentView( )方法來顯示網頁視圖
4.用WebView點鏈接看了很多頁以後為了讓WebView支持回退功能,需要覆蓋覆蓋Activity類的onKeyDown()方法,如果不做任何處理,點擊系統回退剪鍵,整個瀏覽器會調用finish()而結束自身,而不是回退到上一頁面
5.需要在AndroidManifest.xml文件中添加許可權,否則會出現Web page not available錯誤。
<uses-permission android:name="android.permission.INTERNET" />
第二種方法的步驟:
1、在布局文件中聲明WebView
2、在Activity中實例化WebView
3、調用WebView的loadUrl( )方法,設置WevView要顯示的網頁
4、為了讓WebView能夠響應超鏈接功能,調用setWebViewClient( )方法,設置 WebView視圖
5、用WebView點鏈接看了很多頁以後為了讓WebView支持回退功能,需要覆蓋覆蓋Activity類的onKeyDown()方法,如果不做任何處理,點擊系統回退剪鍵,整個瀏覽器會調用finish()而結束自身,而不是回退到上一頁面
6、需要在AndroidManifest.xml文件中添加許可權,否則出現Web page not available錯誤。
<uses-permission android:name="android.permission.INTERNET"/>