web-dev-qa-db-ja.com

Web MVC Springアプリの場合、@ Transactionalはコントローラーまたはサービスに移行する必要がありますか?

WebApplicationContextの場合、コントローラーまたはサービスに@Transactionalアノテーションを配置する必要がありますか? Springのドキュメントは私を少し混乱させています。

これが私のweb.xmlです:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://Java.Sun.com/xml/ns/javaee" xmlns:web="http://Java.Sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://Java.Sun.com/xml/ns/javaee http://Java.Sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
  <display-name>Alpha v0.02</display-name>
  <servlet>
    <servlet-name>spring</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
  </servlet>

  <servlet-mapping>
    <servlet-name>spring</servlet-name>
    <url-pattern>*.htm</url-pattern>
  </servlet-mapping>

  <servlet-mapping>
    <servlet-name>spring</servlet-name>
    <url-pattern>*.json</url-pattern>
  </servlet-mapping>

  <welcome-file-list>
    <welcome-file>index.jsp</welcome-file>
  </welcome-file-list>
</web-app>

これが、Springディスパッチャーサーブレットを定義する私のapplication-context.xmlです。

<?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:mvc="http://www.springframework.org/schema/mvc"
        xmlns:tx="http://www.springframework.org/schema/tx"
        xsi:schemaLocation="
            http://www.springframework.org/schema/tx
            http://www.springframework.org/schema/tx/spring-tx.xsd
            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/mvc 
            http://www.springframework.org/schema/mvc/spring-mvc.xsd">

    <context:annotation-config />
    <mvc:annotation-driven />
    <tx:annotation-driven />

    <context:component-scan base-package="com.visitrend" />

    <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <property name="suffix" value=".jsp"/>
    </bean>

     <bean id="dataSource"  class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
        <property name="driverClass" value="org.postgresql.Driver" />
        <property name="jdbcUrl" value="jdbc:postgresql://localhost:5432/postgres" />
        <property name="user" value="someuser" />
        <property name="password" value="somepasswd" />
    </bean>

    <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="configLocation" value="classpath:test.hibernate.cfg.xml" />
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
      <property name="dataSource" ref="dataSource" />
      <property name="sessionFactory" ref="sessionFactory" />
    </bean>    
</beans>

サービスインターフェイスは次のとおりです。

public interface LayerService {
    public void createLayer(Integer layerListID, Layer layer);
}

サービスの実装は次のとおりです。

@Service
public class LayerServiceImpl implements LayerService {

    @Autowired
    public LayerDAO layerDAO;

    @Transactional
    @Override
    public void createLayer(Integer layerListID, Layer layer) {
        layerDAO.createLayer(layerListID, layer);
    }
}

そして、これが私のコントローラーです:

@Controller
public class MainController {

    @Autowired
    private LayerService layerService;

    @RequestMapping(value = "/addLayer.json", method = RequestMethod.POST)
    public @ResponseBody
    LayerListSetGroup addLayer(@RequestBody JSONLayerFactory request) {
        layerService.createLayer(request.getListId(), request.buildLayer());
        return layerService.readLayerListSetGroup(llsgID);
    }
}

Springのドキュメントには少し混乱があります。 WebApplicationContextを使用するということは、サービスではなく、コントローラーのみが@Transactionalアノテーションについて調査されることを意味しているようです。その間、私はサービスをコントローラーではなくトランザクションにするためのたくさんの推奨事項を見ています。上記のspring-servlet.xmlで<context:component-scan base-package="com..." />を使用してサービスパッケージを含めると、サービスがコンテキストの一部になるため、トランザクションアノテーションが「調査」されると思います。これは正確ですか?

これが私を混乱させたSpringのドキュメントの宣伝文句です:

@EnableTransactionManagementは、定義されているのと同じアプリケーションコンテキスト内のBeanでのみ@Transactionalを検索します。つまり、DispatcherServletのWebApplicationContextにアノテーション駆動型構成を配置すると、コントローラー内の@Transactional Beanのみがチェックされ、チェックされません。あなたのサービス。

さらに、コントローラーメソッドをトランザクションとして定義し、それが別のクラスのトランザクションメソッドを呼び出す場合、パフォーマンスへの影響または「悪さ」はありますか?ドキュメントに基づくと、私の勘はありませんが、それを検証したいと思います。

16
user2208384

本サービスは、取引の境界を定めるのに最適な場所です。サービスは、ユーザーインタラクションの詳細レベルのユースケース動作を保持する必要があります。つまり、トランザクションで論理的に連携するものを保持する必要があります。また、そのようにして、Webアプリケーションのグルーコードとビジネスロジックの間の分離が維持されます。

重要なビジネスロジックを持たないCRUDアプリケーションはたくさんあります。コントローラーとデータアクセスオブジェクトの間でデータを渡すだけのサービスレイヤーがあるため、役に立ちません。そのような場合は、データアクセスオブジェクトにトランザクションアノテーションを配置することで回避できます。

トランザクションアノテーションをコントローラーに配置すると、問題が発生する可能性があります。[SpringMVCドキュメント] [1]、17.3.2を参照してください。

アノテーション付きコントローラークラスを操作する際の一般的な落とし穴は、コントローラーオブジェクトのプロキシの作成を必要とする機能(@Transactionalメソッドなど)を適用するときに発生します。通常、JDK動的プロキシを使用するために、コントローラのインターフェースを導入します。これを機能させるには、@ RequestMappingアノテーション、およびその他のタイプおよびメソッドレベルのアノテーション(@ ModelAttribute、@ InitBinderなど)をインターフェースに移動する必要があります。マッピングメカニズムは、によって公開されたインターフェースのみを「見る」ことができます。プロキシ。または、コントローラーに適用される機能の構成でproxy-target-class = "true"をアクティブ化することもできます(のトランザクションシナリオで)。そうすることは、インターフェースベースのJDKプロキシの代わりにCGLIBベースのサブクラスプロキシを使用する必要があることを示しています。さまざまなプロキシメカニズムの詳細については、9.6項「プロキシメカニズム」を参照してください。

属性に設定したトランザクション伝播動作によって、トランザクションメソッドが別のトランザクションメソッドを呼び出したときに何が起こるかが決まります。呼び出されたメソッドが同じトランザクションを使用するように、または常に新しいトランザクションを使用するように構成できます。

サンプルコードでサービスを複数回呼び出すことで、サービスのトランザクション目的を無効にすることになります。トランザクションアノテーションをサービスに配置すると、サービスへのさまざまな呼び出しがさまざまなトランザクションで実行されます。

4
Nathan Hughes

特にHibernateを使用して簡単な操作を実行する場合は、@ Transactionalコントローラーメソッドがあると非常に便利な場合があります。 XML構成を使用してこれを有効にするには、これをdispatch-servlet.xmlに追加します。

<beans ...
 xmlns:tx="http://www.springframework.org/schema/tx"
 xsi:schemaLocation="...
 http://www.springframework.org/schema/tx
 http://www.springframework.org/schema/tx/spring-tx-3.2.xsd">    
  <tx:annotation-driven transaction-manager="transactionManager"
   proxy-target-class="true" />
  ..
</beans>

Proxy-target-classの目的は、コントローラー上のAOPが機能するために必要なCGLIBプロキシーを使用することです。これを追加しないと、起動時にエラーが発生します。また、コントローラーにfinalメソッドがある場合は、それらをプロキシできない(特にトランザクションにする)ことができないことに注意してください。また、起動時に、これらのメソッドごとにCGLIBから警告が表示されます。

0
Jeff Tsay