JSP
SQL Mapper Framework
SQL 실행에 필요한 객체와 SQL 실행 후 조회된 데이터를 저장할 객체를 매핑하면PreparedStatement처리와 ResuletSet처리에서 자동으로 값을 매핑시켜서 SQL를 실행한다.
mybatis의 내장 타입 별칭
Integer -> int, Long -> long, Double -> double, String -> string,
Date -> date, Map -> map, HashMap - > hashmap, ArrayList -> arraylist
BigDecimal -> bigdecimal
mapper 파일 작성 규칙
<select id="쿼리 식별자" parameterType="전체클래스명 혹은 별칭" resultType="전체클래스명 혹은 별칭">
SELECT 구문
</select>
insert 쿼리
<insert id="쿼리 식별자" parameterType="전체클래스명 혹은 별칭">
INSERT 구문
</insert>
<insert id="쿼리 식별자" parameterType="전체클래스명 혹은 별칭">
<selectKey keyProperty="조회된 값이 저장될 필드명"
resultType="조회된 값의 타입"
order="selectKey구문의 실행 순서(BEFORE, AFTER 중 하나)">
PK로 사용할 값을 조회하는 구문
</selectKey>
INSERT 구문
</insert>
Primary Key를 조회해서 먼저 필드에 저장하고 insert 작업을 실행
insert작업이 종료되면 객체의 필드에 시퀀스값이 저장되어 있다.
작성예)
Order order = new Order();
order.setUserNo(user.getNo());
order.setCreateDate(new Date());
---------------------------------------> Order [no=0, userNo=20, creatDate=2020-01-20]
<insert id="insertOrder" parameterType="Order">
<selectKey keyProperty="no"
resultType="int"
order="BEFORE">
select bookstore_orders_seq.nextval from dual
</selectKey>
insert into (order_no, user_no, create_date)
values (#{no}, #{userNo}, #{createDate})
</insert>
orderDao.insertOrder(order);
---------------------------------------> Order [no=100, userNo=20, creatDate=2020-01-20]
OrderItem item = new OrderItem();
item.setOrderNo(order.getNo()); // 시퀀스로 획득한 값이 Order에 저장되어 있기 때문에
item.setBookNo(100000) // 현재 저장된 주문번호가 필요하면 Order객체에서 조회하면 된다.
item.setAmount(2);
${} 표현식을 사용한 문자열 대체하기
<select id="getAllBooks" parameterType="map" resultType="Book">
select *
from bookstore_books
order by ${columnName} ${sortType}
</select>
다이나믹 SQL
mybatis는 ibatis나 다른 SQL Mapper 프레임워크보다 훨씬 강력한 동적 SQL 작성기능을 제공한다.
mybatis의 동적 SQL 관련 태그는 JSTL의 태그와 사용법이 유사하다.
태그 종류
if 태그
<if test="조건식">
조건식의 결과가 참인 경우 실행될 SQL 구문
</if>
if 태그 사용시 주의점
문자열 비교에서 한 글자 비교는
<if test='grade == "A"'>
와 같은 형식으로 조건식을 작성한다.
두 개 이상의 조건식을 사용될 때는 논리연산자가 필요하다.
논리연산자는 and나 or로 표시하면 된다.
<choose>
<when test="조건식1">
결과가 참인 경우 실행될 SQL 구문
</when>
<when test="조건식2">
조건식2의 결과가 참인 경우 실행될 SQL 구문
</when>
<otherwise>
제시된 조건을 만족하지 못할 경우 실행될 SQL 구문
</otherwise>
</choose>
<foreach index="인덱스" item="변수명" collection="프로퍼티명" separator="구분문자" open="반복시작 전 표시할 내용" close="반복종료 후 표시할 내용">
#{변수명}
</foreach>
where 태그
<where>
if태그, choose,when,otherwise 등의 태그를 포함하고 있다.
</where>
```
* <where>태그 내부에서 동적 SQL의 실행결과로 SQL이 반환되면 where태그를 추가한다.
<where> 태그를 사용하면 아래와 같은 경우가 방지할 수 있다.
```sql
select *
from tables
where
<where>태그 내부에서 동적 SQL의 실행결과로 반환된 SQL에 'and'나 'or'가 있으면 지워버린다.
아래의 동적쿼리에서 startDate가 null이고 endDate가 null이 아닌경우
아래와 같은 경우일 때 and를 제거한다.
select *
from tables
where
and event_date <= '2019-12-31'
<where>
<if test="startDate != null">
event_date >= #{startDate}
</if>
<if test="endDate != null">
and event_date <= #{endDate}
</if>
<if test="eventType != null">
and event_type = #{eventType}
</if>
</where>
set 태그
update bookstore_user
<set>
<if test="point > 0">
user_point = #{point},
</if>
<if test="grade != null">
user_grade = #{grade},
</if>
<if test="filename != null">
user_image_filename = #{filename},
</if>
<if test="enabled != null">
user_enabled = #{enabled}
</if>
</set>
where
user_no = #{no}
두 개 이상의 테이블을 조인하는 경우
<resultMap id="식별자" type="클래스의 별칭 혹은 클래스의 전체이름">
<id property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<association property="객체의 변수명" javaType="클래스의 별칭 혹은 클래스의 전체이름">
<id property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
</assocication>
<collection property="객체의 변수명" ofType="클래스의 별칭 혹은 클래스의 전체이름">
<id property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
<result property="객체의 변수명" column="컬럼명 혹은 컬럼의 별칭" />
</collection>
</result>
class Dept { class Manager { class Emp {
int id; int id; int id;
String name; String name; String firstName;
Manager mgr; } }
List<Emp> employees;
}
<resultMap id="DeptDetail" type="Dept">
<id property="id" column="dept_id" />
<result property="name" column="dept_name"/>
<association property="mgr" javaType="Manager">
<id property="id" column="mgr_id" />
<result property="name" column="mgr_name"/>
</association>
<collection property="employees" ofType="Emp">
<id property="id" column="emp_id"/>
<result property="firstName" column="emp_name"/>
</collection>
</resultMap>
<select id="complexSql" parameterType="int" resultMap="DeptDetail">
select
A.department_id dept_id,
A.department_name dept_name,
B.employee_id mgr_id,
B.first_name mgr_name,
C.employee_id emp_id,
C.first_name emp_name
from departments A, employees B, employees C
where A.manager_id = B.employee_id
and A.department_id = C.department_id
and A.department_id = #{value}
</select>
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"></property>
<property name="configLocation" value="classpath:/META-INF/mybatis/mybatis-config.xml" />
<property name="mapperLocations" value="classpath:/META-INF/mybatis/mappers/*.xml" />
</bean>
3. ConnectionPool, mybatis 환경설정 파일, mybatis 매퍼파일 경로 주입
4. MapperScannerConfigurer 빈 등록
<mybatis-spring:scan base-package="kr.co.jhta.xxx.dao" factory-ref="sqlSessionFactory"/>
* base-package는 xxxDao 인터페이스가 정의되어 있는 패키지를 지정한다.
* factory-ref에는 mybatis의 핵심객체 SqlSessionFactory를 주입한다.
* <mybatis-spring:scan> 태그는 MapperScannerConfigurer을 스프링의 빈으로 등록한다.
* MapperScannerConfigurer은 base-package로 지정된 패키지에서 xxxDao 인터페이스를 전부 스캔해서 xxxDao 인터페이스에 대한
구현객체를 자동으로 생성하고, 생성된 구현객체를 스프링의 빈으로 등록한다.
xxxServiceImpl에서 xxxDao 주입받기
@Service
public class xxxServiceImpl implements xxxService {
@AutoWired
xxxDao dao; // xxxDao인터페이스 타입의 필드를 정의하고
// @AutoWired 어노테이션을 부착해두면
// 스프링 컨테이너는 보유하고 있는 빈들 중에서
// xxxDao 타입의 객체를 주입한다.
// xxxDao 타입의 객체는 MapperScannerConfigurer이 구현해서 등록한 객체다.
}
<?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="products">
<!--
새 상품정보를 추가하는 SQL 매핑정보 정의하기
1. INSERT 구문을 정의한다.
2. INSERT 구문의 식별자 이름을 "insertProduct"로 정의한다.
3. INSERT 구문을 실행하는 필요한 값은 com.sample.vo.Product객체로 전달받는다.
4. INSERT 구문 실행시 #{}으로 표시된 곳에 실제값이 치환된다.
* #{name}은 com.sample.voProduct객체의 name 변수에 저장된 값이 치환된다
* #{maker}은 com.sample.voProduct객체의 maker 변수에 저장된 값이 치환된다
* #{category}은 com.sample.voProduct객체의 category 변수에 저장된 값이 치환된다
* #{price}은 com.sample.voProduct객체의 price 변수에 저장된 값이 치환된다
* #{discountPrice}은 com.sample.voProduct객체의 discountPrice 변수에 저장된 값이 치환된다
public void insertProduct(Porduct product) {
sqlSession.insert("insertProduct", product);
}
-->
<insert id="insertProduct" parameterType="com.sample.vo.Product">
INSERT INTO sample_products
(product_no, product_name, product_maker, product_category, product_price, product_discount_price)
VALUES
(product_seq.nextval, #{name}, #{maker}, {category}, #{price}, #{discountPrice})
</insert>
<!--
상품정보를 삭제하는 SQL 매핑정보 정의하기
1. DELETE 구문을 정의한다.
2. DELETE 구문의 식별자 이름을 "deleteProduct"로 정의한다.
3. DELETE 구문을 실행하는 필요한 값은 정수값 하나다
4. DELETE 구문 실행시 #{}으로 표시된 곳에 실제값이 치환된다.
* #{value}에 정수값이 치환된다.
public void deleteProduct(int no) {
sqlSession.insert("deleteProduct", 10);
}
-->
<delete id="deleteProduct" parameterType="int">
DELETE FROM sample_products
WHERE product_no = #{value}
</delete>
<!--
상품정보를 변경하는 SQL 매핑정보 정의하기
1. UPDATE 구문을 정의한다.
2. UPDATE 구문의 식별자 이름을 "updateProduct"로 정의한다.
3. UPDATE 구문을 실행하는 필요한 값은 com.sample.vo.Product객체로 전달한다
4. UPDATE 구문 실행될 때 #{}가 실제값을 치환된다
* #{변수명}에서 변수명은 수정하는데 필요한 값을 가지고 있는 Product객체의 멤버변수 이름이다.
public void updateProduct(Product product) {
sqlSession.update("updateProduct", product);
}
-->
<update id="updateProduct" parameterType="com.sample.vo.Product">
UPDATE sample_products
SET
product_name = #{name},
product_maker = #{maker},
product_category = #{category},
product_price = #{price},
product_discount_price = {discountPrice},
product_stock = #{stock},
product_sold_out = #{soldOut}
WHERE product_no = #{no}
</update>
<!--
모든 상품정보를 조회하는 SQL 매핑구문 정의하기
1. SELECT 구문을 정의한다
2. SELECT 구문을 식별하는 이름은 "getAllProducts"다
3. 이번 SELECT 구문은 치환시킬 값이 필요없어서 prrameterType은 생략했다.
4. SELECT 구문의 실행결과는 resultType으로 지정한 com.sample.vo.Product객체를 생성해서 저장한다
5. SELECT 구문의 실행결과가 com.sample.vo.Product객체에 저장될 떄는 컬럼명과 동일한 이름을 가진 멤버변수에 저장된다.
6. SELECT 구문의 실행결과가 여러 행만 조회된다면 selectList() 메소드를 사용한다.
* selectList() 메소드는 조회된 행의 갯수만큼 Product객체를 생성해서 각 행의 값을 저장한다.
* 생성된 모든 Product객체는 List객체에 담긴다.
7. SELECT 구문이 실행이 완료되면 Product객체가 여러개가 담긴 List객체가 반환된다.
public List<Product> getAllProduct() {
List<Product> products = sqlSession.selectList("getAllProducts");
}
-->
<select id="getAllProducts" resultType="com.sample.vo.Product">
SELECT product_no as no,
product_name as name,
product_maker as maker,
product_category as category,
product_price as price,
product_discount_price as discountPrice,
product_stock as stock,
product_sold_out as soldOut,
product_created_date as createdDate
FROM sample_products
ORDER BY product_no DESC
</select>
<!--
한 상품정보를 조회하는 SQL 매핑구문 정의하기
1. SELECT 구문을 정의한다
2. SELECT 구문을 식별하는 이름은 "getAllProducts"다
3. SELECT 구문을 실행하는데 필요한 값은 정수값이다
4. SELECT 구문이 실행될 떄, #{value}에 정수값이 치환된다.
5. SELECT 구문의 실행결과는 resultType으로 지정한 com.sample.vo.Product객체를 생성해서 저장한다
6. SELECT 구문의 실행결과가 com.sample.vo.Product객체에 저장될 떄는 컬럼명과 동일한 이름을 가진 멤버변수에 저장된다.
7. SELECT 구문의 실행결과가 한 행만 조회된다면 selectOne() 메소드를 사용한다.
8. SELECT 구문이 실행이 완료되면 상품정보가 저장된 Product객체 한 개가 반환된다.
public Product getProductByNo(int no) {
Product product = sqlSession.selectOne("getProductByNo", no);
return product;
}
-->
<select id="getProductByNo" parameterType="int" resultType="com.sample.vo.Product">
SELECT product_no as no,
product_name as name,
product_maker as maker,
product_category as category,
product_price as price,
product_discount_price as discountPrice,
product_stock as stock,
product_sold_out as soldOut,
product_created_date as createdDate
FROM sample_products
WHERE product_no = #{value}
</select>
<select id="getProductByName" parameterType="string" resultType="com.sample.vo.Product">
SELECT product_no as no,
product_name as name,
product_maker as maker,
product_category as category,
product_price as price,
product_discount_price as discountPrice,
product_stock as stock,
product_sold_out as soldOut,
product_created_date as createdDate
FROM sample_products
WHERE product_name like '%' || #{value} || '%'
</select>
</mapper>