gpt4 book ai didi

java - 使用带有自定义 Bean 解析器的 .properties 文件

转载 作者:搜寻专家 更新时间:2023-11-01 02:50:04 25 4
gpt4 key购买 nike

我有一个 AbstractSingleBeanDefinitionParser 的自定义实现,允许我在我的 spring 配置中定义 3D vector ,比其他方式需要的仪式更少。

<rbf:vector3d id="test_vector" delimeter=";" value="45;46;47"/>

效果很好,我已经使用了几个月没有任何问题。昨天我尝试在 .properties 文件中定义值,如下所示:

在 test.properties 我有:

vector3d.value=1,2,3

在我的 xml 文件中:

<context:property-placeholder location="test.properties"/>
<rbf:vector3d id="test_vector_with_properties" delimeter="," value="${vector3d.value}"/>

当我尝试运行我的单元测试时,它崩溃了,我得到了这个异常:

Caused by: java.lang.NumberFormatException: For input string: "${vector3d.value}"
at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1222)
at java.lang.Double.parseDouble(Double.java:510)
at scala.collection.immutable.StringLike$class.toDouble(StringLike.scala:234)
at scala.collection.immutable.StringOps.toDouble(StringOps.scala:31)
at rb.foundation.spring.xml.Vector3DBeanDefinitionParser$$anonfun$1.apply(Vector3DBeanDefinitionParser.scala:25)

当我将 .properties 文件用于普通 bean 时,它工作得很好,这让我相信我在实现解析器时忽略了一个微妙之处。它是用 Scala 编写的,但您应该能够理解它:

class Vector3DBeanDefinitionParser extends AbstractSingleBeanDefinitionParser
{
override def getBeanClass(element : Element) = classOf[Vector3D]

override def doParse(element: Element, builder: BeanDefinitionBuilder)
{
val delim = element.getAttribute("delimeter")
val value = element.getAttribute("value")

val values = value.split(delim).map(_.toDouble)

builder.addConstructorArgValue(values(0))
builder.addConstructorArgValue(values(1))
builder.addConstructorArgValue(values(2))
}
}

如有必要,我很乐意添加 key 替换,我只需要知道在哪里/如何进行。

想法?

最佳答案

所以这不起作用的原因是您的 BeanDefinitionParser 在解析属性占位符之前运行了很多。我理解的简单概述:

  1. BeanDefinitionParsers 将 XML 解析为内存中的 BeanDefinition 对象
  2. 然后将 BeanDefinition 加载到 BeanFactory 中
  3. BeanFactoryPostProcessors(包括属性占位符配置器)在 bean 定义上执行
  4. bean 是根据 bean 定义创建的

(当然在此过程中还会发生其他事情,但这些是此处的相关步骤。)

因此,为了将已解析的属性值获取到 Vector3D 对象中,我认为您将不得不延迟指定 Vector3D 构造函数的参数,直到 BeanFactoryPostProcessors 运行之后。我想到的一种方法是让 BeanDefinitionParser 为 Spring FactoryBean 而不是 Vector3D 本身构造一个 bean 定义。然后,您当前在 Vector3DBeanDefinitionParser 中拥有的 vector 值的拆分将需要在 FactoryBean 实现中进行。

抱歉,我不太熟悉 Scala,所以这将使用 Java。

FactoryBean 类看起来像这样:

import org.springframework.beans.factory.FactoryBean;

public class Vector3DFactoryBean implements FactoryBean<Vector3D> {
private String delimiter;
private String value;
private transient Vector3D instance;

public String getDelimiter() { return delimiter; }
public void setDelimiter(String delimiter) { this.delimiter = delimiter; }
public String getValue() { return value; }
public void setValue(String value) { this.value = value; }

@Override
public Vector3D getObject() {
if (instance == null) {
String[] values = value.split(delimiter);
instance = new Vector3D(
Double.parseDouble(values[0]),
Double.parseDouble(values[1]),
Double.parseDouble(values[2])
);
}
return instance;
}
@Override
public Class<?> getObjectType() {
return Vector3D.class;
}
@Override
public boolean isSingleton() {
return true;
}
}

然后您的 Vector3DBeanDefinitionParser 将原封不动地将 delimitervalue 传递给 Vector3DFactoryBean bean 定义:

class Vector3DBeanDefinitionParser extends AbstractSingleBeanDefinitionParser
{
override def getBeanClass(element : Element) = classOf[Vector3DFactoryBean]

override def doParse(element: Element, builder: BeanDefinitionBuilder)
{
val delim = element.getAttribute("delimeter")
val value = element.getAttribute("value")

builder.addPropertyValue("delimiter", delim)
builder.addPropertyValue("value", value)
}
}

稍后当占位符属性配置器运行时,它应该解析 Vector3DFactoryBean bean 定义中的属性值。当最终从 bean 定义创建 bean 时,Vector3DFactoryBean 将解析 vector 值并创建 Vector3D 对象。

关于java - 使用带有自定义 Bean 解析器的 .properties 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12936080/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com