6年前我问过这个问题。与此同时,Android 开发的最佳实践发生了变化,我成为了一名更好的开发人员。
从那时起,我意识到使用onClick
XML 属性是一种不好的做法,并从我工作的任何代码库中删除了它。
我所有的点击处理程序现在都在应用程序的代码中定义,而不是 XML 布局!
我从不使用的原因onClick
是
- XML属性的值很容易出错
onClick
,从而导致运行时错误 - 开发人员可能会重构单击处理程序方法的名称,而没有意识到它是从布局中调用的(参见原因 1)
- 找出实际调用的方法并不总是显而易见的。特别是如果 Fragment 正在使用布局
- 将布局与行为的关注点分开是好的。使用
onClick
将它们混合在一起,这很糟糕!
我希望我已经说服你永远不要onClick
在布局中使用 :) !
下面是我最初的问题,很好地说明了为什么使用onClick
是一个坏主意。
===
我在 XML 中定义菜单项,并尝试使用在 API 11 中添加的 onClick 属性。在运行 4.0.3 的模拟器中启动 Activity 时,会发生以下异常:
FATAL EXCEPTION: main
android.view.InflateException: Couldn't resolve menu item onClick handler
onFeedbackMenu in class android.view.ContextThemeWrapper
...
Caused by: java.lang.NoSuchMethodException: onFeedbackMenu
[interface com.actionbarsherlock.view.MenuItem]
at java.lang.Class.getConstructorOrMethod(Class.java:460)
我不明白是什么导致了异常,因为我的活动中定义了以下方法
import com.actionbarsherlock.view.MenuItem;
...
public void onFeedbackMenu( MenuItem menuItem ) {
Toast.makeText( this, "onFeedBack", Toast.LENGTH_LONG ).show();
}
我的 XML 菜单定义文件包含:
<menu xmlns:android="http://schemas.android.com/apk/res/android" >
...
<item
android:id="@+id/menu_feedback"
android:icon="@drawable/ic_action_share"
android:showAsAction="ifRoom"
android:title="@string/menu_feedback"
android:onClick="onFeedbackMenu" />
</menu>
为了向后兼容,我使用的是 ActionBarSherlock,当我在 2.3.x 上运行应用程序时也得到了一个非常相似的异常。
这是堆栈跟踪的更完整版本
FATAL EXCEPTION: main
android.view.InflateException: Couldn't resolve menu item onClick handler
onFeedbackMenu in class android.view.ContextThemeWrapper
at com.actionbarsherlock.view.MenuInflater$InflatedOnMenuItemClickListener.<init>(MenuInflater.java:204)
at com.actionbarsherlock.view.MenuInflater$MenuState.setItem(MenuInflater.java:410)
at com.actionbarsherlock.view.MenuInflater$MenuState.addItem(MenuInflater.java:445)
at com.actionbarsherlock.view.MenuInflater.parseMenu(MenuInflater.java:175)
at com.actionbarsherlock.view.MenuInflater.inflate(MenuInflater.java:97)
...
Caused by: java.lang.NoSuchMethodException: onFeedbackMenu
[interface com.actionbarsherlock.view.MenuItem]
at java.lang.Class.getConstructorOrMethod(Class.java:460)
at java.lang.Class.getMethod(Class.java:915)
at com.actionbarsherlock.view.MenuInflater$InflatedOnMenuItemClickListener.<init>(MenuInflater.java:202)
... 23 more