一. Android系統版本及詳細資訊
最新資料 https://developer.android.com/about/dashboards/
二. Android專案初探
1. 使用android studio創建一個工程
Application Name:“My First App”
Company Domain:“example.com”
Project和module聯系和區別
一個 Project 可以有多個 Module,目前主流的大型專案是多Module結構的,模塊之間彼此可以相互依賴,他們之間應該都是處于同一個專案業務情況下的模塊,彼此之間是有不可分割的業務關系的,
Android studio中,一個Project代表一個完整的APP,Module表示APP中的一些依賴庫或獨立開發的模塊,比如可以新建一個library做為module,然后在主APP上點右鍵 open module setting的Dependencies中添加一個模塊依賴,然后主APP中就可以使用module中的類了,
2. 目錄結構
Android視圖
Project視圖
關于gradle.build檔案參考: https://developer.android.google.cn/studio/build/index.html
3. Logcat的使用
日志列印工具
|
方法 |
級別 |
|
Log.v() |
verbose |
|
Log.d() |
debug |
|
Log.i() |
info |
|
Log.w() |
warn |
|
Log.e() |
error |
注意:在android studio中創建一個activity 為我們完成了三步:(繼承自AppCompatActivity是為了向后兼容)
創建activity、在menifest中注冊activity、創建activity對應的layout檔案
4. 隱藏標題欄
方法一:
繼承自AppCompatActivity:
if (getSupportActionBar() != null){
getSupportActionBar().hide();
}
方法二: 在style中加入如下代碼: <item name="windowNoTitle">true</item>
5. 在活動中使用Toast
Toast.makeText(this,"you clicked",Toast.LENGTH_SHORT).show();
6. 在活動中添加選單
(1)添加選單xml檔案:
<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item
android:id="@+id/add_item"
android:title="add"/>
<item
android:id="@+id/delete_item"
android:title="delete"/>
</menu>
(2)重寫onCreateOptionsMenu方法
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.main,menu);
return true; //回傳true表示顯示該選單
}
(3)重寫onOptionsItemSelected
public boolean onOptionsItemSelected(MenuItem item){
switch(item.getItemId()){
case R.id.add_item:
Toast.makeText(this,"add clicked",Toast.LENGTH_SHORT).show();
break;
case R.id.delete_item:
Toast.makeText(this,"delete clicked",Toast.LENGTH_SHORT).show();
break;
}
return true;
}
更多詳情參見考:https://developer.android.google.cn/guide/topics/ui/menus
7. 銷毀活動
finish(); //第一講到此+java預備知識
三. Android事件處理
Android提供了兩套事件處理機制:監聽、回呼
1. 基于監聽的事件處理
事件監聽的處理模型中,主要涉及三類物件:
Event Source(事件源) 通常是各個組件,如按鈕、視窗、選單等;
Event(事件):事件封裝了界面組件上發生的特定事情,
Event Listener(事件監聽器):負責監聽事件源所發生的事件,并對各種事件作出相應的相應,
事件監聽涉及的三個問題:
事件源: 任何組件都可作為事件源,
事件監聽器:由程式員實作,關鍵是實作處理方法,
注冊監聽器:呼叫事件源的setXxxListener方法即可,
另外:對于鍵盤事件、觸摸屏事件等,此時程式需要獲取事件發生的詳細資訊,如,鍵盤事件獲取是哪個鍵觸發的,觸摸屏事件需要獲取事件發生的位置的,Android會將事件資訊封裝成XxxEvent物件,并將該物件作為引數傳遞給事件處理器,
所謂事件監聽器,其實就是實作了特定介面的java類的實體,實作事件監聽器,通常有如下幾種形式:
l 內部類
l 匿名內部類
l 外部類
l Activity本身作為事件監聽器
l 直接系結到標簽
其他事件如下表所示:
|
|
setOnClickListener Register a callback to be invoked when this view is clicked. |
|
|
setOnContextClickListener Register a callback to be invoked when this view is context clicked. |
|
|
setOnCreateContextMenuListener Register a callback to be invoked when the context menu for this view is being built. |
|
|
setOnDragListener Register a drag event listener callback object for this View. |
|
|
setOnFocusChangeListener Register a callback to be invoked when focus of this view changed. |
|
|
setOnGenericMotionListener Register a callback to be invoked when a generic motion event is sent to this view. |
|
|
setOnHoverListener Register a callback to be invoked when a hover event is sent to this view. |
|
|
setOnKeyListener Register a callback to be invoked when a hardware key is pressed in this view. |
|
|
setOnLongClickListener Register a callback to be invoked when this view is clicked and held. |
|
|
setOnScrollChangeListener Register a callback to be invoked when the scroll X or Y positions of this view change. |
|
|
setOnTouchListener Register a callback to be invoked when a touch event is sent to this view. |
2. 基于回呼的事件處理
當用戶在GUI組件上激發某事件時,組件自己的特定方法負責處理該事件,方法是自定義控制元件使其繼承GUI組件類,在自定義控制元件中重寫事件處理方法,
Demo
l 基于回呼的事件傳播
幾乎所有基于回呼的事件處理方法都有一個boolean型別的回傳值,用于標志該處理方法是否已完全處理該事件,
回傳true,表明已處理完,不會傳播出去,
回傳false,表明未處理完,會傳播出去,
幾種方法呼叫的先后順序: 組件的監聽器à組件自身的回呼方法à組件所在Activity的回呼方法,其中任何一個事件處理方法回傳了true,那么該事件將不會繼續向外傳播,
四. 啟動活動的方法:
1. 顯式Intent demo
//第一個引數Context是啟動活動的背景關系,第二個引數是制定要啟動的目標活動,
Intent intent=new Intent(MainActivity.this,SecondActivity.class);
startActivity(intent);
2. 隱式Intent demo
不明確指定啟動哪一個活動,而是指定一系列action 和category,由系統分析啟動哪一個活動,只有<action>和<category>中的內容同時能匹配上Intent中指定的action和category時,這個活動才能回應該Intent,
每個Intent只能指定一個action,但可以指定多個category,如果沒有匹配的活動可以啟動程式將崩潰;如果有多個匹配的活動,系統將提供串列讓用戶選擇啟動哪個活動,
顯示參考不能啟動其他行程的Activity物件,因為無法獲取其他行程的Activity物件的位元組碼,而隱式啟動則可以通過配置Intent Filter啟動其他行程的Activity物件,因此在應用內,我們一般都是使用顯示啟動的方式啟動Activity,而如果需要啟動其他應用的Activity時,一般使用隱式啟動的方式,
l 更多隱式Intent的用法
呼叫系統的瀏覽器打開一個網頁,
Intent intent=new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com"));
Uri.parse() 將字串決議成一個Uri物件,setData() 方法接收一個Uri物件,指定當前Intent正在操作的資料,
只有<data>標簽中指定的內容和Intent中攜帶的data完全一致時,當前活動才能相應該Intent.
<data> 使用一個或多個指定資料 URI 各個方面(scheme、host、port、path 等)和 MIME 型別的屬性,宣告接受的資料型別,
https://developer.android.google.cn/guide/components/intents-common#Browser
五. 活動之間傳遞資料
1. 向下一個活動傳遞資料
Intent intent=new Intent(MainActivity.this,SecondActivity.class);
intent.putExtra("extraData_Key","I am data");
startActivity(intent);
在下一個活動中將資料取出:
Intent intent=getIntent();
String data=https://www.cnblogs.com/duhuamei/p/intent.getStringExtra("extraData_Key");
使用Bundle
兩個Activity交換資料通過Intent完成,只需將交換的資料放入Intent中即可,
putExtra(String key, Xxx value) 向Intent中按key-value對的形式存入資料
Xxx getXxxExtra(String key) 從Intent中按key取出指定型別的資料
putExtras(Bundle data) 向Intent中存入攜帶資料的Bundle
Bundle getExtras() 從Intent中取出攜帶資料的Bundle
向Bundle物件中存取資料:
putXxx(String key, Xxx data) 向Bundle中放入Xxx型別的資料
Xxx getXxx(String key) 從Bundle中取出Xxx型別的資料
putSerializable(String key, Serializable data) 向Bundle中放入一個可序列化的物件
Serializable getSerializable(String key) 從Bundle中取出可序列化物件
2. 回傳資料給上一個活動
1) 啟動活動時使用onActivityResult方法:
startActivityForResult(intent,1);
2) 重寫onActivityResult()方法,對回傳的資料進行處理:
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data)
{
switch (requestCode){
case 1:
if(resultCode==RESULT_OK) {
String returnedData=https://www.cnblogs.com/duhuamei/p/data.getStringExtra("data_return");
Log.d(TAG, "onActivityResult: "+returnedData);
}
break;
}
}
3) 在第二個活動中回傳資料:
setResult(RESULT_OK,intent);
finish();
六. 活動的生命周期
1. 回傳堆疊(Back Stack)
每啟動新的活動,就會覆寫在原來的活動上,按Back鍵會銷毀上面的活動,下面的活動就會重新顯示出來,
Android使用Task來管理活動,一個Task就是一組存放在堆疊里的活動的集合,這個堆疊被稱為回傳堆疊(Back Stack),參見activity回傳堆疊,
2. 活動狀態
l 運行狀態
l 暫停狀態
l 停止狀態
l 銷毀狀態
3. 活動的生存期
4. 體驗活動的生命周期 (Demo: ActivityLifeCycle)
5. 活動被回收了怎么辦
由于記憶體不足等原因,處于停止狀態的活動有可能被系統回收,當該活動重新可見時,該活動會被重新創建,
onSaveInstanceState()方法在活動被回收之前呼叫,解決活動被回收時保存臨時資料的問題,
1) 活動回收前保存資料
protected void onSaveInstanceState(Bundle outState) {
outState.putString("data_key","I am data"); //保存資料
super.onSaveInstanceState(outState);//必須呼叫父類的方法
}
2) 活動恢復時恢復資料
在onCreate方法中恢復:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); // Always call the superclass first
if (savedInstanceState != null) {
String str=savedInstanceState.getString("data_key");//得到資料
Log.d("MainActivity", "活動恢復,資料為:"+str);
} else {
Log.d("MainActivity", "這是活動正常啟動");
}
//……
}
在onRestoreInstanceState方法中恢復:
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);//可選
String str=savedInstanceState.getString("data_key");//得到資料
Log.d("MainActivity", "恢復資料為:"+str);
}
注意:Always call the superclass implementation of onRestoreInstanceState() so the default implementation can restore the state of the view hierarchy.
開發者可根據具體情況選擇在哪個方法中進行恢復,
3) onSaveInstanceState() 何時被執行
當某個activity變得“容易”被系統回收時,該activity的onSaveInstanceState就會被執行,除非該activity是被用戶主動銷毀的,例如當用戶按BACK鍵(手機下方的回傳鍵)的時候,(why)分為以下幾種情況:
l 從第一個界面跳轉到第二個界面,第一個界面就會執行onSaveInstanceState
l 按下home鍵,運行多個其他程式,這時系統不確定會不會將該activity銷毀,所以會執行onSaveInstanceState方法保存值,
l 關閉手機螢屏時
l 螢屏方向切換時,例如從豎屏切換到橫屏時,(前提是androidMenifest.xml中對應activity標簽沒有配置<activity android:name=".MainActivity" android:configChanges="orientation|keyboardHidden|screenSize"></activity>)
4) onRestoreInstanceState() 何時被執行
Activity被系統回收后,當重新恢復時會呼叫onRestoreInstanceState(),并且是在onStart()之后,The
system calls onRestoreInstanceState() only if there is a saved
state to restore, so you do not need to check whether the Bundle is
null,
而系統創建一個新的Activity實體或重新創建一個以前的實體時都會呼叫onCreate(),因此在onCreate()中需要判斷Bundle是否為null,
5) 模擬當前應用被系統回收
進入開發者選項,選中不保留活動,
6) 為什么我們的UI界面的值不用我們自己保存也可以自動保存,狀態恢復呢?
開發者只需要為這些控制元件指定一個唯一的ID,剩余的事情就可以自動完成了,如果沒有為控制元件指定ID,則這個控制元件就不會進行自動的資料保存和恢復操作,
更多參考:https://developer.android.google.cn/guide/components/activities/activity-lifecycle.html#saras
另外,onSavedInstanceState()只適合保存少量可序列化的資料,use
a combination of ViewModel objects, the onSaveInstanceState() method,
and/or local storage to
persist the UI state across such application and activity instance transitions.
6. 活動的加載模式(Launch Mode)
Often, the way Android manages tasks and the back stack by placing all activities started in succession in the same task and in a "last in, first out" stack.
You can
interrupt the normal behavior, with attributes in the <activity> manifest
element and with flags in the intent that you pass to startActivity().
Launch modes allow you to define how a new instance of an activity is associated with the current task.
<activity android:name=".SecondActivity" android:launchMode="singleTop"/>
l standard (the default mode)
無論當前堆疊頂是哪一個Activity都會開始一個新的Activity,
l singleTop
如果需要創建的Activity已經處于任務(Task)堆疊頂時,復用該Activity,
l singleTask
需要創建的Activity已經處于任務(Task)堆疊時,彈出此Activity上的所有其他Activity,復用該Activity,
l singleInstance
啟動一個新的任務(Task)堆疊,且該任務堆疊中只有這唯一一個Activity,整個系統中創建了該Activity,將不再重新創建,
當從回傳堆疊回退時,如果當前回退堆疊為空了,才會顯示另一回退堆疊的活動,
Demo四種啟動模式演示
onNewIntent方法的使用
除了standard模式,其他三種方式都可能存在復用Activity的情況,通過Intent啟到一個Activity,如果系統已經存在一個實體,系統就會將請求發送到這個實體上,但新的Intent請求可能要處理新的資料,而這時沒有機會呼叫onCreate方法,怎么辦呢?系統為我們準備了onNewIntent方法,
一般,通過在onCreate和onNewIntent方法中呼叫同一個處理資料的方法,使得不管是創建新的Activity實體,還是復用原來的Activity實體,處理資料的方式保持一致,如下所示:
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
processExtraData();
}
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
setIntent(intent);//must store the new intent unless getIntent() will return the old one
processExtraData();
}
private void processExtraData(){
Intent intent = getIntent();
//use the data received here
}
轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/51797.html
標籤:Android
