일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- exception
- 자바
- GRANT VIEW
- 대덕인재개발원
- cursor문
- 예외미루기
- 참조형변수
- Java
- abstract
- 환경설정
- 컬렉션 타입
- oracle
- 집합_SET
- 자동차수리시스템
- 제네릭
- 메소드오버로딩
- 컬렉션프레임워크
- 오라클
- 인터페이스
- NestedFor
- EnhancedFor
- 정수형타입
- 다형성
- 객체 비교
- 예외처리
- 생성자오버로드
- 어윈 사용법
- 한국건설관리시스템
- 사용자예외클래스생성
- 추상메서드
- Today
- Total
거니의 velog
(2) 도서 쇼핑몰 만들기 2 본문
2. 도서 쇼핑몰 기능 알아보기
* 어떤 종류의 쇼핑몰이든 고객들이 가장 자주 사용하는 기능들을 다음 표로 정리해 보았다.
< 쇼핑몰에 구현된 기능 >
기능 | 세부기능 |
메인 페이지 | 메인 페이지 |
빠른 메뉴 | |
회원 기능 | 로그인 |
회원 가입 | |
상품 기능 | 상품 검색 |
장바구니 기능 | 장바구니 상품 조회 |
장바구니 상품 추가 | |
장바구니 상품 수정 | |
장바구니 상품 삭제 | |
주문 기능 | 상품 주문 |
주문 상품 조회 | |
주문 수정 | |
주문 취소 | |
마이페이지 기능 | 주문 내역 조회 |
회원 정보 수정 | |
회원 탈퇴 |
* 지금부터 각 기능을 차례대로 알아보자. 먼저 쇼핑몰의 프로젝트 및 데이터베이스 구조를 살펴보자.
(1) 쇼핑몰 프로젝트 및 데이터베이스 구조
* 먼저 우리가 만들 도서 쇼핑몰의 데이터베이스 구조를 살펴보자.
* 크게 5개의 테이블(회원정보, 주문테이블, 상품정보, 상품이미지정보, 장바구니)로 이루어져 있다. 각 테이블과 관련된 기능은 실습하면서 차차 설명하도록 한다.
* 이 프로젝트에서 사용할 Controller 클래스들의 계층 구조를 다음 그림에 나타내었다.
* 모든 Controller 클래스들은 BaseController 클래스를 상속받는 것을 볼 수 있다. 각각의 구현 클래스(*** ControllerImpl)에서 사용할 메서드를 ***Controller 인터페이스에 선언한다.
* 또한 다음 그림에서는 Service와 DAO 클래스의 계층 구조를 나타내었다.
* 이제는 도서 쇼핑몰 중 메인 페이지와 상품 조회 기능을 구현하는 bookshop01 프로젝트의 소스 구조를 살펴보자.
* 이 프로젝트에 사용되는 CSS, 이미지 파일, 제이쿼리, 자바스크립트 파일 등은 모두 src/main/webapp/resources 폴더에 있다.
* 톰캣을 실행하면 servlet-context.xml에서 src/main/webapp/resources의 CSS, 이미지 파일, 제이쿼리, 자바스크립트 파일을 읽어 들여 설정한다.
<?xml version="1.0" encoding="UTF-8"?>
<beans:beans xmlns="http://www.springframework.org/schema/mvc"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:beans="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!-- DispatcherServlet Context: defines this servlet's request-processing infrastructure -->
<!-- Enables the Spring MVC @Controller programming model -->
<annotation-driven />
<!-- /resources/ 폴더의 HTML, 자바스크립트, 제이쿼리, 이미지들을 읽어들여 설정한다. -->
<resources mapping="/resources/**" location="/resources/" />
<!-- 타일즈 기능을 설정한다. -->
<beans:bean id="tilesConfigurer" class="org.springframework.web.servlet.view.tiles2.TilesConfigurer">
<beans:property name="definitions">
<beans:list>
<beans:value>classpath:tiles/*.xml</beans:value>
</beans:list>
</beans:property>
<beans:property name="preparerFactoryClass" value="org.springframework.web.servlet.view.tiles2.SpringBeanPreparerFactory"/>
</beans:bean>
<beans:bean id="viewResolver" class="org.springframework.web.servlet.view.UrlBasedViewResolver">
<beans:property name="viewClass" value="org.springframework.web.servlet.view.tiles2.TilesView"/>
</beans:bean>
<context:component-scan base-package="com.bookshop01" />
<!-- 2단계와 3단계 요청에 대해 인터셉터를 설정한다. -->
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/*/*.do"/>
<mvc:mapping path="/*/*/*.do"/>
<beans:bean class="com.bookshop01.common.interceptor.ViewNameInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
<!-- 멀티파트 뷰리졸버를 설정한다. -->
<beans:bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<beans:property name="maxUploadSize" value="52428800" />
<beans:property name="maxInMemorySize" value="52428800" />
<beans:property name="defaultEncoding" value="utf-8" />
</beans:bean>
</beans:beans>
3. 메인 페이지 구현하기
* 우리가 만들 도서 쇼핑몰 메인 페이지에는 신간, 베스트셀러, 스테디셀러가 다음과 같이 한 줄에 4권씩 표시된다.
- http://localhost/bookshop01/main/main.do
* 또한 메인 페이지와 관련된 테이블을 보면 상품정보 테이블과 상품이미지정보 테이블의 데이터를 조회한 후 메인 페이지에 표시한다는 것을 알 수 있다.
* 그럼 본격적으로 진행해보자. 먼저 매퍼 파일을 구현한다.
1. 메인 페이지에 표시할 상품을 조회하는 SQL문에 대한 매퍼 파일인 goods.xml을 준비한다.
2. goods.xml에서 조건 값을 베스트셀러, 신간, 스테디셀러로 지정하여 각각 상품 정보와 이미지 정보를 15개씩 조회한다.
<mapper namespace="mapper.goods">
<!-- 상품 정보를 저장하는 resultMap이다. -->
<resultMap id="goodsResult" type="goodsVO">
<result property="goods_id" column="goods_id" />
<result property="goods_title" column="goods_title" />
<result property="goods_writer" column="goods_writer" />
<result property="goods_price" column="goods_price" />
<result property="goods_publisher" column="goods_publisher" />
</resultMap>
<!-- 상품 이미지 정보를 저장하는 resultMap이다. -->
<resultMap id="imageResult" type="ImageFileVO">
<result property="goods_id" column="goods_id" />
<result property="fileName" column="fileName" />
<result property="reg_id" column="reg_id" />
<result property="image_id" column="image_id" />
<result property="fileType" column="fileType" />
</resultMap>
<!-- goods_status의 조건 값으로 '신상품', '베스트셀러', '스테디셀러'를 전달해 각각의 상품을 15개까지 조회한다. -->
<select id="selectGoodsList" parameterType="String" resultMap="goodsResult" >
<![CDATA[
select t.*
from(
select g.*,d.fileName from t_shopping_goods g, t_goods_detail_image d
where g.goods_id=d.goods_id
and d.filetype='main_image'
and goods_status=#{goodsStatus}
order by g.goods_creDate desc) t
where rowNum <16
]]>
</select>
3. 다음으로 메인 페이지를 나타낼 타일즈를 설정하는 tiles_main.xml 파일을 생성한다.
4. tiles_main.xml을 다음과 같이 작성한다.
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE tiles-definitions PUBLIC
"-//Apache Software Foundation//DTD Tiles Configuration 2.0//EN"
"http://tiles.apache.org/dtds/tiles-config_2_0.dtd">
<tiles-definitions>
<!-- 쇼핑몰 기본 레이아웃을 설정한다. -->
<definition name="baseLayout"
template="/WEB-INF/views/common/layout.jsp">
<put-attribute name="title" value="레이아웃 페이지" />
<put-attribute name="header" value="/WEB-INF/views/common/header.jsp" />
<put-attribute name="side" value="/WEB-INF/views/common/side.jsp" />
<put-attribute name="body" value="" />
<put-attribute name="footer" value="/WEB-INF/views/common/footer.jsp" />
<put-attribute name="quickMenu" value="/WEB-INF/views/common/quickMenu.jsp" />
</definition>
<!-- 쇼핑몰 메인 페이지 레이아웃을 설정한다. -->
<definition name="/main/main" extends="baseLayout">
<put-attribute name="title" value="쇼핑몰 메인페이지" />
<put-attribute name="body" value="/WEB-INF/views/main/main.jsp" />
</definition>
</tiles-definitions>
5. 메인 페이지에도 상품 정보가 표시되므로 goods 패키지 기능을 사용한다. 필요한 자바 클래스는 다음과 같다.
6. 메인 컨트롤러인 MainController 클래스에서는 브라우저 요청 시 베스트셀러, 신간, 스테디셀러를 조회한 후 Map에 저장하여 JSP로 전달한다.
package com.bookshop01.main;
import java.util.List;
import java.util.Map;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.servlet.ModelAndView;
import com.bookshop01.common.base.BaseController;
import com.bookshop01.goods.service.GoodsService;
import com.bookshop01.goods.vo.GoodsVO;
@Controller("mainController")
@EnableAspectJAutoProxy
public class MainController extends BaseController {
@Autowired
private GoodsService goodsService;
@RequestMapping(value= "/main/main.do" ,method={RequestMethod.POST,RequestMethod.GET})
public ModelAndView main(HttpServletRequest request, HttpServletResponse response) throws Exception{
HttpSession session;
ModelAndView mav=new ModelAndView();
String viewName=(String)request.getAttribute("viewName");
mav.setViewName(viewName);
session=request.getSession();
session.setAttribute("side_menu", "user"); // 속성, 즉 side_menu의 값에 따라 화면 왼쪽에 표시되는 메뉴 항목을 다르게 한다.
Map<String,List<GoodsVO>> goodsMap=goodsService.listGoods(); // 베스트셀러, 신간, 스테디셀러 정보를 조회해 Map에 저장한다.
mav.addObject("goodsMap", goodsMap); // 메인 페이지로 상품 정보를 전달한다.
return mav;
}
}
7. Service 클래스인 GoodsServiceImpl에서는 베스트셀러, 신간, 스테디셀러를 SQL문의 조건으로 전달하여 각각의 도서 정보를 조회해서 HashMap에 저장한 후 컨트롤러로 반환한다.
@Service("goodsService")
@Transactional(propagation=Propagation.REQUIRED)
public class GoodsServiceImpl implements GoodsService{
@Autowired
private GoodsDAO goodsDAO;
public Map<String,List<GoodsVO>> listGoods() throws Exception {
Map<String,List<GoodsVO>> goodsMap=new HashMap<String,List<GoodsVO>>();
List<GoodsVO> goodsList=goodsDAO.selectGoodsList("bestseller");
goodsMap.put("bestseller",goodsList);
goodsList=goodsDAO.selectGoodsList("newbook");
goodsMap.put("newbook",goodsList);
goodsList=goodsDAO.selectGoodsList("steadyseller");
goodsMap.put("steadyseller",goodsList);
return goodsMap; // newbook, bestseller, steadyseller를 조건으로 각각 도서 정보를 조회해서 HashMap에 저장한 후 반환한다.
}
...
8. GoodsDAOImpl 클래스를 다음과 같이 작성한다.
@Repository("goodsDAO")
public class GoodsDAOImpl implements GoodsDAO {
@Autowired
private SqlSession sqlSession;
@Override
public List<GoodsVO> selectGoodsList(String goodsStatus) throws DataAccessException {
// selectList() 메서드 호출 시 전달된 조건으로 도서 정보를 조회한다.
List<GoodsVO> goodsList = (ArrayList) sqlSession.selectList("mapper.goods.selectGoodsList", goodsStatus);
return goodsList;
}
...
9. 이제는 각 상품의 이미지를 다운로드하는 컨트롤러를 구현할 차례이다.
10. FileDownloadController 클래스를 다음과 같이 구현한다.
@Controller
public class FileDownloadController {
private static String CURR_IMAGE_REPO_PATH = "C:\\shopping\\file_repo";
@RequestMapping("/download")
protected void download(@RequestParam("fileName") String fileName,
@RequestParam("goods_id") String goods_id, // 이미지 파일 이름과 상품 id를 가져온다.
HttpServletResponse response) throws Exception {
OutputStream out = response.getOutputStream();
String filePath=CURR_IMAGE_REPO_PATH+"\\"+goods_id+"\\"+fileName;
File image=new File(filePath);
response.setHeader("Cache-Control","no-cache");
response.addHeader("Content-disposition", "attachment; fileName="+fileName);
FileInputStream in=new FileInputStream(image);
byte[] buffer=new byte[1024*8];
while(true){
int count=in.read(buffer); //버퍼에 읽어들인 문자개수
if(count==-1) //버퍼의 마지막에 도달했는지 체크
break;
out.write(buffer,0,count);
}
in.close();
out.close();
}
@RequestMapping("/thumbnails.do")
protected void thumbnails(@RequestParam("fileName") String fileName,
@RequestParam("goods_id") String goods_id,
HttpServletResponse response) throws Exception {
OutputStream out = response.getOutputStream();
String filePath=CURR_IMAGE_REPO_PATH+"\\"+goods_id+"\\"+fileName;
File image=new File(filePath);
if (image.exists()) {
Thumbnails.of(image).size(121,154).outputFormat("png").toOutputStream(out);
} // 메인 페이지 이미지를 썸네일로 표시한다.
byte[] buffer = new byte[1024 * 8];
out.write(buffer);
out.close();
}
...
11. 마지막으로 메인 페이지와 왼쪽 메뉴를 표시하는 JSP 파일들을 준비한다.
12. main.jsp를 다음과 같이 표시한다.
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8" isELIgnored="false"
%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
<c:set var="contextPath" value="${pageContext.request.contextPath}" />
<%
request.setCharacterEncoding("UTF-8");
%>
<div id="ad_main_banner">
<ul class="bjqs">
<li><img width="775" height="145" src="${contextPath}/resources/image/main_banner01.jpg"></li>
<li><img width="775" height="145" src="${contextPath}/resources/image/main_banner02.jpg"></li>
<li><img width="775" height="145" src="${contextPath}/resources/image/main_banner03.jpg"></li>
</ul>
</div>
<div class="main_book">
<c:set var="goods_count" value="0" />
<h3>베스트셀러</h3>
<%-- 베스트셀러를 표시한다. --%>
<c:forEach var="item" items="${goodsMap.bestseller }">
<c:set var="goods_count" value="${goods_count+1 }" />
<div class="book">
<%-- 이미지 클릭 시 상품 상세 페이지로 넘어간다. --%>
<a href="${contextPath}/goods/goodsDetail.do?goods_id=${item.goods_id }">
<img class="link" src="${contextPath}/resources/image/1px.gif">
</a>
<%-- 썸네일 이미지를 표시한다. --%>
<img width="121" height="154" src="${contextPath}/thumbnails.do?goods_id=${item.goods_id}&fileName=${item.goods_fileName}">
<div class="title">${item.goods_title }</div>
<div class="price">
<fmt:formatNumber value="${item.goods_price}" type="number" var="goods_price" />
${goods_price}원
</div>
</div>
<%-- 베스트셀러가 15개 이상이면 16번째 이미지에는 more를 표시한다. --%>
<c:if test="${goods_count==15 }">
<div class="book">
<font size=20> <a href="#">more</a></font>
</div>
</c:if>
</c:forEach>
</div>
<div class="clear"></div>
<div id="ad_sub_banner">
<img width="770" height="117" src="${contextPath}/resources/image/sub_banner01.jpg">
</div>
<div class="main_book">
<c:set var="goods_count" value="0" />
<h3>새로 출판된 책</h3>
<c:forEach var="item" items="${goodsMap.newbook }">
<c:set var="goods_count" value="${goods_count+1 }" />
<div class="book">
<a href="${contextPath}/goods/goodsDetail.do?goods_id=${item.goods_id }">
<img class="link" src="${contextPath}/resources/image/1px.gif">
</a>
<img width="121" height="154" src="${contextPath}/thumbnails.do?goods_id=${item.goods_id}&fileName=${item.goods_fileName}">
<div class="title">${item.goods_title }</div>
<div class="price">
<fmt:formatNumber value="${item.goods_price}" type="number" var="goods_price" />
${goods_price}원
</div>
</div>
<c:if test="${goods_count==15 }">
<div class="book">
<font size=20> <a href="#">more</a></font>
</div>
</c:if>
</c:forEach>
</div>
<div class="clear"></div>
<div id="ad_sub_banner">
<img width="770" height="117" src="${contextPath}/resources/image/sub_banner02.jpg">
</div>
<div class="main_book">
<c:set var="goods_count" value="0" />
<h3>스테디셀러</h3>
<c:forEach var="item" items="${goodsMap.steadyseller }">
<c:set var="goods_count" value="${goods_count+1 }" />
<div class="book">
<a href="${contextPath}/goods/goodsDetail.do?goods_id=${item.goods_id }">
<img class="link" src="${contextPath}/resources/image/1px.gif">
</a>
<img width="121" height="154" src="${contextPath}/thumbnails.do?goods_id=${item.goods_id}&fileName=${item.goods_fileName}">
<div class="title">${item.goods_title }</div>
<div class="price">
<fmt:formatNumber value="${item.goods_price}" type="number" var="goods_price" />
${goods_price}원
</div>
</div>
<c:if test="${goods_count==15 }">
<div class="book">
<font size=20> <a href="#">more</a></font>
</div>
</c:if>
</c:forEach>
</div>
13. side.jsp에서는 컨트롤러에서 전달된 side_menu의 값에 따라 사용자, 관리자, 마이페이지 각각의 메뉴 항목을 표시한다.
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"
isELIgnored="false"
%>
<%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles" %>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<c:set var="contextPath" value="${pageContext.request.contextPath}" />
<nav>
<ul>
<c:choose>
<%-- 관리자 메뉴를 표시한다. --%>
<c:when test="${side_menu=='admin_mode' }">
<li>
<H3>주요기능</H3>
<ul>
<li><a href="${contextPath}/admin/goods/adminGoodsMain.do">상품관리</a></li>
<li><a href="${contextPath}/admin/order/adminOrderMain.do">주문관리</a></li>
<li><a href="${contextPath}/admin/member/adminMemberMain.do">회원관리</a></li>
<li><a href="#">배송관리</a></li>
<li><a href="#">게시판관리</a></li>
</ul>
</li>
</c:when>
<%-- 마이페이지 메뉴를 표시한다. --%>
<c:when test="${side_menu=='my_page' }">
<li>
<h3>주문내역</h3>
<ul>
<li><a href="${contextPath}/mypage/listMyOrderHistory.do">주문내역/배송 조회</a></li>
<li><a href="#">반품/교환 신청 및 조회</a></li>
<li><a href="#">취소 주문 내역</a></li>
<li><a href="#">세금 계산서</a></li>
</ul>
</li>
<li>
<h3>정보내역</h3>
<ul>
<li><a href="${contextPath}/mypage/myDetailInfo.do">회원정보관리</a></li>
<li><a href="#">나의 주소록</a></li>
<li><a href="#">개인정보 동의내역</a></li>
<li><a href="#">회원탈퇴</a></li>
</ul>
</li>
</c:when>
<%-- 그 외 사용자 메뉴를 표시한다. --%>
<c:otherwise>
<li>
<h3> 국내외 도서</h3>
<ul>
<li><a href="${contextPath}/goods/goodsList.do">IT/인터넷</a></li>
<li><a href="#">경제/경영</a></li>
<li><a href="#">대학교재</a></li>
<li><a href="#">자기계발</a></li>
<li><a href="#">자연과학/공학</a></li>
<li><a href="#">역사/인문학</a></li>
</ul>
</li>
<li>
<h3> 음반</h3>
<ul>
<li><a href="#">가요</a></li>
<li><a href="#">록</a></li>
<li><a href="#">클래식</a></li>
<li><a href="#">뉴에이지</a></li>
<li><a href="#">재즈</a></li>
<li><a href="#">기타</a></li>
</ul>
</li>
</c:otherwise>
</c:choose>
</ul>
</nav>
<div class="clear"></div>
<div id="banner">
<a href="#"><img width="190" height="163" src="${contextPath}/resources/image/n-pay.jpg"> </a>
</div>
<DIV id="notice">
<H2>공지사항</H2>
<UL>
<c:forEach var="i" begin="1" end="5" step="1">
<li><a href="#">공지사항입니다.${ i}</a></li>
</c:forEach>
</ul>
</div>
<div id="banner">
<a href="#"><img width="190" height="362" src="${contextPath}/resources/image/side_banner1.jpg"></a>
</div>
<div id="banner">
<a href="#"><img width="190" height="104" src="${contextPath}/resources/image/call_center_logo.jpg"></a>
</div>
<div id="banner">
<a href="#"><img width="190" height="69" src="${contextPath}/resources/image/QnA_logo.jpg"></a>
</div>
</html>
14. 실행하면 화면 왼쪽에 일반 사용자를 위한 메뉴 항목이 표시된다.
- http://localhost/bookshop01/main/main.do
'Java > Java_Spring mini project 쇼핑몰' 카테고리의 다른 글
(6) 도서 쇼핑몰 만들기 6 (0) | 2023.11.25 |
---|---|
(5) 도서 쇼핑몰 만들기 5 (0) | 2023.11.24 |
(4) 도서 쇼핑몰 만들기 4 (0) | 2023.11.24 |
(3) 도서 쇼핑몰 만들기 3 (2) | 2023.11.24 |
(1) 도서 쇼핑몰 만들기 1 (1) | 2023.11.23 |