在Java中,对super()参数列表中的静态方法的调用是有效的。为什么? - java

让我们看一下下面的Java代码段。

package trickyjava;

class A
{
    public A(String s)
    {
        System.out.println(s);
    }
}

final class B extends A
{
    public B()
    {
        super(method());      // Calling the following method first.      
    }

    private static String method()
    {
        return "method invoked";
    }
}

final public class Main
{
    public static void main(String[] args)
    {
        B b = new B();
    }
}

按照约定,Java中的 super()构造函数必须是相关构造函数主体中的第一条语句。在上面的代码中,我们在super()构造函数参数列表本身中调用静态方法 super(method());

这意味着在构造函数 B()的super调用中,一个方法正在
在调用super之前被调用!编译器应禁止这样做,但效果很好。这在某种程度上等同于以下语句。

String s = method();
super(s);

但是,这是非法的,导致出现编译时错误,表明“对super的调用必须是构造函数中的第一条语句”。为什么?以及为什么等效于 super(method()); 有效并且编译器不再抱怨了吗?

参考方案

这里的关键是static修饰符。静态方法绑定到类,实例方法(普通方法)绑定到对象(类实例)。构造函数从一个类初始化一个对象,因此该类必须已经完全加载。因此,调用静态方法作为构造函数的一部分是没有问题的。

加载类和创建对象的事件顺序如下:

  • 加载类
  • 初始化静态变量
  • 创建对象
  • 使用构造函数
  • 初始化对象<-

  • 对象现在可以使用了
  • (简体*)

    到对象构造函数被调用时,静态方法和变量可用。

    将类及其static成员视为该类对象的蓝图。您只能在蓝图已经存在的情况下创建对象。

    构造函数也称为初始化程序。如果从构造函数中抛出异常并打印堆栈跟踪,您会注意到它在堆栈框架中称为<init>。实例方法只能在构造对象之后调用。在构造函数中,不可能将实例方法用作super(...)调用的参数。

    如果创建相同类的多个对象,则步骤1和2仅发生一次。

    (为清晰起见,省略了静态初始化程序和实例初始化程序)

    Java:静态字段在内存中的哪个位置? - java

    如果我们将对象存储在对象的静态字段中,那么JVM如何为它分配内存?它是否存在于“隐式”(不确定我是否使用正确的单词)类对象中?静态字段与对象字段有何不同? 参考方案 静态字段是类变量,并且在该类的所有实例之间共享。实例变量(或我认为您引用它们的对象字段)属于类的各个实例,并且不共享。至于它们存储在内存中的位置将根据JVM的实现而定,因此没有理由需要两个不同的…

    Java:线程池如何将线程映射到可运行对象 - java

    试图绕过Java并发问题,并且很难理解线程池,线程以及它们正在执行的可运行“任务”之间的关系。如果我创建一个有10个线程的线程池,那么我是否必须将相同的任务传递给池中的每个线程,或者池化的线程实际上只是与任务无关的“工人无人机”可用于执行任何任务?无论哪种方式,Executor / ExecutorService如何将正确的任务分配给正确的线程? 参考方案 …

    JAVA:字节码和二进制有什么区别? - java

    java字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…

    java:继承 - java

    有哪些替代继承的方法? java大神给出的解决方案 有效的Java:偏重于继承而不是继承。 (这实际上也来自“四人帮”)。他提出的理由是,如果扩展类未明确设计为继承,则继承会引起很多不正常的副作用。例如,对super.someMethod()的任何调用都可以引导您通过未知代码的意外路径。取而代之的是,持有对本来应该扩展的类的引用,然后委托给它。这是与Eric…

    Java:BigInteger,如何通过OutputStream编写它 - java

    我想将BigInteger写入文件。做这个的最好方式是什么。当然,我想从输入流中读取(使用程序,而不是人工)。我必须使用ObjectOutputStream还是有更好的方法?目的是使用尽可能少的字节。谢谢马丁 参考方案 Java序列化(ObjectOutputStream / ObjectInputStream)是将对象序列化为八位字节序列的一种通用方法。但…