当我在Android Studio上创建一个空片段时,它会生成以下代码:
/**
* A simple {@link Fragment} subclass.
* Activities that contain this fragment must implement the
* {@link TestFragment.OnFragmentInteractionListener} interface
* to handle interaction events.
* Use the {@link TestFragment#newInstance} factory method to
* create an instance of this fragment.
*/
public class TestFragment extends Fragment {
// TODO: Rename parameter arguments, choose names that match
// the fragment initialization parameters, e.g. ARG_ITEM_NUMBER
private static final String ARG_PARAM1 = "param1";
private static final String ARG_PARAM2 = "param2";
// TODO: Rename and change types of parameters
private String mParam1;
private String mParam2;
private OnFragmentInteractionListener mListener;
public TestFragment() {
// Required empty public constructor
}
/**
* Use this factory method to create a new instance of
* this fragment using the provided parameters.
*
* @param param1 Parameter 1.
* @param param2 Parameter 2.
* @return A new instance of fragment TestFragment.
*/
// TODO: Rename and change types and number of parameters
public static TestFragment newInstance(String param1, String param2) {
TestFragment fragment = new TestFragment();
Bundle args = new Bundle();
args.putString(ARG_PARAM1, param1);
args.putString(ARG_PARAM2, param2);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mParam1 = getArguments().getString(ARG_PARAM1);
mParam2 = getArguments().getString(ARG_PARAM2);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_test, container, false);
}
// TODO: Rename method, update argument and hook method into UI event
public void onButtonPressed(Uri uri) {
if (mListener != null) {
mListener.onFragmentInteraction(uri);
}
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
if (context instanceof OnFragmentInteractionListener) {
mListener = (OnFragmentInteractionListener) context;
} else {
throw new RuntimeException(context.toString()
+ " must implement OnFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
/**
* This interface must be implemented by activities that contain this
* fragment to allow an interaction in this fragment to be communicated
* to the activity and potentially other fragments contained in that
* activity.
* <p>
* See the Android Training lesson <a href=
* "http://developer.android.com/training/basics/fragments/communicating.html"
* >Communicating with Other Fragments</a> for more information.
*/
public interface OnFragmentInteractionListener {
// TODO: Update argument type and name
void onFragmentInteraction(Uri uri);
}
}
对于一个空片段来说,这相当复杂。创建的每个部分的目的是什么,我可以删除一些代码以使其“更简单”吗? 没什么必要因为当我创建一个空活动时,它更加“干净”
答案 0 :(得分:3)
我可以删除一些代码以使其“更简单”吗?没什么?
您实际上可以删除所有代码。唯一真正必要的部分是类定义本身:
public class TestFragment extends Fragment {
}
您甚至不需要“必需的空公共构造函数”,因为编译器会为您添加它。您永远不应该手动指定构造函数(除非在此问题范围之外的非常狭窄的情况下)。
当然,在最次使用片段时,将使用它们向用户显示信息,在这种情况下,您需要包括onCreateView()
方法:
public class TestFragment extends Fragment {
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_test, container, false);
}
}
此方法负责创建片段的视图。您不需要使视图膨胀(可以手工制作一个视图并返回它),但是如果您想让用户看到任何东西,则必须 do 返回一些非null的视图。从此方法返回的任何内容都可以稍后通过调用getView()
来检索。
请注意,使用没有视图的片段(例如retained fragment used only as a container for long-running tasks)是完全有效的。
与片段有关的另一种常见做法是使用onAttach()
和onDetach()
将托管片段的活动转换为某个interface
。只要每个活动都实现接口,就可以使片段类与多个主机活动一起使用。
public class TestFragment extends Fragment {
private MyListenerInterface mListener;
@Override
public void onAttach(Context context) {
super.onAttach(context);
this.mListener = (MyListenerInterface) context;
}
@Override
public void onDetach() {
super.onDetach();
this.mListener = null;
}
public interface MyListenerInterface {
void someMethod();
}
}
您可以直接强制转换上下文,而无需在生成的代码中包含if
语句,因为无论哪种方式,您都将引发运行时异常(您将只获得ClassCastException
代替。)
您可以在此界面中定义所需的任何方法。
最后,生成的片段演示了newInstance()
模式,该模式用于“传递参数”到该片段。如前所述,永远不要使用带有片段的构造函数,那么如何将数据传递给它们呢?您可以使用“参数” Bundle
,将其设置为片段,然后再使用getArguments()
检索数据。
每次都手动执行这些操作会很烦人,因此您可以将它们全部封装在newInstance()
静态工厂方法中以简化操作:
public class TestFragment extends Fragment {
private static final String ARG_PARAM1 = "param1";
private static final String ARG_PARAM2 = "param2";
private String mParam1;
private String mParam2;
public static TestFragment newInstance(String param1, String param2) {
Bundle args = new Bundle();
args.putString(ARG_PARAM1, param1);
args.putString(ARG_PARAM2, param2);
TestFragment fragment = new TestFragment();
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.mParam1 = getArguments().getString(ARG_PARAM1);
this.mParam2 = getArguments().getString(ARG_PARAM2);
}
}
放置此位置后,任何需要创建此片段实例的人都可以调用
TestFragment f = TestFragment.newInstance("Hello", "world");
不必写:
TestFragment f = new TestFragment();
Bundle args = new Bundle();
args.putString("param1", "Hello");
args.putString("param2", "world");
f.setArguments(args);
Bundle
类可以处理许多不同类型的数据。此示例使用字符串,但是您可以使用整数或列表等。我已经从if
中删除了onCreate()
检查,因为我们知道,参数束将始终为非-null,因此检查无效。使其不为非null的唯一方法是,如果有人不打电话给newInstance()
,那么与崩溃并解决该问题相比,最好尝试在没有警报的情况下继续前进。我们需要的数据。
您可以通过任何方法调用getArguments()
,而不仅仅是onCreate()
。
答案 1 :(得分:0)
如果您的片段如此简单,您可以改用它
public class TestFragment extends Fragment {
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_test, container, false);
}
//ADD THIS
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
//YOUR LOGIC
}
}
其他函数用于回调并在片段之间共享变量
这两个很重要
onCreateView :用于放大布局
onActivityCreated :用于写入逻辑代码