我想知道,在创建新Activity
类然后覆盖该onCreate()
方法时,在 Eclipse 中我总是自动添加:super.onCreate()
. 这是怎么发生的?抽象类或父类中是否有 java 关键字强制执行此操作?
我不知道不调用超类是否违法,但我记得在某些方法中我因为不这样做而引发了异常。这也是Java内置的吗?你可以使用一些关键字来做到这一点吗?或者是怎么做的?
我想知道,在创建新Activity
类然后覆盖该onCreate()
方法时,在 Eclipse 中我总是自动添加:super.onCreate()
. 这是怎么发生的?抽象类或父类中是否有 java 关键字强制执行此操作?
我不知道不调用超类是否违法,但我记得在某些方法中我因为不这样做而引发了异常。这也是Java内置的吗?你可以使用一些关键字来做到这一点吗?或者是怎么做的?
这是在支持注释库中添加的:
dependencies {
compile 'com.android.support:support-annotations:22.2.0'
}
http://tools.android.com/tech-docs/support-annotations
@CallSuper
如果你想强制子类执行父类的逻辑,一个常见的模式是这样的:
public abstract class SuperClass implements SomeInterface
{
// This is the implementation of the interface method
// Note it's final so it can't be overridden
public final Object onCreate()
{
// Hence any logic right here always gets run
// INSERT LOGIC
return doOnCreate();
// If you wanted you could instead create a reference to the
// object returned from the subclass, and then do some
// post-processing logic here
}
protected abstract Object doOnCreate();
}
public class Concrete extends SuperClass
{
@Override
protected Object doOnCreate()
{
// Here's where the concrete class gets to actually do
// its onCreate() logic, but it can't stop the parent
// class' bit from running first
return "Hi";
}
}
这实际上并没有回答您关于什么提示 Eclipse 自动将超类调用插入实现的问题;但是我不认为这是要走的路,因为它总是可以被删除。
您实际上不能强制一个方法必须使用 Java 关键字或类似的东西调用超类的版本。我怀疑您的异常只是来自父类中的某些代码检查预期的不变量,或者您的方法无效的东西。请注意,这与抛出异常略有不同,因为您未能调用super.onCreate()
.
这是来源Activity#onCreate()
- 几乎是所有评论(原始 - 见 ~800 行):
/**
* Called when the activity is starting. This is where most initialization
* should go: calling {@link #setContentView(int)} to inflate the
* activity's UI, using {@link #findViewById} to programmatically interact
* with widgets in the UI, calling
* {@link #managedQuery(android.net.Uri , String[], String, String[], String)} to retrieve
* cursors for data being displayed, etc.
*
* <p>You can call {@link #finish} from within this function, in
* which case onDestroy() will be immediately called without any of the rest
* of the activity lifecycle ({@link #onStart}, {@link #onResume},
* {@link #onPause}, etc) executing.
*
* <p><em>Derived classes must call through to the super class's
* implementation of this method. If they do not, an exception will be
* thrown.</em></p>
*
* @param savedInstanceState If the activity is being re-initialized after
* previously being shut down then this Bundle contains the data it most
* recently supplied in {@link #onSaveInstanceState}. <b><i>Note: Otherwise it is null.</i></b>
*
* @see #onStart
* @see #onSaveInstanceState
* @see #onRestoreInstanceState
* @see #onPostCreate
*/
protected void onCreate(Bundle savedInstanceState) {
mVisibleFromClient = !mWindow.getWindowStyle().getBoolean(
com.android.internal.R.styleable.Window_windowNoDisplay, false);
mCalled = true;
}
所以,我的猜测是 ADT Eclipse 插件是super.onCreate()
为您自动添加该调用的原因。不过,这完全是猜测。
如果你想绝对确保超类方法也被调用,你必须采取一些技巧:不要让超类方法被覆盖,而是让它调用一个可覆盖的受保护方法。
class Super
{
public final void foo() {
foo_stuff();
impl_stuff();
}
protected void impl_stuff() {
some_stuff_that_you_can_override();
}
}
class Base extends Super
{
protected void impl_stuff() {
my_own_idea_of_impl();
}
}
这样,用户必须调用 Super.foo() 或 Base.foo() 并且它将始终是基类版本,因为它被声明为最终版本。特定于实现的东西在 impl_stuff() 中,可以被覆盖。
为了回答您的实际问题,自动创建对 super.onCreate() 的调用是 ADT 插件的一项功能。在java中,您不能直接强制子类调用方法afaik的超级实现(请参阅其他答案中描述的模式以了解解决方法)。但是,请记住,在 Android 中,您不会直接实例化 Activity 对象(或 Service 对象)——您将 Intent 传递给系统,系统会实例化该对象并在其上调用 onCreate()(以及其他生命周期方法)。所以系统有一个对 Activity 实例的直接对象引用,并且能够检查(可能)一些在 onCreate() 的超类实现中设置为 true 的布尔值。虽然我不知道它是如何实现的,但它可能看起来像这样:
class Activity
{
onCreate()
{
superCalled = true;
...
}
...
}
在接收 Intent 并从中实例化 Activity 对象的“系统”级类中:
...
SomeActivitySubclass someActivitySubclassObject = new SomeActivitySubclass();
someActivitySubclassObject.onCreate();
if (!someActivityObject.isSuperCalled())
{
Exception e = new Exception(...) //create an exception with appropriate details
throw e;
}
我的猜测是它可能比这稍微复杂一些,但你明白了。Eclipse 会自动创建调用,因为 ADT 插件会告诉它这样做,以方便使用。快乐编码!
Java 中没有任何东西可以强制调用 super,并且有很多您不想这样做的示例。唯一可以强制调用 super 的地方是在构造函数中。所有构造函数都必须调用超类构造函数。如果您没有显式编写一个(无参数构造函数),则将插入一个(无参数构造函数),如果没有无参数构造函数,则必须显式调用它。
Eclipse 只是提供帮助,提醒您可以根据需要调用超类实现。
你可能会得到一个错误,因为你没有做超类所做的必要的事情,因为你没有调用它的实现。
Eclipse 只是帮助您正确地做事并避免异常。
来自http://developer.android.com/reference/android/app/Activity.html#onCreate(android.os.Bundle )
派生类必须调用该方法的超类实现。如果他们不这样做,将引发异常。