从本文开始,将陆续展示Android Fragment的一些使用方法。
1. 功能说明:在一个Activity中包含Fragment,并且使用一个Button来控制它的显示。在实践过程中,遇到不少问题,记录下来,并且简要说明下。
2. 详细代码:
1) 布局文件
(1) activity_fragment.xml:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".FragmentActivity" >
<TextView
android:id="@+id/text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world" />
<Button
android:id="@+id/btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/cancel" />
<fragment
android:name="com.example.fragmenttest0.FirstFragment"
android:id="@+id/fragment0" android:layout_weight="1"
android:layout_width="0px" android:layout_height="wrap_content" />
</LinearLayout>
(2) labeled_text_edit.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:padding="4dip"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView android:id="@+id/msg"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_weight="0"
android:paddingBottom="4dip" />
<EditText android:id="@+id/saved"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="@string/initial_text"
android:freezesText="true">
<requestFocus />
</EditText>
</LinearLayout>
2) java代码 Fragment的实现: FirstFragment.java:
package com.example.fragmenttest0;
import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
public class FirstFragment extends Fragment {
TextView mTextView;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
//从文件 example_fragment.xml 加载了一个layout
View v = inflater.inflate(R.layout.labeled_text_edit, container, true);
View tv = v.findViewById(R.id.msg);
((TextView)tv).setText("The fragment saves and restores this text.");
mTextView = (TextView)v.findViewById(R.id.saved);
if (savedInstanceState != null) {
mTextView.setText(savedInstanceState.getCharSequence("text"));
}
System.out.println("onCreateView 000000000000000000000");
return v;
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putCharSequence("text", mTextView.getText());
}
}
3) menu文件 fragment_test0.xml:
<menu xmlns:android="http://schemas.android.com/apk/res/android" >
<item
android:id="@+id/action_settings"
android:orderInCategory="100"
android:showAsAction="never"
android:title="@string/action_settings"/>
</menu>
4) java代码 activity的实现: FragmentActivity.java
package com.example.fragmenttest0;
import android.os.Bundle;
import android.app.Activity;
import android.app.Fragment;
import android.app.FragmentTransaction;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class FragmentActivity extends Activity {
Button btn;
Fragment fragment;
FragmentTransaction ft;
TextView t;
boolean commit = false;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_fragment);
ft = getFragmentManager().beginTransaction();
/*为Fragment设置淡入淡出效果,Android开发网提示这里这两个动画资源是android内部资源无需我们手动定义。*/
ft.setCustomAnimations(android.R.animator.fade_in,
android.R.animator.fade_out);
fragment = getFragmentManager().findFragmentById(R.id.fragment0);
ft.add(fragment, "FragmentTest0000");
t = (TextView)findViewById(R.id.text);
btn = (Button)findViewById(R.id.btn);
MyclickListener l = new MyclickListener();
btn.setOnClickListener(l);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.fragment_test0, menu);
return true;
}
public class MyclickListener implements OnClickListener {
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
String str = getString(R.string.ok);
System.out.println("btn.getText=" + btn.getText());
System.out.println("string=" + getString(R.string.ok));
if (btn.getText()==getString(R.string.ok))
{
btn.setText(R.string.cancel);
ft.hide(fragment);
t.setText("fragment hidden");
}
else {
btn.setText(R.string.ok);
ft.show(fragment);
t.setText("fragment show");
}
ft.commit(); // throw exception
}
}
}
3。 分析:
关于Fragment的详细说明,包括生命周期,暂不进行说明,这里主要以例程的方式来进行探讨。主要是对FragmentActivity。java进行说明.
1)。 出现的问题:
点击两次就会 抛出下面的异常:
10-24 13:13:37.910: E/AndroidRuntime(24611): FATAL EXCEPTION: main
10-24 13:13:37.910: E/AndroidRuntime(24611): java.lang.IllegalStateException: commit already called
10-24 13:13:37.910: E/AndroidRuntime(24611): at android.app.BackStackRecord.commitInternal(BackStackRecord.java:533)
说明:如上所说,commit不能被同一个FragmentTransaction调用多次,上面的代码正是违反了这条规定。
将最后一句ft.commit();改为 下面的语句,就不会有异常了。
if (!commit) {
ft.commit();
commit = true;
}
2) 虽然不抛出异常,但是,fragment总是显示不出来。
说明:用如下两种方式来解决:
(1) 利用FragmentTransaction 的add 和remove方式来实现,这样的话,每次都要重新调用FirstFragment的onCreateView函数。代码改为:
。。。。。。
if (btn.getText()==getString(R.string.cancel))
{
ft = getFragmentManager().beginTransaction();
/*为Fragment设置淡入淡出效果,Android开发网提示这里这两个动画资源是android内部资源无需我们手动定义。*/
ft.setCustomAnimations(android.R.animator.fade_in,
android.R.animator.fade_out);
ft.add(fragment, "FragmentTest0000");
ft.commit();
btn.setText(R.string.ok);
t.setText("btn ok!");
}
else {
ft = getFragmentManager().beginTransaction();
/*为Fragment设置淡入淡出效果,Android开发网提示这里这两个动画资源是android内部资源无需我们手动定义。*/
ft.setCustomAnimations(android.R.animator.fade_in,
android.R.animator.fade_out);
ft.remove(fragment);
ft.commit();
btn.setText(R.string.cancel);
t.setText("btn cancel!");
}
(2) 不要把FragmentTransaction ft作为类的数据成员,而是作为本地变量来使用,代码如下:
FragmentTransaction ft = getFragmentManager().beginTransaction();
/*为Fragment设置淡入淡出效果,Android开发网提示这里这两个动画资源是android内部资源无需我们手动定义。*/
ft.setCustomAnimations(android.R.animator.fade_in,
android.R.animator.fade_out);
if (fragment.isHidden()) {
ft.show(fragment);
t.setText("fragment show");
} else {
ft.hide(fragment);
t.setText("fragment hidden");
}
ft.commit();
if (btn.getText()==getString(R.string.cancel))
{
btn.setText(R.string.ok);
t.setText("btn ok!");
}
else {
btn.setText(R.string.cancel);
t.setText("btn cancel!");
}
注意: 这种方式和上一种方式的区别在于:
每次点击button,都是重新创建FragmentTransaction的实例ft,用这个新的实例去操作已经有的fragment,通过hide和show方法。很明显,这种方式不会重新调用FirstFragment的onCreateView函数。
这对于Fragment的生命周期管理是非常重要的。
<本文完>