草庐IT

java - 在 Java 中存储应用程序的 conf 数据的正确方法是什么?

coder 2023-08-28 原文

您在哪里存储特定于用户和特定于机器的运行时间 J2SE 应用程序的配置数据?
(例如,Windows 上的 C:\Users\USERNAME\AppData\Roaming 和 Unix 上的/home/username)
您如何以独立于平台的方式在文件系统中获取这些位置?

最佳答案

先说格式:

  • Java property files适用于键/值对(也自动处理换行符)。通过使用“点符号”可以实现一定程度的结构化。缺点是该结构不允许您轻松枚举顶级配置实体并以向下钻取的方式工作。最适合用于一小组通常可调整的特定于环境的设置
  • XML 文件 - 经常用于各种 Java 框架(特别是 J2EE 和 Spring)的更复杂配置。我建议您至少了解 Spring - 即使您决定不使用它,它也包含许多值得了解的想法。如果您决定推出自己的 XML 配置,我建议您使用 XStream使用自定义序列化选项,或者如果您只需要解析一些 XML,请查看 XOM .顺便说一句,Spring 还允许您插入自定义 XML 配置语言,但它是 relatively complex task . XML 配置最适合用于最终用户无法看到或调整的更复杂的“内部”配置。
  • 序列化 Java 对象 - 一种保持对象状态并稍后恢复的快速简便的方法。如果您编写配置 GUI 并且您不关心配置是否是人类可读的,则很有用。当心 compatibility issues当你进化类(class)时。
  • 首选项 - 在 Java 1.4 中引入, 允许您在特定于平台的存储中存储键入的文本、数字、字节数组和其他原语。在 Windows 上,这是注册表(您可以在 HKLM 或 HKCU 下的/Software/JavaSoft/Prefs 之间进行选择)。在 Unix 下,相同的 API 在用户 home 或/etc 下创建文件。每个偏好配置单元都可以作为 XML 文件导出和导入。您可以指定 PreferencesFactory 的自定义实现接口(interface)通过将“java.util.prefs.PreferencesFactory”JVM 属性设置为您的实现类名称。

  • 一般来说,根据您的应用场景,使用 prefs API 可能是好事也可能是坏事。
  • 如果您计划在具有不同配置的同一台机器上运行相同代码的多个版本,那么使用 Preferences API 是一个坏主意。
  • 如果您计划在受限环境(Windows 域或严格管理的 Unix 机器)中使用该应用程序,您需要确保您可以正确访问必要的注册表项/目录。这让我不止一次感到意外。
  • 当心漫游配置文件(复制的家庭目录),当涉及多台 Activity 机器时,它们弥补了一些有趣的场景。
  • 首选项不像应用程序目录下的配置文件那么明显。大多数桌面支持人员不期望也不喜欢它们。

  • 关于首选项的文件布局,它再次取决于您的应用程序。一个通用的建议是:
  • 将大部分 XML 文件打包在应用程序的 JAR 中,位于根目录或/META-INF 目录下。这些文件将是只读的,并被视为应用程序的私有(private)文件。
  • 将用户可修改的配置放在 $APP_HOME/conf 下。它应该主要由属性文件组成,偶尔还有一个简单的 XML 文件(XStream 序列化)。这些文件作为安装过程的一部分进行了调整,通常用户无法使用。
  • 在 user-home 下,在一个点目录(即“~/.myapplication”)中存储任何用户配置。用户配置可能会覆盖应用程序 conf 目录中的配置。在应用程序内进行的任何更改都在此处(另请参见下一点)。
  • 您还可以使用 $APP_HOME/var 目录来存储特定于该应用程序实例(而不是用户)的任何其他可变数据。这种方法的另一个优点是您可以通过简单复制一个目录来移动和备份整个应用程序及其配置。

  • 这说明了一些用于管理配置的标准技术。您可以使用不同的库和工具来实现它们,从原始 JRE 开始,添加 Spring/Guice 或使用完整的 J2EE 容器(可能带有嵌入式 Spring)

    其他管理配置的方法是:
  • 使用 multiple base directories用于使用不同配置运行应用程序的多个实例。
  • 使用 lightweight registries用于集中配置管理
  • 一个集中管理的配置管理数据库 (CMDB) 文件,包含每台机器的主机特定值,每晚都会同步到所有生产主机。应用程序使用模板化配置,并在运行时根据当前主机名从 CMDB 中进行选择。
  • 关于java - 在 Java 中存储应用程序的 conf 数据的正确方法是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/194349/

    有关java - 在 Java 中存储应用程序的 conf 数据的正确方法是什么?的更多相关文章

    1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    2. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

      我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

    3. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    4. ruby - 将差异补丁应用于字符串/文件 - 2

      对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

    5. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

      我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

    6. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

      我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

    7. ruby - 如何指定 Rack 处理程序 - 2

      Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

    8. ruby - 在 Ruby 中编写命令行实用程序 - 2

      我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

    9. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

      为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

    10. ruby-on-rails - Rails 应用程序之间的通信 - 2

      我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

    随机推荐