Spring Boot 和 JDBCTemplate简介: JDBC Template

Posted Java攻城师

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring Boot 和 JDBCTemplate简介: JDBC Template相关的知识,希望对你有一定的参考价值。

与任何编程语言一样,Java有多种工具可以轻松地在语言和数据库之间进行集成。有几种工具,例如Hibernate,Eclipse Link,JPA规范等等。但是,ORM带来了几个问题,有时使用它,然后再将其用于Java Communication层或JDBC并不有意义。本教程将介绍一种使用Spring JDBC模板简化JDBC代码的方法。

诸如ORM之类的映射框架减少了很多样板,减少了重复的代码,避免了错误,并且没有重新发明轮子。但是,当RDBMS由以面向对象的编程语言编写的应用程序提供服务时,经常会遇到对象关系阻抗不匹配的问题,这会带来一系列概念上和技术上的困难。

一个解决方案可能使用JDBC,但是它增加了处理数据和Java的复杂性。应用程序如何减少JDBC的冗长性?Spring JDBCTemplate是一种强大的机制,可以连接到数据库并执行SQL查询。它在内部使用JDBC API,但消除了JDBC API的许多问题。

从Spring Initializr开始

此示例应用程序将结合使用JDBCTemplate和Spring来使用两个数据库:PostgreSQL用于运行该应用程序和H2在测试范围内。对于所有Spring应用程序,您应该从Spring Initializr开始。Initializr提供了一种快速的方法来提取应用程序所需的所有依赖项,并为您完成了许多设置。此示例需要JDBC API,Spring MVC,PostgreSQL驱动程序和H2数据库依赖项。

<dependency> 
    <groupId>org.springframework.boot</groupId> 
    <artifactId>spring-boot-starter-web</artifactId> 
</dependency> 
<dependency> 
    <groupId>org.springframework.boot</groupId> 
    <artifactId>spring-boot-starter-jdbc</artifactId> 
</dependency> 
<dependency> 
    <groupId>org.postgresql</groupId> 
    <artifactId>postgresql</artifactId> 
    <scope>runtime</scope> 
</dependency> 
<dependency> 
    <groupId>com.h2database</groupId> 
    <artifactId>h2</artifactId> 
    <scope>test</scope> 
</dependency> 
<dependency> 
    <groupId>org.springframework.boot</groupId> 
    <artifactId>spring-boot-starter-test</artifactId> 
   <scope>test</scope> 
</dependency> 

下一步是Car实体,其中包含五个字段:id,名称,城市,型号和颜色。在这个项目中,作为贫血模型的POJO足以应付它。丰富的模型可以保证对象的规则并避免任何封装问题,因此,它是防弹API。但是,它仅适用于复杂的项目。突出显示:当我们谈论软件体系结构时,“取决于”始终是一个很好的答案。

public class Car {
    private Long id;
    private String name;
    private String city;
    private String model;
    private String color;
    //...
    public static CarBuilder builder() {
        return new CarBuilder();
    }
}

为了在Java和JDBC的ResultSet之间实现紧密集成,Spring JDBC具有RowMapper接口。开发人员可以创建一个自定义类,也可以使用BeanPropertyRowMapper,从而减少了样板。但是,该实体应具有公共获取者和设置者;除了提供便利而不是高性能之外,它可能还会遇到封装问题。为了获得最佳性能,请考虑使用自定义RowMapper实现。

public class CarRowMapper implements RowMapper<Car> {
    @Override
    public Car mapRow(ResultSet resultSet, int rowNum) throws SQLException {
        Long id = resultSet.getLong("ID");
        String name = resultSet.getString("name");
        String city = resultSet.getString("city");
        String model = resultSet.getString("model");
        String color = resultSet.getString("color");
        return Car.builder().id(id).name(name)
                .city(city)
                .model(model)
                .color(color).build();
    }
}

实体已准备就绪;让我们谈谈数据访问对象DAO;每次要处理庞大或复杂的查询时,都会讨论SQL查询的去向。简而言之,当脚本经过硬编码后,脚本的工作就变得更清楚了,但是当命令量很大时,阅读和理解就变得很有挑战性。因此,我们可以移动它以从属性中读取它。

car.query.find.by.id=SELECT * FROM CAR WHERE ID = :id
car.query.delete.by.id=DELETE FROM CAR WHERE ID =:id
car.query.update=update CAR set name = :name, city = :city, model= :model, color =:color  where id = :id
car.query.find.all=select * from CAR ORDER BY id LIMIT :limit OFFSET :offset

一旦读者熟悉了代码中的查询,我们将在属性文件中探索查询选项。我们将有一个班级来承担这项职责,并且它将与Spring Configuration紧密集成。

@Component
public class CarQueries {
    @Value("${car.query.find.by.id}")
    private String findById;
    @Value("${car.query.delete.by.id}")
    private String deleteById;
    @Value("${car.query.update}")
    private String update;
    @Value("${car.query.find.all}")
    private String findAll;
    //...
}

一旦findAll具有分页支持,则CarDAO将执行CRUD操作。它使用NamedParameterJdbcTemplate对一组基本的JDBC操作进行分类,从而允许使用命名参数而不是传统的“?” 占位符。为了避免连接泄漏,Spring具有Transactional批注,以控制我们在代码中定义的每个方法中的事务。

@Repository
public class CarDAO {
    private final NamedParameterJdbcTemplate template;
    private final CarQueries queries;
    private final RowMapper<Car> rowMapper;
    private final SimpleJdbcInsert insert;
    @Autowired
    public CarDAO(NamedParameterJdbcTemplate template, CarQueries queries) {
        this.template = template;
        //this.rowMapper = new BeanPropertyRowMapper<>(Car.class);
        this.rowMapper = new CarRowMapper();
        this.queries = queries;
        this.insert = new SimpleJdbcInsert(template.getJdbcTemplate());
        this.insert.setTableName("car");
        this.insert.usingGeneratedKeyColumns("id");
    }
    @Transactional
    public Car insert(Car car) {
        //Number id = insert.executeAndReturnKey(new BeanPropertySqlParameterSource(car));
        Number id = insert.executeAndReturnKey(car.toMap());
        return findBy(id.longValue()).orElseThrow(() -> new IllegalStateException(""));
    }
    public Optional<Car> findBy(Long id) {
        String sql = queries.getFindById();
        Map<String, Object> parameters = Collections.singletonMap("id", id);
        return template.queryForStream(sql, parameters, rowMapper).findFirst();
    }
    @Transactional
    public boolean delete(Long id) {
        String sql = queries.getDeleteById();
        Map<String, Object> paramMap = Collections.singletonMap("id", id);
        return template.update(sql, paramMap) == 1;
    }
    @Transactional
    public boolean update(Car car) {
        String sql = queries.getUpdate();
        Map<String, Object> paramMap = car.toMap();
        return template.update(sql, paramMap) == 1;
    }
    public Stream<Car> findAll(Page page) {
        String sql = queries.getFindAll();
        Map<String, Object> paramMap = new HashMap<>();
        paramMap.put("limit", page.getLimit());
        paramMap.put("offset", page.getOffset());
        return template.queryForStream(sql, paramMap, rowMapper);
    }
}

代码已准备就绪;让我们测试一下。Yeap是一种TDD技术,其理念是从测试开始,然后创建代码。但这不是本文的目标。在测试范围内,我们将使用H2在内存中生成一个DBMS。Spring具有多项功能,可让我们顺利进行测试。多亏了Spring,我们可以在测试中使用H2,而不会影响将在生产环境中运行的驱动程序。

@ExtendWith(SpringExtension.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
class CarDAOTest {
    @Autowired
    private CarDAO carDAO;
    @Autowired
    private JdbcTemplate template;
    @Test
    public void shouldFindById() {
        Assertions.assertNotNull(carDAO);
        Optional<Car> car = carDAO.findBy(1L);
        Assertions.assertNotNull(car);
    }
    @Test
    public void shouldInsertCar() {
        Car car = Car.builder()
                .city("Salvador")
                .color("Red")
                .name("Fiat")
                .model("Model")
                .build();
        Car insert = carDAO.insert(car);
        Assertions.assertNotNull(insert);
        Assertions.assertNotNull(insert.getId());
    }
    @Test
    public void shouldDelete() {
        Car car = Car.builder()
                .city("Salvador")
                .color("Red")
                .name("Fiat")
                .model("Model")
                .build();
        Car insert = carDAO.insert(car);
        carDAO.delete(insert.getId());
        Optional<Car> empty = carDAO.findBy(insert.getId());
        Assertions.assertTrue(empty.isEmpty());
    }
    @Test
    public void shouldUpdate() {
        Car car = Car.builder()
                .city("Salvador")
                .color("Red")
                .name("Fiat")
                .model("Model")
                .build();
        Car insert = carDAO.insert(car);
        insert.update(Car.builder()
                .city("Salvador")
                .color("Red")
                .name("Fiat")
                .model("Update")
                .build());
        carDAO.update(insert);
    }
    @Test
    public void shouldFindAll() {
        template.execute("DELETE FROM CAR");
        List<Car> cars = new ArrayList<>();
        for (int index = 0; index < 10; index++) {
            Car car = Car.builder()
                    .city("Salvador")
                    .color("Red")
                    .name("Fiat " + index)
                    .model("Model")
                    .build();
            cars.add(carDAO.insert(car));
        }
        Page page = Page.of(1, 2);
        List<Car> result = carDAO.findAll(page).collect(Collectors.toList());
        Assertions.assertEquals(2, result.size());
        MatcherAssert.assertThat(result, Matchers.contains(cars.get(0), cars.get(1)));
        Page nextPage = page.next();
        result = carDAO.findAll(nextPage).collect(Collectors.toList());
        Assertions.assertEquals(2, result.size());
        MatcherAssert.assertThat(result, Matchers.contains(cars.get(2), cars.get(3)));
    }
}

结论

在本教程中,我们介绍了Spring JDBC及其操作。除了测试和配置资源外,我们还讨论了一些映射折衷以及存储查询的位置。Spring具有一些可提高开发人员生产率的功能。在第二部分中,我们将讨论有关Spring MVC及其在数据库中的工作。Java学习群,免费领取资料:3907814

代码:https : //github.com/xgeekshq/spring-boot-jdbc-template-sample

以上是关于Spring Boot 和 JDBCTemplate简介: JDBC Template的主要内容,如果未能解决你的问题,请参考以下文章

Spring_11-Spring5总结

Spring--jdbcTemplate

阶段3 2.Spring_10.Spring中事务控制_7 spring基于注解的声明式事务控制

为啥 Spring Boot 应用程序 pom 同时需要 spring-boot-starter-parent 和 spring-boot-starter-web?

Spring Boot Maven 插件 - spring-boot:run 和优雅关闭

spring-boot-starter-web 和 spring-boot-starter-webflux 不能一起工作吗?