編輯:關於Android編程
Android在3.0中引入了Fragments的概念,其目的是用在大屏幕設備上–例如平板電腦上,支持更加動態和靈活的UI設計。平板電腦的屏幕要比手機的大得多,有更多的空間來放更多的UI組件,並且這些組件之間會產生更多的交互。Fragment允許這樣的一種設計,而不需要你親自來管理 Viewhierarchy的復雜變化。 通過將Activity的布局分散到Fragment中, 你可以在運行時修改Activity的外觀,並在由Activity管理的back stack中保存那些變化。
Android開發過程中,我們可以把Fragment想成Activity中的模塊,這個模塊有自己的布局,有自己的生命周期,單獨處理自己的輸入,在Activity運行的時候可以加載或者移除Fragment模塊。從一定程度上來說,與Web開發中的iframe和DIV布局思想有點類似。例像網易新聞、易車等應用程序將一個fragment放在左邊顯示文章列表,在右邊用另一個Fragment來顯示一篇文章——兩個Fragment在同一個Activity中並排著,並且每個Fragment都有其自己的生命周期回調方法序列用以處理各自的用戶輸入事件。因此,用戶可以在同一個Activity中選擇和閱讀文章,而不是在一個Activity中選擇,在另一個Activity中閱讀。這樣設計的好處就是當程序運行在平板尺寸屏幕設備上時,可以在Activity A中嵌入兩個Fragment。但是,當運行在手機尺寸屏幕上,就沒有足夠的空間容納兩個Fragment了,因此Activity A只能引用包含文章列表的Fragment,在當用戶選擇一篇文章時,可以啟動Activity B,它包含了用來閱讀文章的第二個fragment。這樣,應用程序通過以不同組合的復用fragments支持了平板電腦和手機,再比如官方的這個例子:
因為Fragment必須嵌入在Acitivity中使用,所以Fragment的生命周期和它所在的Activity是密切相關的。如果Activity是暫停狀態,其中所有的Fragment都是暫停狀態;如果Activity是stop狀態,這個Activity中所有的Fragment都不能被啟動;如果Activity被銷毀,那麼它其中的所有Fragment都會被銷毀。但是,當Activity在活動狀態,可以獨立控制Fragment的狀態,比如加上或者移除Fragment。當這樣進行fragment transaction(轉換)的時候,可以把fragment放入Activity的back stack中,這樣用戶就可以進行返回操作。
一旦Activity進入resumed狀態(也就是running狀態),我們就可以自由地添加和刪除Fragment了。因此,只有當Activity在resumed狀態時,Fragment的生命周期才能獨立的運轉,其它時候是依賴於Activity的生命周期變化的。
onAttach()——>onCreate()——>onCreateView()——>
onActivityCreated()——>
onStart()——>onResume()——>
onPause()——>onStop()——>
onPause()——>onStop()——>onDestroyView()——>onDestroy() ——>onDetach()——>
onCreate()——>onCreateView()——>onActivityCreated()——>
要使用Fragment,可通過繼承Fragment類,復寫相關方法來實現加載Fragment自身的UI並初始化Fragment相關變量,控制Fragment與Activity的交互,通常都需要指定一個UI,但也可以為Activity創建一個沒有UI只提供後台行為的Fragment。
再繼承Fragment重寫相關方法
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.content.Context;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-19.
*/
public class MainFragment extends Fragment {
private final String TAG="FramentDemo";
@Override
public void onAttach(Context context) {
Log.d(TAG, "onAttach:剛剛與Activity對接");
super.onAttach(context);
}
@Override
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG, "onCreate:初始化Fragment對象");
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
Log.d(TAG, "onCreateView:初始化Fragment的UI");
return inflater.inflate(R.layout.fragment_main, container, false);//參數依次為:布局id,依附的容器
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
Log.d(TAG, "onActivityCreated:Activity和Fragment都已經創建好了");
super.onActivityCreated(savedInstanceState);
}
@Override
public void onStart() {
Log.d(TAG, "onStart:Fragment變為可見,待交互");
super.onStart();
}
@Override
public void onResume() {
Log.d(TAG, "onResume");
super.onResume();
}
@Override
public void onPause() {
Log.d(TAG, "onPause:與Activity的OnPause綁定");
super.onPause();
}
@Override
public void onStop() {
Log.d(TAG, "onStop:與Activity的OnStop綁定");
super.onStop();
}
@Override
public void onDestroyView() {
Log.d(TAG, "onDestroyView:Fragment即將被保存或者刪除");
super.onDestroyView();
}
@Override
public void onDestroy() {
Log.d(TAG, "onDestroy:還與Activity藕斷絲連中可以在Activity中找到");
super.onDestroy();
}
@Override
public void onDetach() {
Log.d(TAG, "onDetach:徹底和Activity分手了");
super.onDetach();
}
}
然後再定義Fragment自身的布局xml
接著在onCreatedView方法裡把布局xml映射為view並返回(一般通過Inflater對象的inflate(R.layout.fragment_main, container, false)方法)
return inflater.inflate(R.layout.fragment_main, container, false);//參數依次為:布局id,依附的容器
getFragmentManager().beginTransaction().add(R.id.id_fragment_main,new MainFragment()).commit();//第一個參數為容器的Id,這裡為FrameLayout的Id,第二個參數為Fragment對象實例
getFragmentManager().beginTransaction().add(R.id.id_fragment_main,new MainFragment(),"MainFragment").commit();//第三個參數為tag
首先還是定義Fragemnt對應的布局文件和繼承Fragment實現自己的Fragment 然後在依附的Activity的布局文件中線使用FrameLayout占位
<framelayout android:id="@+id/id_fragment_main" android:layout_height="match_parent" android:layout_width="match_parent">
</framelayout>
再獲取FragmentManager(在V4包中通過getSupportFragmentManager())
FrgamentManager fragmentManager=getFragmentManager();
通過beginTransaction方法開啟事務FragmentTransaction
FragmentTransaction fragmentTransaction=fragmentManager.beginTransaction();
使用FragmentTransaction 的add或者replace方法向容器內添加Fragment(需要傳入容器的id和Fragment的實例對象。)
fragmentTrancsaction.add(id_fragment_main,new MainFragment());//容器的id,Fragment的實例對象
提交事務
fragmentTransaction.commit();
Fragment的刪除也還是借助FragmentManager和FragmentTransaction,主要有兩個步驟:
先通過FragmentManager對象的findFragmentById和findFragmentByTag獲取對應的Fragment 再傳入到remove方法中,並commit。Fragment的替換replace這個方法真的有點奇怪,並沒有像我們正常的邏輯,至少應該提供源Fragment和目標Fragment兩個參數,但並沒有只是提供了一個容器Id和目標Fragment(即將被添加到容器裡展示的Fragment),當年學習的時候也覺得很奇怪,這裡先不深究,以後再去分析。
fragmentManager.beginTransaction().replace(R.id.id_fragment_main,new OtherFragment()).commit();//容器Id,新的Fragment對象
對於Fragment的hide或者show只需要傳遞我們一個將要hide或者show的Fragment即可。(需要注意的是無論是hide還是show都是針對頂層Fragment,也就是假設你要show的Fragment不是頂層的是show不出來的)
private void hideFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().hide(mainFragment).addToBackStack(tag).commit();
}
private void showFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag); fragmentManager.beginTransaction().show(mainFragment).addToBackStack(tag).commit();
}
fragmentManager.beginTransaction().attach(mainFragment).addToBackStack(tag).commit();
public FragmentTransaction detach(Fragment fragment):與attach所做的工作相反,它將參數中的fragment與UI分離,此時的狀態與把fragment放入返回棧時一樣,雖然將view從viewtree中刪除,並將fragment從Activity的隊列中移除!但對應的fragment實例並不會刪除( 在使用detach()後,使用fragment.isAdded()返回的值是false),還是可以被fragmentManager管理的,所以通過FragmentManager.findViewByTag()仍然是會有值的。
fragmentManager.beginTransaction().detach(mainFragment).addToBackStack(tag).commit();
主布局的xml文件:
<framelayout android:id="@+id/id_fragment_main" android:layout_height="0dp" android:layout_weight="1" android:layout_width="match_parent"></framelayout>
MainFragment的布局文件:
OtherFragment的布局文件:
MainActivity的代碼:
package com.crazymo.fragmentsdemo;
import android.app.Activity;
import android.app.Fragment;
import android.app.FragmentManager;
import android.app.FragmentTransaction;
import android.nfc.Tag;
import android.support.annotation.NonNull;
import android.support.annotation.StringRes;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
public class MainActivity extends Activity implements View.OnClickListener {
private final String TAG="FramentDemo";
private Button mAddBtn,mRemoveBtn,mRelpaceBtn,mHideBtn,mShowBtn;
private FragmentManager fragmentManager=getFragmentManager();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG, "Activity onCreate");
setContentView(R.layout.activity_main);
init();
}
private void init(){
getViews();
setClickListener();
}
private void getViews(){
mAddBtn= (Button) findViewById(R.id.id_addfragment_btn);
mRelpaceBtn=(Button)findViewById(R.id.id_replacefragment_btn);
mRemoveBtn= (Button) findViewById(R.id.id_removefragment_btn);
mHideBtn= (Button) findViewById(R.id.id_hidefragment_btn);
mShowBtn= (Button) findViewById(R.id.id_showfragment_btn);
}
private void setClickListener(){
mAddBtn.setOnClickListener(this);
mRemoveBtn.setOnClickListener(this);
mRelpaceBtn.setOnClickListener(this);
mHideBtn.setOnClickListener(this);
mShowBtn.setOnClickListener(this);
}
@Override
protected void onStart() {
Log.d(TAG, "Activity onStart");
super.onStart();
}
@Override
protected void onRestart() {
Log.d(TAG,"Activity onReStart");
super.onRestart();
}
@Override
protected void onResume() {
Log.d(TAG,"Activity onResume");
super.onResume();
}
@Override
protected void onPause() {
Log.d(TAG,"Activity onPause");
super.onPause();
}
@Override
protected void onStop() {
Log.d(TAG,"Activity onStop");
super.onStop();
}
@Override
protected void onDestroy() {
Log.d(TAG,"Activity onDestory");
super.onDestroy();
}
private void addFragemnt(@StringRes int viewGroupId,@NonNull Fragment fragment,String tag){
fragmentManager=getFragmentManager();
fragmentManager.beginTransaction().add(viewGroupId,fragment,tag)
.addToBackStack(tag).commit();
}
private void removeFragment(String tag){
Fragment fragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().remove(fragment).commit();
}
private void replaceFragment(@StringRes int viewGroupId,@NonNull Fragment fragment){
fragmentManager.beginTransaction().add(viewGroupId, fragment).commit();
}
private void hideFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().hide(mainFragment).addToBackStack(tag).commit();
}
private void showFragment(String tag){
Fragment mainFragment = fragmentManager.findFragmentByTag(tag);
fragmentManager.beginTransaction().show(mainFragment).addToBackStack(tag).commit();
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.id_addfragment_btn:
int viewGroupId=R.id.id_fragment_main;
addFragemnt(viewGroupId,new MainFragment(),"MainFragment");
break;
case R.id.id_removefragment_btn:
removeFragment("MainFragment");
break;
case R.id.id_replacefragment_btn:
//Fragment fragment2 = fragmentManager.findFragmentByTag("MainFragment");
//fragmentManager.beginTransaction().remove(fragment2).commit();
int viewGroup=R.id.id_fragment_main;
replaceFragment(viewGroup,new OtherFragment());
break;
case R.id.id_hidefragment_btn:
hideFragment("MainFragment");
break;
case R.id.id_showfragment_btn:
showFragment("MainFragment");
break;
default:
break;
}
}
}
MainFragment的代碼
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.content.Context;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-19.
*/
public class MainFragment extends Fragment {
private final String TAG="FramentDemo";
@Override
public void onAttach(Context context) {
Log.d(TAG, "onAttach:剛剛與Activity對接");
super.onAttach(context);
}
@Override
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG, "onCreate:初始化Fragment對象");
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
Log.d(TAG, "onCreateView:初始化Fragment的UI");
return inflater.inflate(R.layout.fragment_main, container, false);
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
Log.d(TAG, "onActivityCreated:Activity和Fragment都已經創建好了");
super.onActivityCreated(savedInstanceState);
}
@Override
public void onStart() {
Log.d(TAG, "onStart:Fragment變為可見,待交互");
super.onStart();
}
@Override
public void onResume() {
Log.d(TAG, "onResume");
super.onResume();
}
@Override
public void onPause() {
Log.d(TAG, "onPause:與Activity的OnPause綁定");
super.onPause();
}
@Override
public void onStop() {
Log.d(TAG, "onStop:與Activity的OnStop綁定");
super.onStop();
}
@Override
public void onDestroyView() {
Log.d(TAG, "onDestroyView:Fragment即將被保存或者刪除");
super.onDestroyView();
}
@Override
public void onDestroy() {
Log.d(TAG, "onDestroy:還與Activity藕斷絲連中可以在Activity中找到");
super.onDestroy();
}
@Override
public void onDetach() {
Log.d(TAG, "onDetach:徹底和Activity分手了");
super.onDetach();
}
}
OtherFragment的代碼
package com.crazymo.fragmentsdemo;
import android.app.Fragment;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
/**
* Created by cmo on 16-4-21.
*/
public class OtherFragment extends Fragment {
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
return inflater.inflate(R.layout.fragment_other,null);
}
}
熟悉RxAndroid的使用方法. 要點包含: (1) 鏈式表達式的使用方式. (2) Lambda的應用. (3) Rx處理網絡請求. (4) 線程自動管理, 防止內存
Android mvp 架構的自述中我簡單的介紹了mvp,以及怎麼寫mvp。我自己也將mvp運用到了項目中,其實mvp並沒有固定的寫法,正確的去理解架構的思想,都可以有自
Represents the local device Bluetooth adapter. TheBluetoothAdapterlets you perform fu
用過ios的都知道ios上輸入法關閉的同時會自動關閉輸入框,那麼在android上如何實現監聽輸入法彈出和關閉呢?本篇文章就為你提供了一種可靠的實現方式。演示效果視頻地址