我正在探索Java中的方法和变量继承,尤其是类的实例如何查找字段(静态变量)。
但是,我无法在下面的Java片段中获得预期的行为。
更令人困惑的是,运行用Python编写的相同(?)东西可以获得预期的结果。
class Up {
public static int n = 1;
public int m;
public void setter() {
System.out.println("I looked up " + this.n);
this.m = this.n;
System.out.println(this.m);
}
}
class Sub extends Up {
public static int n = 6;
public int m = 5;
public Sub() {
super();
}
}
class Run {
public static void main(String[] args) {
Sub foo = new Sub();
foo.setter();
System.out.println(foo.m);
}
}
class Up:
n = 1
def setter(self):
print("I looked up", self.n)
self.m = self.n
print(self.m)
class Sub(Up):
n = 6
m = 5
foo = Sub()
foo.setter()
print(foo.m)
Python代码按预期运行并打印出:
I looked up 6
6
6
但是,等效的Java代码打印出来了:
I looked up 1
1
5
我想这里有两个问题:
Java为什么将this.n解释为在Up类中初始化的字段而不是foo,Sub的实际类?
在成功查找1之后,Java应该将foo.m绑定到1。打印出this.m似乎表明它已经这样做了,但是直接从main方法中打印出foo.m却表明它没有,为什么呢? ?
我怀疑这是由于Java“在运行时访问变量”引起的(我仍在思考),我想这解释了数字1,但我希望有人会向我解释什么发生在2号后面?
参考方案
对于第二个问题,您有两个名为“ m”的实例变量。在Class Up中声明的一个,在Sub类中声明的一个在Up中隐藏变量。因此,当您调用setter()(这是Up的方法)时,它将使用在Up中找到的m的值(Up不了解其任何子类中找到的变量)。
但是,当您以后打印foo.m时,由于foo是Sub的不变性,因此它使用在Sub中找到的m的值(仍为5)。
在Sub类中删除m的声明,看看会发生什么。
Java:正则表达式模式匹配器是否有大小限制? - java我的模式类似于OR:“word1 | word2 | word3”我大约有800个字。可能有问题吗? 参考方案 您仅受记忆和理智的限制。 :)
Java:找到特定字符并获取子字符串 - java我有一个字符串4.9.14_05_29_16_21,我只需要获取4.9。数字各不相同,所以我不能简单地获得此char数组的前三个元素。我必须找到最正确的.并将其子字符串化直到那里。我来自Python,因此我将展示Python的实现方法。def foobar(some_string): location = some_string.rfind('.&…
Java:线程池如何将线程映射到可运行对象 - java试图绕过Java并发问题,并且很难理解线程池,线程以及它们正在执行的可运行“任务”之间的关系。如果我创建一个有10个线程的线程池,那么我是否必须将相同的任务传递给池中的每个线程,或者池化的线程实际上只是与任务无关的“工人无人机”可用于执行任何任务?无论哪种方式,Executor / ExecutorService如何将正确的任务分配给正确的线程? 参考方案 …
Java:我可以在Hashmaps中使用数组吗? - java我可以在Hashmaps中使用数组吗?如果是这样,则声明这种哈希图的确切语法是什么?谢谢 参考方案 数组也是对象。甚至像int[]这样的原始数组。Map<String,String[]> map = new HashMap<String,String[]>();
JAVA:字节码和二进制有什么区别? - javajava字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…