53

Spring 3 では、Bean 定義で使用できる新しい式言語 (SpEL)が導入されました。構文自体はかなり適切に指定されています。

明確ではないのは、SpEL が、以前のバージョンに既に存在していたプロパティ プレースホルダー構文と相互作用する場合、それがどのように相互作用するかです。SpEL はプロパティ プレースホルダーをサポートしていますか? それとも、両方のメカニズムの構文を結合する必要がありますか?

具体例を挙げましょう。プロパティ構文を使用したいのですが、が未定義の場合を処理するためにエルビス演算子${x.y.z}によって提供される「デフォルト値」構文が追加されています。${x.y.z}

次の構文を試しましたが成功しませんでした:

  • #{x.y.z?:'defaultValue'}
  • #{${x.y.z}?:'defaultValue'}

最初のものは私に与えます

タイプ「org.springframework.beans.factory.config.BeanExpressionContext」のオブジェクトにフィールドまたはプロパティ「x」が見つかりません

これは、SpEL がこれをプロパティ プレースホルダーとして認識していないことを示唆しています。

2 番目の構文は、プレースホルダーが認識されないことを示す例外をスローするため、プレースホルダー リゾルバー呼び出されますが、プロパティが定義されていないため、期待どおりに失敗します。

ドキュメントはこの相互作用について言及していないため、そのようなことは不可能であるか、文書化されていません。

誰でもこれを行うことができましたか?


わかりました、私はこのための小さな自己完結型のテスト ケースを考え出しました。これはすべてそのまま機能します。

まず、Bean の定義:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xmlns:context="http://www.springframework.org/schema/context"
           xmlns:util="http://www.springframework.org/schema/util"
           xsi:schemaLocation="
                http://www.springframework.org/schema/beans   http://www.springframework.org/schema/beans/spring-beans.xsd
                http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
                http://www.springframework.org/schema/util    http://www.springframework.org/schema/util/spring-util.xsd
           "> 

    <context:property-placeholder properties-ref="myProps"/>

    <util:properties id="myProps">
        <prop key="x.y.z">Value A</prop>
    </util:properties>

    <bean id="testBean" class="test.Bean">
            <!-- here is where the magic is required -->
        <property name="value" value="${x.y.z}"/> 

            <!-- I want something like this
        <property name="value" value="${a.b.c}?:'Value B'"/> 
            --> 
    </bean>     
</beans>

次に、自明な Bean クラス:

パッケージテスト;

public class Bean {

    String value;

    public void setValue(String value) {
        this.value = value;
    }
}

最後に、テスト ケース:

package test;

import static org.hamcrest.Matchers.*;
import static org.junit.Assert.*;

import javax.annotation.Resource;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration
public class PlaceholderTest {

    private @Resource Bean testBean;

    @Test
    public void valueCheck() {
        assertThat(testBean.value, is("Value A"));
    }
}

${x.y.z}課題 -解決できない場合にデフォルト値を指定できるようにする、Beans ファイル内の SpEL 式を考え出すこと。このデフォルト値は、別のプロパティ セットで外部化するのではなく、式の一部として指定する必要があります。

4

9 に答える 9

27

SpEL 式からプロパティ プレースホルダーにアクセスするには、次の構文を使用できます#{'${x.y.z}'}。ただし、解決できない場合に例外がスローされるため、エルビス演算子とデフォルト値の問題を解決${x.y.z}できません。

ただし、プロパティのデフォルト値を宣言するために SpEL は必要ありません。

<context:property-placeholder location="..." properties-ref="defaultValues"/>

<bean id = "defaultValues" class = "org.springframework.beans.factory.config.PropertiesFactoryBean">
    <property name="properties">
        <props>
            <prop key="x.y.z">ZZZ</prop>
        </props>
    </property>
</bean>

<bean ...>
    <property name = "..." value = "${x.y.z}" />
</bean>
于 2010-01-11T12:36:23.483 に答える
11

コロンを逃したようです:

#{ ${x.y.z} ?: 'defaultValue' }
于 2010-01-11T12:22:27.140 に答える
11

${myProps.item:defaultValue}myProps.itemが存在しない場合は を使用することを意味しますdefaultValue。これは、プロパティ プレースホルダーの既定の動作です。

#{defaultValue}リテラル値の SpEL を意味します。

つまり、が存在しない${myProps.item:#{defaultValue}}場合myProps.item、SpEL の値を計算し、それを対象フィールドに割り当てます。

例:

${redis.auth:#{null}}は、redis.authプロパティが存在しない場合は に設定することを意味しますnull

于 2015-04-13T10:37:37.350 に答える
8

プレースホルダーのデフォルト値を設定するだけの場合は、次を参照してください。

   <property name="value" value="${x.y.z:defaultValue}"/> 

これを使用して、SpEL とプレースホルダーとの間の相互作用をテストする場合:

   <!-- set value "77-AA-BB-CC-88" when property "x.y.z" not exist -->
   <property name="value" value="77-#{'AA-${x.y.z:BB}-CC'}-88"/>
于 2013-06-13T09:45:42.240 に答える
3

実際、Property-Placeholder は問題自体を解決できます。つまり、 property を使用して、Spring コンテキストでデフォルト設定を明示的に指定できますpropertieslocalOverride次に、使用する設定の場所を指定し、プロパティを に設定しますtrue。このような場合、外部リソース (プロパティで指定) にあるすべてのプロパティは、locationデフォルトのもの (コンテキスト内で明示的に定義) をオーバーライドします。

私が助けてくれることを願っています。

于 2011-02-01T13:39:31.000 に答える
3

サンプル内で実行するには、これを追加する必要があります

<bean id="testBean" class="elvis.Bean">
        <!-- here is where the magic is required
    <property name="value" value="${x.y.z}"/>
    -->

        <!-- I want something like this -->
    <property name="value" value="#{myProps.get('a.b.c')?:'Value B'}"/>

</bean>

Spring はmemberを持つ memberを持つ${a.b.c}オブジェクトを評価しようとするため、このアプローチは機能しません。存在しないために NPE が発生します。abca

于 2012-03-14T21:54:06.000 に答える
0

私は次のことを試しましたが、うまくいきました(かなり醜いですが):

#{ myProps.getProperty('x.y.z')?:'Value B' }

于 2012-07-24T08:42:04.147 に答える