Spring Data

Spring Data

Configuration #

Dependencies

Add Spring dependencies in pom.xml :

<properties>
    ...
    <hibernate.version>4.1.9.Final</hibernate.version>
</properties>

<dependencies>
    ...
    <!-- Spring -->
    ...
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-orm</artifactId>
        <version>${spring.version}</version>
    </dependency>

    <!-- JPA Vendor -->
    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>${hibernate.version}</version>
    </dependency>
</dependencies>

Install and configure database

  • Database type: MySQL
  • Host: localhost
  • Port: 3306
  • Database: hello_spring
  • Login: root
  • Password: root

Configure MySQL Driver

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>5.1.22</version>
</dependency>

Configure Datasource in Spring

For development purposes:

<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/hello_spring"/>
    <property name="username" value="root"/>
    <property name="password" value="root"/>
</bean>

For more datasources possibilities check: Datasources

Configure local Transaction Manager

<!-- Transactions -->
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<!-- enable the configuration of transactional behavior based on annotations -->
<tx:annotation-driven transaction-manager="transactionManager" />

Configure EntityManager

Option 1: without persistence.xml :

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="generateDdl" value="true" />
            <property name="showSql" value="true" />
        </bean>
    </property>
    <property name="packagesToScan" value="io.github.kospiotr.model"/>
    <property name="dataSource" ref="dataSource"/>
</bean>

Option 2: with persistence.xml :

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"/>
    </property>
    <property name="dataSource" ref="dataSource"/>
</bean>

Then META-INF/persistence.xml:

<persistence xmlns="http://java.sun.com/xml/ns/persistence"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://java.sun.com/xml/ns/persistence 
             http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
             version="2.0">
    <persistence-unit name="PU" transaction-type="RESOURCE_LOCAL">
        <properties>
            <property name="hibernate.show_sql" value="true"/>
            <property name="hibernate.format_sql" value="true"/>
            <property name="hibernate.hbm2ddl.auto" value="create-drop"/>
        </properties>
    </persistence-unit>
</persistence>

More configuration properties: https://docs.jboss.org/hibernate/core/4.3/manual/en-US/html_single#configuration-optional

Example Model #

Java model

import javax.persistence.Entity;
import javax.persistence.Id;
 
@Entity
public class Product {
 
    @Id
    private Integer id;
    private String name;
 
    public Product() {
    }
 
    public Product(Integer id, String name) {
        this.id = id;
        this.name = name;
    }
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
 
    @Override
    public String toString() {
        return "Product [id=" + id + ", name=" + name + "]";
    }
 
}

Database schema

No need if hibernate.hbm2ddl.auto='create-drop'.

CREATE TABLE `Product` (
  `id` int(11) unsigned NOT NULL,
  `name` varchar(20) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

Pure JPA #

DAO

@Component
public class ProductRepository {

	@PersistenceContext
	private EntityManager em;

	public Product findById(Integer productId) {
		return em.find(Product.class, productId);
	}

	public List<Product> findAll() {
		return em.createQuery("SELECT p FROM Product p").getResultList();
	}

	public Product create(Product product) {
		em.persist(product);
		return product;
	}

	public Product update(Product product) {
		return em.merge(product);
	}

	public void delete(Integer id) {
		Product toRemove = findById(id);
		if (toRemove == null) {
			throw new RuntimeException("Can't find Product with given id: " + id);
		}
		em.remove(toRemove);
	}
}

Service

@Component
public class ProductService {

	@Autowired
	private ProductRepository productDao;

	@Transactional(readOnly = true)
	public List<Product> findAll() {
		return productDao.findAll();
	}

	public Product findById(Integer productId) {
		return productDao.findById(productId);
	}

	@Transactional
	public Product save(Product product) {
		if (product.getId() == null) {
			return productDao.create(product);
		} else {
			return productDao.update(product);
		}
	}

	@Transactional
	public void delete(Integer id) {
		productDao.delete(id);
	}
}

Spring Data #

What is it for?

Simplify JPA access by removing boilerplate code and introduce higher level of abstraction. Thanks to Spring Data we can focus on business value.

Dependency

<dependencies>
	...
	<dependency>
	    <groupId>org.atteo.moonshine</groupId>
	    <artifactId>spring-data</artifactId>
	    <version>${spring-data.version}</version>
	</dependency>
</dependencies>

Configuration

    <jpa:repositories base-package="io.github.kospiotr.repository"/>

DAO

public interface ProductRepository extends JpaRepository<Product, Integer> {

}

Service

@Component
public class ProductService {

	@Autowired
	private ProductRepository productRepository;

	@Transactional(readOnly = true)
	public List<Product> findAll() {
		return productRepository.findAll();
	}

	public Product findById(Integer productId) {
		return productRepository.findOne(productId);
	}

	@Transactional
	public Product save(Product product) {
		return productRepository.save(product);
	}

	@Transactional
	public void delete(Integer id) {
		productRepository.delete(id);
	}
}

Custom by property

public interface ProductRepository extends JpaRepository<Product, Integer> {

  List<Product> findByName(String name); 
}

Custom JPQL

public interface ProductRepository extends JpaRepository<Product, Integer> {

  @Query("<JPQ statement here>")
  List<Product> findBySomeInputs(Integer age); 
}