- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这是我的问题...
我在包 pkg3 中有一个注释:
package pkg3;
import java.lang.annotation.*;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface TestAnno {
}
此外,我在包 pkg1 中有两个类,一个具有公共(public)访问权限,一个具有默认访问权限
package pkg1;
import pkg3.TestAnno;
class Class1 {
@TestAnno
public void test1() { }
public void test2() { }
}
和
package pkg1;
import pkg3.TestAnno;
public class Class2 extends Class1 {
@TestAnno
public void test3() { }
public void test4() { }
}
最后我在 pkg2 包中得到了一个主类
package pkg2;
import java.lang.reflect.Method;
import pkg1.Class2;
import pkg3.TestAnno;
public class MainClass {
public static void main(String[] args) {
Class2 cls = new Class2();
for(Method m: cls.getClass().getMethods()) {
System.out.println(m);
if (m.getAnnotation(TestAnno.class) != null) {
System.out.println(" > hass anno");
}
}
}
}
运行这个例子我希望看到信息,两个方法有@TestAnno存在 - test1和test3,但我只看到一个test3,并且......奇怪的是,方法test1和test2按原样列出在类 Class2 中声明。
public void pkg1.Class2.test3()
> hass anno
public void pkg1.Class2.test4()
public void pkg1.Class2.test1()
public void pkg1.Class2.test2()
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
... rest methods from java.lang.Object
我知道, getMethods 仅返回给定玻璃中的公共(public)方法(并且都是父类(super class)),但是......这对我来说很奇怪。
我使用它是为了将生成的类(具有默认访问权限)与实现类(它们是公共(public)的,并且它们正在扩展生成的类)分开。我是否要在生成的类中使用公共(public)访问(我希望它们对世界其他地方不可见),或者有什么方法可以从 Class1 获取带注释的公共(public)方法?
最佳答案
这个答案只是一个人在凌晨 3 点提出的“一个大的可能”,因此需要 JLS 对抗、更好的术语和更多信息。我本来打算将其作为评论发布,但不幸的是它太长了:/
<小时/>让我们看一下这个类
class SomeClass {
@TestAnno
public void test(){}
}
class SomeDefaultClass extends SomeClass {
}
public class SomePublicClass extends SomeClass {
}
现在看看这段代码及其结果
Method m1 = SomePublicClass.class.getMethod("test");
Method m2 = SomeDefaultClass.class.getMethod("test");
System.out.println(m1 + "\t> " + m1.getAnnotation(TestAnno.class));
System.out.println(m2 + "\t\t> " + m2.getAnnotation(TestAnno.class));
输出
public void SomePublicClass.test() > null
public void SomeClass.test() > @TestAnno()
如您所见,使用 package 修饰符扩展类的公共(public)类不会继承注释,但使用 package 修饰符的类会继承注释。
<小时/>这是为什么?SomeDefaultClass
和 SomePublicClass
都“继承”了 test()
方法,但方式不同。
如果您查看 javap SomeDefaultClass.class
的结果,您会看到
class SomeDefaultClass extends SomeClass {
SomeDefaultClass();
}
因此它的二进制文件中没有 test()
方法,因此它将使用具有 TestAnno
注释的 SomeClass
方法。
另一方面,如果您查看 javap SomePublicClass
的结果,您将看到
public class SomePublicClass extends SomeClass {
public SomePublicClass();
public void test();
}
这意味着 test()
方法的代码已在 SomePublicClass
中被重写,因此该方法已在 SomePublicClass
中再次声明,但是 不幸的是没有以前的注释并且由于重写的方法没有注释,因此您不会在代码中看到它们。 (为什么编译器重写方法时不添加注释?老实说,我不知道:/)
为什么会发生覆盖?我怀疑,因为 SomePublicClass
是 public 并且 test
也是 public 它应该可以从所有包访问,但由于 SomeClass
具有 default/package 可见性,因此无法通过 SomeClass
访问此方法从其包装外部。
为了防止将 test
方法从一个类移动/复制到另一个类,您可以将这两个类设置为 public
或 default/package
。
关于java - 方法注释和默认访问级别的奇怪行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17624453/
Android 项目中最低(最低 sdk)和最高(目标 sdk)级别是否有任何影响。这些东西是否会影响项目的可靠性和效率。 最佳答案 没有影响,如果您以 SDK 级别 8 为目标,那么您的应用将以 9
我将现有的 android 项目升级到 API 级别 31。我使用 Java 作为语言。我改变了 build.gradle compileSdkVersion 31 defaultConfig {
我正在使用 ionic 3 创建一个 android 应用程序,当我尝试上传到 playstore 时,我收到一个错误,提示我的应用程序以 api 25 为目标,当我检查我的 project.prop
我刚刚尝试将应用程序的目标和编译 API 级别更新为 29 (Android 10),并注意到我无法再编译,因为 LocationManager.addNmeaListener 只接受 OnNmeaM
我的代码没有在 Kitkat 上显示工具栏。 这是我的两个 Android 版本的屏幕截图。 Kitkat 版本: Lollipop 版: 这背后的原因可能是什么。 list 文件
我正在构建面向 API 级别 8 的 AccessabilityService,但我想使用 API 级别 18 中引入的功能 (getViewIdResourceName())。这应该可以通过使用 A
当我想在我的电脑上创建一个 android 虚拟机时,有两个选项可以选择目标设备。它们都用于相同的 API 级别。那么我应该选择哪一个呢?它们之间有什么区别? 最佳答案 一个是基本的 Android,
当我选择 tagret 作为 Android 4.2.2(API 级别 17)时,模拟器需要很长时间来加载和启动。 所以我研究它并通过使用 找到了解决方案Intel Atom(x86) 而不是 ARM
我有一个使用 Android Studio 创建的 Android 项目。我在项目中添加了一些第三方依赖项,但是当我尝试在 Android Studio 中编译时,我遇到了以下错误: Error:Ex
如上所述,如何使用 API 8 获取移动设备网络接口(interface)地址? 最佳答案 NetworkInterface.getInetAddresses() 在 API8 中可用。 关于andr
我想显示 Snackbar并使用图像而不是文本进行操作。 我使用以下代码: val imageSpan = ImageSpan(this, R.drawable.star) val b
我有一个用 python 编写的简单命令行程序。程序使用按以下方式配置的日志记录模块将日志记录到屏幕: logging.basicConfig(level=logging.INFO, format='
使用下面的代码,实现游戏状态以控制关卡的最简单和最简单的方法是什么?如果我想从标题画面开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人能解释处理这个问题的最简单方法,那就太好了! impor
我想创建一个可以找到嵌套树结构深度的属性。下面的静态通过递归找出深度/级别。但是是否可以将此函数作为同一个类中的属性而不是静态方法? public static int GetDepth(MenuGr
var myArray = [{ title: "Title 1", children: [{ title: "Title 1.1", children: [{
通过下面的代码,实现游戏状态来控制关卡的最简单、最容易的方法是什么?如果我想从标题屏幕开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人可以解释处理这个问题的最简单方法,那就太好了! impo
我有一个树结构,其中每个节点基本上可以有无限个子节点,它正在为博客的评论建模。 根据特定评论的 ID,我试图找出该评论在树中的深度/级别。 我正在关注 this guide that explains
考虑任何给定的唯一整数的数组,例如[1,3,2,4,6,5] 如何确定“排序度”的级别,范围从 0.0 到 1.0 ? 最佳答案 一种方法是评估必须移动以使其排序的项目数量,然后将其除以项目总数。 作
我如何定义一个模板类,它提供一个整数常量,表示作为输入模板参数提供的(指针)类型的“深度”?例如,如果类名为 Depth,则以下内容为真: Depth::value == 3 Depth::value
我的场景是:文件接收器应该包含所有内容。另一个接收器应包含信息消息,但需要注意的是 Microsoft.* 消息很烦人,因此这些消息应仅限于警告。两个sink怎么单独配置?我尝试的第一件事是: str
我是一名优秀的程序员,十分优秀!