하이버네이트를 통해 쿼리를 사용하다보면 꼭 한 번은 만나볼 오류 중 하나이다.
org.hibernate.hql.internal.ast.QuerySyntaxException: board is not mapped [from board order by id]
org.hibernate.hql.internal.ast.QuerySyntaxException.generateQueryException(QuerySyntaxException.java:79)
org.hibernate.QueryException.wrapWithQueryString(QueryException.java:103)
org.hibernate.hql.internal.ast.QueryTranslatorImpl.doCompile(QueryTranslatorImpl.java:220)
org.hibernate.hql.internal.ast.QueryTranslatorImpl.compile(QueryTranslatorImpl.java:144)
org.hibernate.engine.query.spi.HQLQueryPlan.<init>(HQLQueryPlan.java:113)
org.hibernate.engine.query.spi.HQLQueryPlan.<init>(HQLQueryPlan.java:73)
org.hibernate.engine.query.spi.QueryPlanCache.getHQLQueryPlan(QueryPlanCache.java:162)
org.hibernate.internal.AbstractSharedSessionContract.getQueryPlan(AbstractSharedSessionContract.java:636)
org.hibernate.internal.AbstractSharedSessionContract.createQuery(AbstractSharedSessionContract.java:748)
org.hibernate.internal.AbstractSharedSessionContract.createQuery(AbstractSharedSessionContract.java:848)
org.hibernate.internal.AbstractSharedSessionContract.createQuery(AbstractSharedSessionContract.java:114)
com.bigbell.noticeboard.dao.BoardDAOImpl.getBoard(BoardDAOImpl.java:26)
com.bigbell.noticeboard.service.BoardServiceImpl.getBoard(BoardServiceImpl.java:22)
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
java.lang.reflect.Method.invoke(Method.java:498)
org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:344)
org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:198)
org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:163)
org.springframework.transaction.interceptor.TransactionInterceptor$1.proceedWithInvocation(TransactionInterceptor.java:123)
org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:388)
org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:119)
org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186)
org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:215)
com.sun.proxy.$Proxy48.getBoard(Unknown Source)
com.bigbell.noticeboard.controller.NoticeBoardController.listBoard(NoticeBoardController.java:23)
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
java.lang.reflect.Method.invoke(Method.java:498)
org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:197)
org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:141)
org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:106)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:895)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:808)
org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1064)
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:963)
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898)
javax.servlet.http.HttpServlet.service(HttpServlet.java:655)
org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)
javax.servlet.http.HttpServlet.service(HttpServlet.java:764)
org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)
전제
필자는 지금 Board라는 클래스를 Entity로 설정하여 hibernate의 query를 이용해 데이터베이스 테이블인 "board"의 정보를 가져오려고 시도 중이었다.
오류해석
첫번째 줄을 잘 살펴보자.
org.hibernate.hql.internal.ast.QuerySyntaxException: board is not mapped [from board order by id]
board로 mapping되지 않는다는 말인데 대체로 2가지 문제 중 하나로 인해 생긴다. 하나씩 살펴보자.
1. query를 적을 때 잘못된 클래스명을 사용
@Entity
@Table(name="board")
public class Board {
@Id
@GeneratedValue(strategy=GenerationType.IDENTITY)
@Column(name="id")
private int id;
......
}
위와 같은 Entity가 있다. 클래스명은 Board 이고 이를 table인 "board"와 연결시켰다.
잘못된 query 작성
@Repository
public class BoardDAOImpl implements BoardDAO {
// inject the sessionFactory
@Autowired
private SessionFactory sessionFactory;
@Override
public List<Board> getBoard() {
Session currentSession = sessionFactory.getCurrentSession();
Query<Board> query =
currentSession.createQuery("from board order by id", Board.class);
List<Board> boards = query.getResultList();
return boards;
}
}
위 파일은 데이터베이스 테이블에서 정보를 가져오기 위한 DAO.java 파일의 일부분이다.
여기서 어떤 것이 잘못됐을까? 당연히 쿼리문이다.
오류원인
Query<Board> query =
currentSession.createQuery("from board order by id", Board.class);
쿼리문을 적을 때 from 뒤에 테이블이름이 아닌 Entity로 설정된 클래스명을 적어줘야하는데
여기서 대소문자 구분이 아주 중요하다!!
오류해결
Query<Board> query =
currentSession.createQuery("from Board order by id", Board.class);
board를 클래스명과 대소문자도 아예 같게 Board로 고쳐주면 정상실행될 것이다.
(웬만하면 여기서 해결된다. 그래도 안된다면 2번, 3번 참고)
2. sevlet.xml을 활용하는 경우, servlet.xml 에서의 오류(동적 웹 프로젝트, 메이븐 둘 다 해당)
1번에서 문제가 해결되지 않는다면?
java.lang.IllegalArgumentException: org.hibernate.hql.internal.ast.QuerySyntaxException: Board is not mapped [from Board order by id]
위와 같은 오류가 발생했을 것이다.
확인해보면 "Board is not mapped [from Board order by id]" 이라고 적혀있다.
분명히 Board로 클래스명과 똑같이 맞춰줬는데도 같은 오류가 발생하고 있다. 왜일까?
여기서 문제는 servlet.xml일 가능성이 크다.
스프링과 하이버네이트를 사용하면 따로 servlet.xml을 만들어 의존성을 주입하는데
sessionFactory를 구성할 때 문제가 생긴 것이다.
일단 Board의 파일 경로를 보자. com.bigbell.noticeboard.entity 로 되어있다.
servlet.xml에서 sessionFactory가 어떤 경로를 탐색하는지 살펴보자.
<bean id="sessionFactory"
class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
<property name="dataSource" ref="myDataSource" />
<property name="packagesToScan" value="com.bigbell.noticeboard.entiii" />
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
<prop key="hibernate.show_sql">true</prop>
</props>
</property>
</bean>
오류원인
<property name="packagesToScan" value="com.bigbell.noticeboard.entiii" />
경로가 com.bigbell.noticeboard.entity와는 마지막 부분이 약간 다르게 되어 있다.
대부분 오타로 인해 문제가 발생할 가능성이 높다.
문제해결
<property name="packagesToScan" value="com.bigbell.noticeboard.entity" />
경로를 Entity인 Board 클래스와 똑같이 맞춰주면 문제는 말끔히 해결될 것이다.
3. No Xml인 프로젝트(@Configuration 사용)의 경우, properties파일 내용 오류
아래와 같이 하이버네이트 컨트롤에 사용되는 properties파일이 있다고 해보자.
# JDBC connection properties
#
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/데이터베이스명?useSSL=false&serverTimezone=UTC
jdbc.user=mysql사용자아이디
jdbc.password=mysql사용자비밀번호
#
# Connection pool properties
#
connection.pool.initialPoolSize=5
connection.pool.minPoolSize=5
connection.pool.maxPoolSize=20
connection.pool.maxIdleTime=3000
#
# Hibernate properties
#
hibernate.dialect=org.hibernate.dialect.MySQLDialect
hibernate.show_sql=true
hibernate.packagesToScan=com.bigbell.springdemo.entiii
하이버네이트를 사용할 때 properties파일을 사용해서 연결을 시도해야하는데,
이 과정 중 package 스캔 부분을 잘못 설정하면 QuerySyntaxException 오류가 발생하곤 한다.
위의 코드 중 무엇이 잘못됐을까?
오류원인
hibernate.packagesToScan=com.bigbell.springdemo.entiii
패키지 스캔을 진행할 패키지명에 오류가 있다.
문제해결
hibernate.packagesToScan=com.bigbell.springdemo.entity
패키지명의 오타를 확인하고 잘 고쳐주자.