java @ transmitting not working with readOnly = true

neekobn8  于 2023-10-14  发布在  Java
关注(0)|答案(3)|浏览(127)

我有下面的代码,我设置@Transactional(readOnly = true)
main()方法中的代码:

ApplicationContext context = Utils.getContext();
    AnnotatedCrudDao service = new AnnotatedCrudDao();
    DataSource dataSource = (DataSource) context.getBean("mySqlDataSource");
    service.setDataSource(dataSource);
    service.insert(account, user, movie);

@Transactional
public class AnnotatedCrudDao extends JdbcDaoSupport {

    private static Logger logger;

    static {
    logger = Logger.getLogger(AnnotatedCrudDao.class);
    }

    @Transactional(readOnly = true)
    public void insert(Account account, User user, MovieTicket movie) {

    TicketUtils.insertAccount(getJdbcTemplate(), account);
    TicketUtils.insertUser(getJdbcTemplate(), user);
    TicketUtils.insertMovie(getJdbcTemplate(), movie);
    }
}

class TicketUtils{

    public static void insertUser(JdbcTemplate template, User user) {
    String queryUser = "INSERT INTO t_user_txn (ID, NAME, ACCOUNT_ID, TICKETID) VALUES (?,?,?,?)";
    logger.debug("queryUser" + queryUser);
    template.update(queryUser, new Object[] { user.getId(), user.getName(),
    user.getAccount().getId(), user.getTicketId() });
    }

    public static void insertMovie(JdbcTemplate template, MovieTicket movie) {
    String queryMovie = "INSERT INTO t_movieticket_txn (ID, MOVIENAME, TOTALTICKETSCOUNT, PRICE) VALUES (?,?,?,?)";
    logger.debug("queryMovie:" + queryMovie);
    template.update(queryMovie, new Object[] { movie.getId(),
    movie.getMovieName(), movie.getTotalTicketsCount(),
    movie.getPrice() });
    }

    public static void insertAccount(JdbcTemplate template, Account account) {
    String queryAccount = "INSERT INTO t_account_txn (ID, AMOUNT) VALUES (?,?)";
    logger.debug("queryAccount:" + queryAccount);
    template.update(queryAccount, new Object[] { account.getId(),
    account.getAmount() });
    }

}

上下文:

<bean id="mySqlDataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
    <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/>
    <property name="url" value="jdbc:oracle:thin:@localhost:1521:qadb7"/>
    <property name="username" value="tp2"/>
    <property name="password" value="tp2"/>
    </bean>

    <bean id="dsTxnMgr" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="mySqlDataSource" />
    </bean>

    <!-- Add this tag to enable annotations transactions -->
    <tx:annotation-driven transaction-manager="dsTxnMgr" />

即使我已经为insert方法设置了@Transactional(readOnly = true),仍然会执行insert操作。设置为true的属性readOnly是否应该注意不能在此方法上执行插入。

jdzmm42g

jdzmm42g1#

几件事
1.你正在创建你自己的dao示例,这不是spring管理的,因此@Transactional注解没有做任何事情。

  1. readonly="true"不强制执行任何东西不要期望抛出异常或不发生插入,很少有jdbc提供程序实际上使用readOnly标志做了一些事情。这仅仅是对潜在系统的一个暗示,仅此而已。一般来说,它被ORM工具使用和理解,如Hibernate等。但是对于普通的JDBC访问,在大多数情况下它会被忽略。
rdlzhqv9

rdlzhqv92#

您将自己创建AnnotatedCrudDao对象。因此,Spring无法处理它并添加@Transactional行为。

AnnotatedCrudDao service = new AnnotatedCrudDao();

让Spring管理你的对象。在你的上下文中为它声明一个<bean>元素,然后使用它。

8gsdolmq

8gsdolmq3#

您需要放置一个@Repository注解才能使其工作

@Transactional
@Repository
public class AnnotatedCrudDao extends JdbcDaoSupport {

相关问题