Saturday, 28 September 2013

Programmatic Spring Transaction Management With Example





Programmatic and Declarative transaction management

There are two types of transaction management in spring:

  • Programmatic transaction management: This will give us more flexibility but maintaining this is a bit difficult. In this transaction, we have to manage the transaction with the help of programming.


  • Declarative transaction management: It will help to separate the transaction management from the business code. For achieving this we can use annotations or XML based configuration to manage the transactions.


Example to achieve Programmatic transaction management:-

Below are the two tables query, which we need to execute in mysql test database , If you want to change the database name from test to any other then do the change while configuring the datasource in the xml file.

EMPLOYEE_DATA Table

CREATE TABLE EMPLOYEE_DATA(
EMP_ID INT NOT NULL AUTO_INCREMENT,
FIRST_NAME VARCHAR(20) NOT NULL,
LAST_NAME VARCHAR(20) NOT NULL,
SALARY DOUBLE NOT NULL,
PRIMARY KEY (EMP_ID) );

ATTENDENCE_DETAILS Table

CREATE TABLE ATTENDENCE_DETAILS(
EID INT NOT NULL,
NO_OF_LEAVE INT NOT NULL,
NO_OF_WORKING_DAYS INT NOT NULL,JOINING_DATE DATE NOT NULL );

System Requirements:-
  •             Eclipse Editor or any other.
  •            JDK 1.5 or higher(I am using jdk 1.7.0_03)
  •            Spring jars version 2.5.5.
  •            Other related jars.


Required Jars are:-

spring-tx.jar
spring-beans.jar
spring-context.jar
spring-context-support.jar
spring-core.jar
spring-jdbc.jar
spring-orm.jar
commons-logging.jar
commons-collections-3.2.jar
mysql-connector-java-5.0.4-bin.jar

Steps for creating Eclipse java project for implementing spring transaction management:-
  • Create a java project in eclipse and name it as SpringTransactionExample.
  • Create a package in the src folder with the name as com.gaurav.springtransaction.dao.
  • Create a package in the src folder with the name com.gaurav.springtransaction.bean .
  • Create a package in the src folder with the name com.gaurav.springtransaction.mapper.
  • Create a package in the src folder with the name com.gaurav.springtransaction.dao.impl.
  • Create a package in the src folder with the name com.gaurav.springtransaction.caller.
  • Create an XML file and name it as spring-transactionmanagement.xml and place this file in the classpath outside the src folder.
  • Place the below available files source codes in the corresponding packages.
  • Execute the TestSpringTransactionManagement.java by selecting the option Run as Java Application.


EmployeeDAO.java


package com.gaurav.springtransaction.dao;

import java.util.Date;
import java.util.List;

import com.gaurav.springtransaction.bean.EmployeeData;

public interface EmployeeDAO {

      /**
       * This method is used to insert a new record in the employee_data and
       * attendence_details tables.
       */
      public void insertNewRow(String firstName, String lastName, Double salary,
                  Integer no_of_leave, Integer no_of_working_days, Date joining_date);

      /**
       * This method is used to retrieve all the employees records as list
       * employee_data and attendence_details tables.
       */
      public List<EmployeeData> retrieveEmployeeList();
}


EmployeeData.java


package com.gaurav.springtransaction.bean;

import java.util.Date;

public class EmployeeData {
      private int empId;
      private int eid;
      private String firstName;
      private String lastName;
      private double salary;
      private int noOfLeave;
      private int noOfWorkingDays;
      private Date joiningDate;

      /**
       * @return the empId
       */
      public int getEmpId() {
            return empId;
      }

      /**
       * @param empId
       *            the empId to set
       */
      public void setEmpId(int empId) {
            this.empId = empId;
      }

      /**
       * @return the eid
       */
      public int getEid() {
            return eid;
      }

      /**
       * @param eid
       *            the eid to set
       */
      public void setEid(int eid) {
            this.eid = eid;
      }

      /**
       * @return the firstName
       */
      public String getFirstName() {
            return firstName;
      }

      /**
       * @param firstName
       *            the firstName to set
       */
      public void setFirstName(String firstName) {
            this.firstName = firstName;
      }

      /**
       * @return the lastName
       */
      public String getLastName() {
            return lastName;
      }

      /**
       * @param lastName
       *            the lastName to set
       */
      public void setLastName(String lastName) {
            this.lastName = lastName;
      }

      /**
       * @return the salary
       */
      public double getSalary() {
            return salary;
      }

      /**
       * @param salary
       *            the salary to set
       */
      public void setSalary(double salary) {
            this.salary = salary;
      }

      /**
       * @return the noOfLeave
       */
      public int getNoOfLeave() {
            return noOfLeave;
      }

      /**
       * @param noOfLeave
       *            the noOfLeave to set
       */
      public void setNoOfLeave(int noOfLeave) {
            this.noOfLeave = noOfLeave;
      }

      /**
       * @return the noOfWorkingDays
       */
      public int getNoOfWorkingDays() {
            return noOfWorkingDays;
      }

      /**
       * @param noOfWorkingDays
       *            the noOfWorkingDays to set
       */
      public void setNoOfWorkingDays(int noOfWorkingDays) {
            this.noOfWorkingDays = noOfWorkingDays;
      }

      /**
       * @return the joiningDate
       */
      public Date getJoiningDate() {
            return joiningDate;
      }

      /**
       * @param joiningDate
       *            the joiningDate to set
       */
      public void setJoiningDate(Date joiningDate) {
            this.joiningDate = joiningDate;
      }

}

EmployeeAttendenceDetailsMapper.java


package com.gaurav.springtransaction.mapper;

import java.sql.ResultSet;
import java.sql.SQLException;

import org.springframework.jdbc.core.RowMapper;

import com.gaurav.springtransaction.bean.EmployeeData;

public class EmployeeAttendenceDetailsMapper implements RowMapper {
      @Override
      public EmployeeData mapRow(ResultSet rs, int rowNum) throws SQLException {

            EmployeeData employeeData = new EmployeeData();

            employeeData.setEmpId(rs.getInt("EMP_ID"));
            employeeData.setFirstName(rs.getString("FIRST_NAME"));
            employeeData.setLastName(rs.getString("LAST_NAME"));
            employeeData.setSalary(rs.getDouble("SALARY"));

            employeeData.setEid(rs.getInt("EID"));
            employeeData.setNoOfLeave(rs.getInt("NO_OF_LEAVE"));
            employeeData.setNoOfWorkingDays(rs.getInt("NO_OF_WORKING_DAYS"));
            employeeData.setJoiningDate(rs.getDate("JOINING_DATE"));
            return employeeData;
      }
}


EmployeeDAOImpl.java

package com.gaurav.springtransaction.dao.impl;

import java.util.Date;
import java.util.List;

import javax.sql.DataSource;

import org.springframework.dao.DataAccessException;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionDefinition;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.DefaultTransactionDefinition;

import com.gaurav.springtransaction.bean.EmployeeData;
import com.gaurav.springtransaction.dao.EmployeeDAO;
import com.gaurav.springtransaction.mapper.EmployeeAttendenceDetailsMapper;

public class EmployeeDAOImpl implements EmployeeDAO {
      private JdbcTemplate jdbcTemplate;
      private PlatformTransactionManager transactionManager;

      /**
       * @param DataSource
       *            the DataSource to be inject in jdbcTemplate
       */
      public void setDataSource(DataSource ds) {
            this.jdbcTemplate = new JdbcTemplate(ds);
      }

      /**
       * @param transactionManager
       *            the transactionManager to set
       */
      public void setTransactionManager(
                  PlatformTransactionManager transactionManager) {
            this.transactionManager = transactionManager;
      }

      @Override
      public void insertNewRow(String firstName, String lastName, Double salary,
                  Integer no_of_leave, Integer no_of_working_days, Date joining_date) {

            TransactionDefinition def = new DefaultTransactionDefinition();
            TransactionStatus status = transactionManager.getTransaction(def);

            try {
                  String insertQueryForEmployee = "insert into EMPLOYEE_DATA (FIRST_NAME, LAST_NAME, SALARY) values (?, ?, ?)";

                  Object[] queryParameterForEmployeeQuery = { firstName, lastName,
                              salary };
                  jdbcTemplate.update(insertQueryForEmployee,
                              queryParameterForEmployeeQuery);

                  /**
                   * Get the latest employee id from employee_data table, so that we
                   * can use it for eid in the attendence_details table
                   **/
                  String getMaxIdQuery = "select max(EMP_ID) from EMPLOYEE_DATA";
                  int eid = jdbcTemplate.queryForInt(getMaxIdQuery);

                  String insertQueryForAttendenceDetails = "insert into ATTENDENCE_DETAILS(EID, NO_OF_LEAVE, NO_OF_WORKING_DAYS, JOINING_DATE) "
                              + "values (?, ?, ?, ?)";

                  Object[] queryParameterForAttendenceDetailsQuery = { eid,
                              no_of_leave, no_of_working_days, joining_date };

                  jdbcTemplate.update(insertQueryForAttendenceDetails,
                              queryParameterForAttendenceDetailsQuery);

                  System.out.println("\n");
                  System.out.println("Inserted Query is having First Name = "
                              + firstName + ", Last Name = " + lastName
                              + ", And Joining Date " + joining_date);

                  transactionManager.commit(status);

            } catch (DataAccessException e) {

                  System.out
                              .println("Error occured during record insertion , so needs to rollback");
                  transactionManager.rollback(status);

                  throw e;
            }
            return;
      }

      @Override
      public List<EmployeeData> retrieveEmployeeList() {
            String SQL = "select * from EMPLOYEE_DATA, ATTENDENCE_DETAILS where EMPLOYEE_DATA.EMP_ID = ATTENDENCE_DETAILS.EID";

            @SuppressWarnings("unchecked")
            List<EmployeeData> employeeDataLst = jdbcTemplate.query(SQL,
                        new EmployeeAttendenceDetailsMapper());
            return employeeDataLst;
      }

}

spring-transactionmanagement.xml

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

      <!-- mysql data source connection -->
      <bean id="dataSource"
            class="org.springframework.jdbc.datasource.DriverManagerDataSource">
            <property name="driverClassName" value="com.mysql.jdbc.Driver" />
            <property name="url" value="jdbc:mysql://localhost:3306/test" />
            <property name="username" value="root" />
            <property name="password" value="root" />
      </bean>

      <!-- Spring TransactionManager Initialization -->
      <bean id="transactionManager"
            class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
            <property name="dataSource" ref="dataSource" />
      </bean>

      <!-- Provding a definition to EmployeeData for getting data source and spring transaction manager-->
      <bean id="employeeDAOImplBean" class="com.gaurav.springtransaction.dao.impl.EmployeeDAOImpl">
            <property name="dataSource" ref="dataSource" />
            <property name="transactionManager" ref="transactionManager" />
      </bean>

</beans>


TestSpringTransactionManagement.java


package com.gaurav.springtransaction.caller;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.List;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.gaurav.springtransaction.bean.EmployeeData;
import com.gaurav.springtransaction.dao.impl.EmployeeDAOImpl;

public class TestSpringTransactionManagement {
      public static void main(String[] args) throws ParseException {
            ApplicationContext context = new ClassPathXmlApplicationContext(
                        "spring-transactionmanagement.xml");

            EmployeeDAOImpl employeeDAOImplBean = (EmployeeDAOImpl) context
                        .getBean("employeeDAOImplBean");

            System.out
                        .println("********** Inserting the Records in the EmployeeData table ********** ");
            int noOfWorkingDays = 22;

            employeeDAOImplBean.insertNewRow("Kumar", "Gaurav", 70000.00, 5,
                        (noOfWorkingDays - 5), (new SimpleDateFormat("yyyy-MM-dd")).parse("2007-05-08"));
            employeeDAOImplBean.insertNewRow("Anurag", "Kumar", 30000.00, 12,
                        (noOfWorkingDays - 12), (new SimpleDateFormat("yyyy-MM-dd")).parse("2009-03-27"));
            employeeDAOImplBean.insertNewRow("Riya", "Saxena", 45000.00, 7,
                        (noOfWorkingDays - 7), (new SimpleDateFormat("yyyy-MM-dd")).parse("2011-12-15"));

            System.out
                        .println("\n********** Retrieving all the employee records **********");
            List<EmployeeData> employeeDataLst = employeeDAOImplBean
                        .retrieveEmployeeList();
            for (EmployeeData employee : employeeDataLst) {
                  System.out.println("\n");
                  System.out.print("Employee ID : " + employee.getEmpId());
                  System.out.print(", First Name : " + employee.getFirstName());
                  System.out.print(", Last Name : " + employee.getLastName());
                  System.out.print(", Salary : " + employee.getSalary());
                  System.out.print(", No of Leave : " + employee.getNoOfLeave());
                  System.out.print(", No of Working Days : "
                              + employee.getNoOfWorkingDays());
                  System.out.print(", Joining Date : " + employee.getJoiningDate());

            }
      }
}

Result:-

Sep 29, 2013 12:22:54 AM org.springframework.context.support.AbstractApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@16546ef: display name [org.springframework.context.support.ClassPathXmlApplicationContext@16546ef]; startup date [Sun Sep 29 00:22:54 IST 2013]; root of context hierarchy
Sep 29, 2013 12:22:54 AM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [spring-transactionmanagement.xml]
Sep 29, 2013 12:22:54 AM org.springframework.context.support.AbstractApplicationContext obtainFreshBeanFactory
INFO: Bean factory for application context [org.springframework.context.support.ClassPathXmlApplicationContext@16546ef]: org.springframework.beans.factory.support.DefaultListableBeanFactory@a353ed
Sep 29, 2013 12:22:54 AM org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@a353ed: defining beans [dataSource,transactionManager,employeeDAOImplBean]; root of factory hierarchy
Sep 29, 2013 12:22:54 AM org.springframework.jdbc.datasource.DriverManagerDataSource setDriverClassName
INFO: Loaded JDBC driver: com.mysql.jdbc.Driver
********** Inserting the Records in the EmployeeData table **********


Inserted Query is having First Name = Kumar, Last Name = Gaurav, And Joining Date Tue May 08 00:00:00 IST 2007


Inserted Query is having First Name = Anurag, Last Name = Kumar, And Joining Date Fri Mar 27 00:00:00 IST 2009


Inserted Query is having First Name = Riya, Last Name = Sexena, And Joining Date Thu Dec 15 00:00:00 IST 2011

********** Retrieving all the employee records **********


Employee ID : 1, First Name : Kumar, Last Name : Gaurav, Salary : 70000.0, No of Leave : 5, No of Working Days : 17, Joining Date : 2007-05-08

Employee ID : 2, First Name : Anurag, Last Name : Kumar, Salary : 30000.0, No of Leave : 12, No of Working Days : 10, Joining Date : 2009-03-27

Employee ID : 3, First Name : Riya, Last Name : Saxena, Salary : 45000.0, No of Leave : 7, No of Working Days : 15, Joining Date : 2011-12-15


Below is the data availability status in the corresponding tables after successful program execution:-

EMPLOYEE_DATA





ATTENDENCE_DETAILS