ตัวอย่างการเขียน Spring-boot WebFlux JdbcTemplate
pom.xml
...
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.1.5.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-webflux</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-core</artifactId>
<version>5.4.3.Final</version>
</dependency>
<dependency>
<groupId>org.postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>42.2.5</version>
</dependency>
<dependency>
<groupId>com.zaxxer</groupId>
<artifactId>HikariCP</artifactId>
<version>3.3.1</version>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<scope>provided</scope>
</dependency>
</dependencies>
...spring-boot-starter-webfluxใช้สำหรับเขียน webfluxspring-boot-starter-data-jpaไว้สำหรับเขียนคำสั่ง query, method queryhibernate-coreสำหรับทำ ORM (Object Relational Mapping) ไว้เขียนพวก entity class สำหรับ mapping java class ไปยัง database table รวมถึงการ mapping พวก relation ต่าง ๆ ของ table เช่น One to One, One to Many, Many to Manypostgresqlเป็น postgresql database driverHikariCPเป็นตัวจัดการ database connection poollombokเป็น annotation code generator สามารถ generate code at compile time ได้ ทำให้เราไม่ต้องเขียน code บางส่วนเอง เช่น getter setter method ตัว lombox จะทำให้
@SpringBootApplication
@ComponentScan(basePackages = {"com.pamarin"})
public class AppStarter {
public static void main(String[] args) {
SpringApplication.run(AppStarter.class, args);
}
}@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Entity
@Table(name = User.TABLE_NAME)
public class User implements Serializable {
public static final String TABLE_NAME = "user";
@Id
private String id;
@Column(name = "username", nullable = false, unique = true)
private String username;
@Column(name = "password", nullable = false)
private String password;
}@Dataเป็น annotation ของ lombox เอาไว้ generate code เช่น getter/setter method, hashcode + equals ให้@Entityเป็น annotation ที่เอาไว้ระบุว่า class นี้เป็น entity class@Tableเป็น annotation ที่เอาไว้ระบุว่าให้ class นี้ map ไปที่ database table ใด@Idเป็น annotation ที่เอาไว้ระบุว่าจะให้ attribute ใดเป็น primary key@Columnเป็นการใช้ระบุข้อมูล column
ประกาศ interface
public interface UserRepository {
List<User> findAll();
Optional<User> findById(String id);
User save(User user);
void deleteAll();
void deleteById(String id);
}implement interface
@Slf4j
@Repository
public class UserRepositoryImpl implements UserRepository {
private final JdbcTemplate jdbcTemplate;
private final String schema;
@Autowired
public UserRepositoryImpl(
DataSource dataSource,
@Value("${spring.jpa.properties.hibernate.default_schema}") String schema
) {
this.jdbcTemplate = new JdbcTemplate(dataSource);
this.schema = schema;
}
private String schema(String table) {
return schema + "." + table;
}
private String sql(String sql) {
return String.format(sql, schema(User.TABLE_NAME));
}
@Override
public List<User> findAll() {
try {
return jdbcTemplate.queryForObject(
sql("SELECT * FROM %s"),
(ResultSet rs, int i) -> {
List<User> list = new ArrayList<>();
do {
list.add(convertToEntity(rs));
} while (rs.next());
return list;
});
} catch (EmptyResultDataAccessException ex) {
return Collections.emptyList();
}
}
@Override
public Optional<User> findById(String id) {
try {
return Optional.ofNullable(jdbcTemplate.queryForObject(
sql("SELECT * FROM %s WHERE id = ?"),
new Object[]{id},
(ResultSet rs, int i) -> convertToEntity(rs)
));
} catch (EmptyResultDataAccessException ex) {
return Optional.empty();
}
}
private User convertToEntity(ResultSet rs) throws SQLException {
return User.builder()
.id(rs.getString("id"))
.username(rs.getString("username"))
.password(rs.getString("password"))
.build();
}
private boolean existsById(String id) {
try {
return jdbcTemplate.queryForObject(
sql("SELECT COUNT(id) FROM %s WHERE id = ?"),
new String[]{id},
String.class
) != null;
} catch (EmptyResultDataAccessException ex) {
return false;
}
}
private void insert(User user) {
jdbcTemplate.update(
sql("INSERT INTO %s (id, username, password) VALUES (?, ?, ?)"),
new Object[]{
user.getId(),
user.getUsername(),
user.getPassword()
}
);
}
private void update(User user) {
jdbcTemplate.update(
sql("UPDATE %s SET username = ?, password = ? WHERE id = ?"),
new Object[]{
user.getUsername(),
user.getPassword(),
user.getId()
}
);
}
private String randomId() {
return UUID.randomUUID().toString();
}
@Override
public User save(User user) {
Assert.notNull(user, "require user.");
if (hasText(user.getId())) {
if (existsById(user.getId())) {
update(user);
} else {
insert(user);
}
} else {
user.setId(randomId());
insert(user);
}
return findById(user.getId()).get();
}
@Override
public void deleteAll() {
jdbcTemplate.update(sql("DELETE FROM %s"));
}
@Override
public void deleteById(String id) {
jdbcTemplate.update(sql("DELETE FROM %s WHERE id = ?"), id);
}
}@RestController
public class UserController {
private final UserRepository userRepository;
@Autowired
public UserController(UserRepository userRepository) {
this.userRepository = userRepository;
}
@GetMapping({"", "/"})
public Flux<User> home() {
return findAll();
}
@GetMapping("/users")
public Flux<User> findAll() {
return Flux.fromIterable(userRepository.findAll());
}
@GetMapping("/users/{id}")
public Mono<User> findById(@PathVariable("id") String id) {
return Mono.justOrEmpty(userRepository.findById(id))
.switchIfEmpty(Mono.error(new NotFoundException("Not found user of id " + id)));
}
@PostMapping("/users")
public Mono<User> save(@RequestBody User user) {
return Mono.just(userRepository.save(user));
}
@DeleteMapping("/users")
public Mono<Void> deleteAll() {
userRepository.deleteAll();
return Mono.empty();
}
@DeleteMapping("/users/{id}")
public Mono<Void> deleteById(@PathVariable("id") String id) {
userRepository.deleteById(id);
return Mono.empty();
}
}#------------------------------------ JPA --------------------------------------
spring.jpa.hibernate.ddl-auto=none
spring.jpa.properties.hibernate.cache.use_second_level_cache=false
spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.PostgreSQLDialect
spring.jpa.hibernate.use-new-id-generator-mappings=true
spring.jpa.show-sql=true
spring.jpa.properties.hibernate.proc.param_null_passing=true
spring.jpa.properties.hibernate.default_schema=*****
#------------------------------------ Hikari -----------------------------------
spring.datasource.hikari.minimumIdle=1
spring.datasource.hikari.maximumPoolSize=10
spring.datasource.hikari.idleTimeout=30000
spring.datasource.hikari.connectionTestQuery=SELECT 1 FROM DUAL
spring.datasource.hikari.validationTimeout=3000
#------------------------------------ Postgresql -------------------------------
spring.datasource.url=jdbc:postgresql:*****
spring.datasource.username=*****
spring.datasource.password=*****
spring.datasource.driver-class-name=org.postgresql.Driver
spring.datasource.platform=postgres
spring.datasource.type=org.postgresql.ds.PGSimpleDataSourcecd ไปที่ root ของ project จากนั้น
$ mvn clean install$ mvn spring-boot:run \
-Dserver.port=8080 \
-Dspring.datasource.url=jdbc:postgresql://<HOST>:<PORT>/<DATABASE_NAME>?sslmode=require \
-Dspring.datasource.username=<DATABASE_USERNAME> \
-Dspring.datasource.password=<DATABASE_PASSWORD> \
-Dspring.jpa.properties.hibernate.default_schema=<DATABASE_SCHEMA>ให้เปลี่ยน ค่า <> เป็นของตัวเองน่ะครับ
- HOST คือ ip หรือ domain name ของ database server
- PORT คือ port ที่ใช้
- DATABASE_NAME คือ ชื่อ database
- DATABASE_USERNAME คือ ชื่อ username ที่ login เข้าใช้งาน database
- DATABASE_PASSWORD คือ รหัสผ่านที่คู่กับ username ที่ใช้
- DATABASE_SCHEMA คือ database schema ที่่ใช้
เปิด browser แล้วเข้า http://localhost:8080