当前位置: 首页 > 面试题库 >

如何创建JVM全局Singleton?

洪博涛
2023-03-14
问题内容

如何创建一个Java类实例,以确保整个JVM进程只能使用一次?然后,在该JVM上运行的每个应用程序都应该能够使用该单例实例。


问题答案:

实际上,您可以实现这种单例。在注释中向您描述的问题是可能有多个ClassLoaders
加载类。ClassLoader然后,每个都可以定义一个相同名称的类,该类将错误地假定为唯一。

但是,您可以通过实现单例的访问器来避免这种情况,该访问器显式依赖于检查特定ClassLoader名称的类,该类又包含您的单例。这样,您可以避免由两个不同的ClassLoaders
提供一个单例实例,从而避免了您需要在整个JVM中唯一的实例。

出于稍后说明的原因,我们将把SingletonSingletonAccessor分为两个不同的类。对于以下类,我们需要稍后确保始终使用特定的类来访问它ClassLoader

package pkg;
class Singleton {
  static volatile Singleton instance;
}

方便ClassLoader的是系统类加载器。系统类加载器知道JVM的类路径上的所有类,并且按照定义将扩展和引导类加载器作为其父类。这两个类加载器通常不知道任何特定于域的类,例如我们的Singleton类。这使我们免于意外的意外。此外,我们知道它在整个JVM实例中都是可访问和全局的。

现在,让我们假设Singleton该类在类路径上。这样,我们可以使用反射通过该访问器接收实例:

class SingletonAccessor {
  static Object get() {
    Class<?> clazz = ClassLoader.getSystemClassLoader()
                                .findClass("pkg.Singleton");
    Field field = clazz.getDeclaredField("instance");
    synchronized (clazz) {
      Object instance = field.get(null);
      if(instance == null) {
        instance = clazz.newInstance();
        field.set(null, instance);
      }
      return instance;
    }
  }
}

通过指定我们明确pkg.Singleton要从系统类加载器加载,我们确保无论哪个类加载器加载了我们的实例,我们总是收到相同的实例SingletonAccessor。在上面的示例中,我们另外确保Singleton仅实例化一次。另外,您可以将实例化逻辑放入Singleton类本身,并在未Singleton加载其他类的情况下使未使用的实例腐烂。

但是,这有一个很大的缺点。您会错过所有类型安全的方法,因为您不能假设您的代码始终从ClassLoader将类加载委托Singleton给系统类加载器的方式运行。这是尤其如此。这通常是工具儿童第一语义的类加载器和它的应用服务器上运行的应用程序
要求系统类加载器已知类型的,但首先会尝试加载自己的类型。请注意,运行时类型具有两个功能:

  1. 它的全名
  2. 它的 ClassLoader

因此,该SingletonAccessor::get方法需要返回Object而不是Singleton

另一个缺点是Singleton必须在类路径上找到该类型才能起作用。否则,系统类加载器将不知道这种类型。如果可以将Singleton类型放到类路径中,则可以在此处完成。没问题。

如果您无法做到这一点,那么还有另一种方法,例如使用我的代码生成库Byte
Buddy。使用这个库,我们可以在运行时简单地定义这样的类型,并将其注入系统类加载器中:

new ByteBuddy()
  .subclass(Object.class)
  .name("pkg.Singleton")
  .defineField("instance", Object.class, Ownership.STATIC)
  .make()
  .load(ClassLoader.getSytemClassLoader(), 
        ClassLoadingStrategy.Default.INJECTION)

您刚刚pkg.Singleton为系统类加载器定义了一个类,以上策略再次适用。

另外,您可以通过实现包装器类型来避免类型安全问题。您还可以借助Byte Buddy将其自动化:

new ByteBuddy()
  .subclass(Singleton.class)
  .method(any())
  .intercept(new Object() {
    @RuntimeType
    Object intercept(@Origin Method m, 
                     @AllArguments Object[] args) throws Exception {
      Object singleton = SingletonAccessor.get();
      return singleton.getClass()
        .getDeclaredMethod(m.getName(), m.getParameterTypes())
        .invoke(singleton, args);
    }
  })
  .make()
  .load(Singleton.class.getClassLoader(), 
        ClassLoadingStrategy.Default.INJECTION)
  .getLoaded()
  .newInstance();

您刚刚创建了一个委托,该委托重写了Singleton该类的所有方法,并将它们的调用委派给JVM全局单例实例的调用。请注意,即使反射方法具有相同的签名,也需要重新加载它们,因为我们不能依赖ClassLoader委托的s和JVM全局类的相同。

在实践中,您可能希望缓存对调用SingletonAccessor.get()甚至反射方法查找的调用(与反射方法调用相比,这是相当昂贵的)。但是这种需求在很大程度上取决于您的应用程序域。如果您对构造函数的层次结构有疑问,也可以将方法签名分解为一个接口,并为上述访问器和您的Singleton类实现此接口。



 类似资料:
  • 问题内容: 我有一个全局变量,需要在我的ViewController之间共享。 在Objective-C中,我可以定义静态变量,但是找不到在Swift中定义全局变量的方法。 您知道这样做的方法吗? 问题答案: 来自官方的Swift编程指南: 全局变量是在任何函数,方法,闭包或类型上下文之外定义的变量。全局常数和变量总是延迟计算的。 您可以在任何文件中定义它,也可以在任何位置访问它。因此,您可以在任

  • 问题内容: 我正在尝试在文件中创建如下。 问题是我发现了问题 类型“配置”没有成员“窗口” 请让我知道如何解决该问题。 问题答案: 表示该类中有一个对象,事实并非如此。 您将需要与with 一起使用,但这无济于事,因为此窗口实际上并不代表任何现有的窗口,而且它也不是视图控制器。 因此,我建议您将要使用的实际视图控制器传递给该方法: 然后从可使用UIViewController对象的类中调用它。

  • 问题内容: 在我的android应用程序中,我需要放置可变成员id的位置。问题是,它是从在线API获取的,我需要找到一种存储/检索它的方法 我尝试将其放在自定义类中,但是问题是,如果我取消活动,它将丢失,我还知道有一种方法可以扩展应用程序。 所以我想知道存储全局变量的最佳方法是什么? 我必须实现: 将变量保存在onSaveState上 将其保存在sharepref 手动保存 手动检索 谢谢 更新:

  • 问题内容: 我试图设置一个全局变量。就我而言,只是一个布尔标志,用于指示是否首次显示视图: 呈现视图后,我想将此标志设置为false: 然后检查它: 因此,我想创建一个全局变量。在哪里以及如何?我试过了: 在我的视图控制器领域 在我的AppDelegate.swift文件中的方法中 在班上 没运气。我收到一条错误消息 (注意:我意识到,在这个问题中,我背叛了我对Swift中如何处理范围的无知。请原

  • 我得到一个编译错误的代码: 我想让“match”成为一个全局变量。 我的编译错误是: “内部类TestingProgramm中的静态声明非法。匹配修饰符”static“只允许在常量变量声明中使用 在初始化期间使用静态非final变量。“ 我不知道这个错误是什么意思,也不知道如何修复它。

  • 问题内容: 我正在http://www.cafeaulait.org/javafaq.html上阅读#6.10项,然后我开始怀疑大型企业如何创建自己的JVM实现。一个人会尝试(或可行)实验性的东西吗? 问题答案: 从技术上讲,创建该新JVM所需的所有信息都是该语言和目标平台的公共规范。即使字节码解释在很大程度上相同,JVM还是需要根据其是要在台式机还是手机上运行而有所不同。 一些开始寻找信息的地方