유로파니 2020. 9. 18. 15:51

● 매퍼

XML 사용시 : SQL을 XML에 정의하는 매퍼 설정파일(context-mapper.xml)

어노테이션 사용시 : 인터페이스의 메소드마다 어노테이션을 정의한 매퍼 인터페이스

 

Mybatis의 특징

- 자원 생성 해제 필요 X

- 파라미터, 결과셋 설정 코드 X

- SQL문을 따로 관리(xml or 인터페이스)

- 구성요소 5가지 : 설정파일, 매퍼, 파라미터타입(parameterType), 결과타입(resultType), 결과매핑(resultMap)

- 파라미터, 결과 타입 : Map객체, Java 모델 클래스, 원시타입

 

속성명과 변수명이 다를때 맵핑하는 방법은 resultMap을 사용하거나 Alias를 사용함 (Alias 권장)

 

● #{} 와 ${}의 차이점

 - # : 문자면 '   '를 달아줌 (숫자의경우 '  '을 달지않음)

 - $ : 어떠한 경우도 '  '을 달지않음

 

 

mybatis-config.xml (mybatis 설정파일)

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org/DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <typeAliases>
        <typeAlias type="mybatis.Comment" alias="Comment"/>	<!-- 자바빈에 별명 부여 -->
    </typeAliases>
    <environments default="development">
        <environment id="development">
            <transactionManager type="JDBC" />
            <dataSource type="POOLED">
                <property name="driver" value="oracle.jdbc.driver.OracleDriver"/>
                <property name="url" value="jdbc:oracle:thin:@localhost:1521:xe"/>
                <property name="username" value="scott"/>
                <property name="password" value="1111"/>
            </dataSource>
        </environment>
    </environments>
    <mappers>
        <mapper resource="mybatis/CommentMapper.xml"/>		<!-- 매퍼 등록 -->
        <mapper class="mybatis.anno.CommentMapper"/>
    </mappers>
</configuration>

 

● CommentMapper.xml (매퍼 설정파일)

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org/DTD Config 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
	
<mapper namespace="CommentMapper">		<!-- 네임스페이스는 보통 이름과 동일하게 사용 -->
	<cache />
	<sql id="BaseColumns">
		comment_no AS commentNo,
		user_id AS userId,
		comment_content AS commentContent,
		reg_date AS regDate
	</sql>
	<select id="selectComment" parameterType="hashmap" resultType="Comment">
        SELECT
            <include refid="BaseColumns" />
        FROM tcomment
        <where>
            <if test="commentNo != null">
                comment_no = #{commentNo}		<!-- commentNo key값 or getCommentNo() -->
            </if>
        </where>
    </select>
    <insert id="insertComment" parameterType="Comment">
        INSERT INTO tcomment(comment_no, user_id, comment_content, reg_date)
            VALUES (#{commentNo}, #{userId}, #{commentContent}, #{regDate})
    </insert>
</mapper>

→ #{}를 사용할 때 parameterType이 hashmap일경우 key값, DTO일경우 getter를 의미함

 

● Repository 클래스(=DAO)

public class CommentSessionRepository {
    private final String namespace = "CommentMapper";	// <mapper namespace="CommentMapper"> 와 일치해야함
    
    private SqlSessionFactory getSqlSessionFactory() {
        String resource = "mybatis/mybatis-config.xml";		// Mybatis 설정파일
        InputStream inputStream;
        try {
            inputStream = Resources.getResourceAsStream(resource);
        } catch (IOException e) {
            throw new IllegalArgumentException(e);
        }
        
        return new SqlSessionFactoryBuilder().build(inputStream);
    }
    
    public List<Comment> selectComment(Map<String, Object> condition) {
	SqlSession sqlSession = getSqlSessionFactory().openSession();
	try {
            return sqlSession.selectList(namespace + ".selectComment", condition);	// CommentMapper.selectComment
	} finally {			// 결과가 여러개일경우 : selectList
            sqlSession.close();
	}
    }
    
    public Comment selectCommentByPrimaryKey(Long commentNo) {
	SqlSession sqlSession = getSqlSessionFactory().openSession();
	try {
            return sqlSession.selectOne(namespace + ".selectCommentByPrimaryKey", commentNo);	// CommentMapper.selectCommentByPrimaryKey
	} finally {			// 결과가 1개일경우 : selectOne
            sqlSession.close();
	}
    }
    
    public Integer insertComment(Comment comment) {
	SqlSession sqlSession = getSqlSessionFactory().openSession();
	    try {
             int result = sqlSession.insert(namespace + ".insertComment", comment);	// CommentMapper.insertComment
            if (result > 0) {
                sqlSession.commit();
            } else {
                sqlSession.rollback();
            }
            return result;
        } finally {
            sqlSession.close();
        }
    }

    public static void main(String[] args) {
        CommentSessionRepository c = new CommentSessionRepository();
        
        // selectComment
        Map<String, Object> map = new HashMap<String, Object>();
        map.put("commentNo", 3L);
        System.out.println(c.selectComment(map));
        
        // insertComment
        Comment comment = new Comment();
        comment.setCommentNo(7L);
        comment.setCommentContent("sample_content");
        comment.setRegDate(new Date());
        comment.setUserId("sample user_Id");
        c.insertComment(comment);
    }
}

 

요소

● sql : 재사용 가능한 sql구문 정의 (id부여)

 - <include refid=""> 요소를 사용하여 SQL에 삽입

<sql id="BaseColumns">
    comment_no AS commentNo,
    user_id AS userId,
    comment_content AS commentContent,
    reg_date AS regDate
</sql>

 

● select

<select id="selectComment" parameterType="hashmap" resultType="mybatis.Comment">
    SELECT
        <include refid="BaseColumns" />
    FROM comment
</select>

 

● insert, update, delete

 - result가 없으므로 파라미터타입만 기입 (resultType, resultMap 필요 X)

<update id="insertComment" parameterType="Comment">
    UPDATE tcomment SET comment_content = #{commentContent} WHERE comment_no = #{commentNo}
</update>

 

● where : where절 사용

 - SQL구문에 WHERE를 추가, 게다가 구문이 AND 나 OR로 시작하면 그 구문을 지워줌 

 

● if

<where>
    <if test="commentNo != null">
        comment_no = #{commentNo}
    </if>
    <if test="userId != null">
        AND user_id = #{userId}
    </if>
</where>

 

● choose, when, otherwise : switch-case-default 구문

<select id="selectCommentByConditionChoose" parameterType="hashmap" resultType="Comment">
    SELECT comment_no AS commentNo, user_id AS userId, comment_content AS commentContent, reg_date AS regDate FROM tcomment
    <choose>
        <when test="commentNo != null">
            WHERE comment_no = #{commentNo}
        </when>
        <when test="userId != null">		<!-- 첫번째 when에 걸릴경우 실행되지 않음 -->
            WHERE user_id = #{userId}
        </when>
        <otherwise>				<!-- 첫번째 when에 걸릴경우 실행되지 않음 -->
            WHERE comment_no = 1 AND user_id = 'from'
        </otherwise>
    </choose>
</select>

 

● foreach : IN 구문에 많이 사용함

 - collection : 배열 or 컬렉션

 - item : each값

 - index : 인덱스

 - separator : 각 값의 구분자

 - open, close : 시작과 끝에 삽입할 문자

<select id="selectCommentByConditionForeach" parameterType="hashmap" resultType="Comment">
    SELECT comment_no AS commentNo, user_id AS userId, comment_content AS commentContent, reg_date AS regDate FROM tcomment
    <where>
        <if test="commentNos != null">
            comment_no IN
            <foreach collection="commentNos" item="commentNo" index="index" open="(" close=")" separator=", ">	<!-- (?, ?) -->
                #{commentNo}		<!-- item값 -->
            </foreach>
        </if>
    </where>
</select>

 

● selectKey : selectKey 구문이 먼저 실행되고 keyProperty에 결과를 넣은 후 그아래 SQL문이 실행됨

 - keyProperty : selectKey 구문의 결과가 셋팅될 프로퍼티

 - order : selectKey 구문의 순서 설정, SQL이 수행되기 전(BEFORE)인지 후(AFTER)인지

<insert id="insertCommentKey" parameterType="Comment">
	<selectKey keyProperty="commentNo" resultType="long" order="BEFORE">
		SELECT max(comment_no)+1 FROM tcomment
	</selectKey>
	INSERT INTO tcomment(comment_no, user_id, comment_content, reg_date)
		VALUES (#{commentNo}, #{userId}, #{commentContent}, #{regDate})
</insert>