개발/Spring boot
MapperScan을 통한 Spring boot DB 연결 - Oracle
henry.hong
2019. 12. 10. 09:58
작업자의 작업 환경
- Intellij
- Oracle 11g
- Windows 10
File > New Project > Spring Initializer 선택
Project Type은 Gradle Project로 설정합니다.
Spring Web, MyBatis Framework, Oracle Driver 3개를 선택하고 프로젝트 셋팅을 해줍니다.
필요에 따라 dependencies를 추가해줍니다.
compile 'org.springframework:spring-web:5.2.0.RELEASE'
compile group: 'org.springframework', name: 'spring-webmvc', version: '5.2.0.RELEASE'
compile 'org.springframework.boot:spring-boot-starter'
implementation 'org.springframework.boot:spring-boot-starter-web'
implementation 'org.mybatis.spring.boot:mybatis-spring-boot-starter:2.1.1'
compileOnly 'org.projectlombok:lombok'
runtimeOnly 'mysql:mysql-connector-java'
annotationProcessor 'org.projectlombok:lombok'
testImplementation('org.springframework.boot:spring-boot-starter-test') {
exclude group: 'org.junit.vintage', module: 'junit-vintage-engine'
}
//boot 자동빌드
compile 'org.springframework.boot:spring-boot-devtools'
//mybatis
compile 'org.mybatis.spring.boot:mybatis-spring-boot-starter'
compile 'org.springframework.boot:spring-boot-starter-jdbc'
//oracle
runtimeOnly 'com.oracle.ojdbc:ojdbc8'
Yml 파일에 대해 설정합니다.
spring:
profiles:
active: dev
datasource:
db1:
driver-class-name: oracle.jdbc.OracleDriver
jdbc-url: jdbc:oracle:thin:@localhost:1521:ORCL
username: db1
password: db1
type: com.zaxxer.hikari.HikariDataSource
hikari:
maximum-pool-size: 10
connection-test-query: SELECT 1 FROM SYS.DUAL
minimum-idle: 5
connection-timeout: 100000
db2:
driver-class-name: oracle.jdbc.OracleDriver
jdbc-url: jdbc:oracle:thin:@localhost:1521:orcl
username: db2
password: db2
type: com.zaxxer.hikari.HikariDataSource
hikari:
maximum-pool-size: 10
connection-test-query: SELECT 1 FROM SYS.DUAL
minimum-idle: 5
connection-timeout: 100000
devtools:
livereload:
enabled: true
server:
port: 8080
DB를 두개 연결하기 때문에 db1: 과 db2: 로 구분합니다. DB 한개만 쓰시려면 db2: 이하는 삭제해주세요.
DB1 Controller 설정
package conn.multiple.dbone.controller;
import conn.multiple.dbone.service.OneOracleService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping(value = "/one")
public class OneOracleController {
@Autowired private OneOracleService oneOracleService;
@RequestMapping(value = "/getHostName", method = RequestMethod.GET)
private String getHostName(){
return this.oneOracleService.getHostName();
}
}
DB1 Service 설정
package conn.multiple.dbone.service;
public interface OneOracleService {
String getHostName();
}
package conn.multiple.dbone.service;
import conn.multiple.dbone.dao.OneOracleDao;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Slf4j
@Service
public class OneOracleServiceImpl implements OneOracleService{
@Autowired
private OneOracleDao oneOracleDao;
@Override
public String getHostName() {
log.info("one oracle :: getHostName");
return this.oneOracleDao.getHostName();
}
}
DB1 Repository 설정
package conn.multiple.dbone.dao;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface OneOracleDao {
String getHostName();
}
DB1 Mapper (xml) 설정
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="conn.multiple.dbone.dao.OneOracleDao">
<select id="getHostName" resultType="java.lang.String">
SELECT host_name
FROM v$instance
</select>
</mapper>
구현의 편의를 위해 DB의 host name 정보를 가져옵니다.
DB1 Configure 설정
package conn.multiple.config;
import com.zaxxer.hikari.HikariDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.SqlSessionFactoryBean;
import org.mybatis.spring.SqlSessionTemplate;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import javax.sql.DataSource;
@Configuration
@MapperScan(basePackages = {"conn.multiple.dbone.**.dao"}, sqlSessionFactoryRef = "oracleOneSqlSessionFactory")
public class MybatisDbOneConfig {
@Bean(name = "oracleOneDatasource")
@ConfigurationProperties(prefix = "spring.datasource.db1")
@Primary
public DataSource oracleOneDatasource(){
return DataSourceBuilder.create().type(HikariDataSource.class).build();
}
@Bean(name= "oracleOneSqlSessionFactory")
@Primary
public SqlSessionFactory oracleOneSqlSessionFactory(@Qualifier(value = "oracleOneDatasource") DataSource oracleOneDatasource) throws Exception{
SqlSessionFactoryBean bean = new SqlSessionFactoryBean();
bean.setDataSource(oracleOneDatasource);
PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();
//oracle mapper 설정
bean.setMapperLocations(resolver.getResources("classpath:mapper/oracle/one/**/*.xml"));
return bean.getObject();
}
@Bean(name="oracleOneSqlSessionTemplate")
@Primary
public SqlSessionTemplate oracleOneSqlSessionTemplate(@Qualifier(value = "oracleOneSqlSessionFactory")SqlSessionFactory oracleOneSqlSessionFactory) throws Exception {
return new SqlSessionTemplate(oracleOneSqlSessionFactory);
}
}
DB2를 쓰지 않을거면, @Primary 가 적힌 3부분에 대해 삭제합니다.
@MapperScan에 basePackages 에 있는 conn.multiple.dbone.**.dao 의 경로에 존재하는 dao(repository)만
oracleOneDatasource로 연결합니다.
DB2 Controller 설정
package conn.multiple.dbtwo.controller;
import conn.multiple.dbtwo.service.TwoOracleService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping(value = "/two")
public class TwoOracleController {
@Autowired private TwoOracleService twoOracleService;
@RequestMapping(value = "/getHostName", method = RequestMethod.GET)
private String getHost0Name(){
return this.twoOracleService.getHostName();
}
}
DB2 Service 설정
package conn.multiple.dbtwo.service;
public interface TwoOracleService {
String getHostName();
}
package conn.multiple.dbtwo.service;
import conn.multiple.dbtwo.dao.TwoOracleDao;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Slf4j
@Service
public class TwoOracleServiceImpl implements TwoOracleService {
@Autowired
private TwoOracleDao twoOracleDao;
@Override
public String getHostName() {
log.info("one oracle :: getHostName");
return this.twoOracleDao.getHostName();
}
}
DB2 Repository 설정
package conn.multiple.dbtwo.dao;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface TwoOracleDao {
String getHostName();
}
DB2 Configure 설정
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="conn.multiple.dbtwo.dao.TwoOracleDao">
<select id="getHostName" resultType="java.lang.String">
SELECT host_name
FROM v$instance
</select>
</mapper>