[SPRING/Mybatis] POI를 이용한 대용량 데이터 추출 (엑셀 다운) -1

오랜만에 티스토리에 개발관련 기록을 남기는 것같다. 디비의 값을 엑셀로 다운로드을 수 있도록 해달라는 요청이 있었다. (요즘은 Android를 파는 중이지만...) 요청은 하나의 대상 테이블에 대한

kyome.tistory.com

 

이전 게시글에 이어서,

이제 ResultHandler 를 작성한다.

 

0. ResultHandler (ExcelHandler.java)

 

ResultHandler는 handleResult라는 메서드를

Override 하기만 하면 쉽게 사용할 수 있다.

handleResult는 데이터가 들어오는대로

건마다 호출되는 걸로 보인다.

 

 

1. 변수 및 생성자 선언

 

ResultHandler는 Generic으로 선언할 수 있고 

내가 만들 ExcelHandler는 Map으로 받아져야하기 때문에 

Generic 변수 T는 Map을 상속받는 변수야한다는 조건을 걸었다.

 

ResultHandler를 통해 다운로드를 받을 수 있도록 할 계획이기 때문에

생성자로 Response를 받으며 파일명이나

컬럼 순서를 받을 수 있도록 생성자를 추가했다.

 

import ...

public class ExcelHandler<T extends Map<String,Object>> implements ResultHandler<T> {
	public static final Logger LOGGER = LoggerFactory.getLogger(ExcelHandler.class);
	
	private T result;
	private String title;
	private String filename;
	private SXSSFWorkbook  workbook;
	private SXSSFSheet  sheet;
	private HttpServletResponse response;
	private ResultContext<? extends T> context;
	private List<String> columnTitleList;
	private int rownum;
	
	final int TITLE = 0;
	final int BODY = 1;

	
	private ExcelHandler() {
		super();
		rownum = 0;
	}
	
	public ExcelHandler(HttpServletResponse response, String filename) {
		this();
		this.response = response;
		this.title = filename;
		try {
			this.filename = URLEncoder.encode(filename.replace(" ", "_"),"UTF-8");
		} catch (UnsupportedEncodingException e) {
			e.printStackTrace();
		};
		workbook = new SXSSFWorkbook(10000);
		sheet = workbook.createSheet(title);
	}
	
	public ExcelHandler(HttpServletResponse response, 
    	String filename,List<String> orderedColumnTitleList) {
		this(response,filename);
		this.columnTitleList = orderedColumnTitleList;
	}

	...
    

 

 

 

 

 

 

 

 

2.  엑셀에 쓰기 메서드 구현

 

DB 에서 조회한 결과의 Column이 몇 개이고 이름이 무엇이든 상관없이

독립적으로 실행이 되어야 재사용이 가능해지기 때문에 

 

엑셀에 입력하는 기능에만 집중했고 

나머지는 파라미터로 받도록 했다.

 

private void write(int type,int currentRow,CellStyle style) {
		
	if (columnTitleList == null) {
		columnTitleList = new ArrayList<String>(context.getResultObject().keySet());
	}
	SXSSFRow row = sheet.createRow(currentRow);
	
	if(columnTitleList.size() == 0 ) {
		return;
	}
	
	for(int i = 0 ; i < columnTitleList.size() ; i++) {
		SXSSFCell cell = row.createCell(i);
		String tempValue = "";
		
		switch (type) {
			case TITLE:
				tempValue = columnTitleList.get(i);
				break;
			case BODY:
				if(context.getResultObject().containsKey(columnTitleList.get(i))) {
					tempValue = context.getResultObject().get(columnTitleList.get(i)).toString();
				}
				
				break;
		}
//		스타일 객체가 없다면 기본으로
		if(style != null) {
			cell.setCellStyle(style);
		}
		cell.setCellValue(tempValue);	
	}
}
	

 

 

 

3. handleResult Override

 

데이터가(한개의 행이) 호출 될 때마다 workbook에 입력해주고

현재의 행을 나타내는 변수를 ++ 해준다.

 

헤더전용 CellStyle 과 내용전용 CellStyle을 

미리 선언해두어서 write에 마지막 파라미터로 넣어주면

필요에 따라 셀스타일을 변경하면서 입력할 수도 있다.

 

	...

@Override
public void handleResult(ResultContext<? extends T> resultContext) {
	if(resultContext.getResultObject() == null) {
		return;
	}
	this.context = resultContext;
	result =  context.getResultObject();
	
	
	Font headerFont = workbook.createFont();
	headerFont.setFontName("맑은 고딕");
	headerFont.setBold(true);
       
	CellStyle headerStyle = workbook.createCellStyle();
	headerStyle.setBorderTop(BorderStyle.THIN);
	headerStyle.setBorderBottom(BorderStyle.THIN);
	headerStyle.setBorderLeft(BorderStyle.THIN);
	headerStyle.setBorderRight(BorderStyle.THIN);
        
	headerStyle.setAlignment(HorizontalAlignment.CENTER);
	headerStyle.setVerticalAlignment(VerticalAlignment.CENTER);
	headerStyle.setFillForegroundColor(IndexedColors.LIGHT_CORNFLOWER_BLUE.getIndex());
	headerStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND);
	headerStyle.setFont(headerFont);

	Font bodyFont = workbook.createFont();
	bodyFont.setFontName("맑은 고딕");
        
	CellStyle bodyStyle= workbook.createCellStyle();
	bodyStyle.setBorderTop(BorderStyle.THIN);
	bodyStyle.setBorderBottom(BorderStyle.THIN);
	bodyStyle.setBorderLeft(BorderStyle.THIN);
	bodyStyle.setBorderRight(BorderStyle.THIN);
        
	bodyStyle.setAlignment(HorizontalAlignment.LEFT);
	bodyStyle.setFont(bodyFont);
		
	if(rownum == 0 ) {			
		write(TITLE, rownum,headerStyle);
		write(BODY, rownum+1,bodyStyle);
	}else {
		write(BODY, rownum+1,bodyStyle);
	}
	rownum++;
}

...

 

 

 

 

 

 

4. 엑셀 다운로드

 

생성자로 Response 객체를 받는 이유는 여기에 있다.

Response 객체의 헤더에 첨부파일을 넣어서 

OutStream으로 파일을 내보낸다.

 

파일 처리는 항상 그렇듯 예외처리를 성실하게 해줘야하고

종료시에 close 를 호출해줘야한다.

 

	...
    
public void download() throws IOException{
	LOGGER.debug("## start excel download : "+filename);
	response.setHeader("Content-Disposition", "attachment; filename=" 
        				+ filename.replaceAll(" ", "_") + ".xlsx;");
	response.setCharacterEncoding(Constants.ENCODING);
		
	ServletOutputStream stream = response.getOutputStream();
	OutputStream out = new BufferedOutputStream(stream);
	
	try {
		response.resetBuffer();
		response.setBufferSize(1024 * 4);
		workbook.write(out);
		
	} catch (Exception e) {
		out.flush();
		out.close();
		stream.close();
	} finally {
		out.flush();
		out.close();
		stream.close();
	}
	
	if (workbook != null) {
		try {
			workbook.dispose();
		} catch (Exception e) {
			workbook.close();
		} finally {
			workbook.close();
		}
	}
	workbook.close();
}

public void close() {
	workbook.dispose();
	try {
		workbook.close();
	} catch (IOException e) {
		e.printStackTrace();
	}
}

...

 

 

 

 

 

도움이 되었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

 

오랜만에 티스토리에 개발관련 기록을 남기는 것같다.

디비의 값을 엑셀로 다운로드을 수 있도록 해달라는 요청이 있었다.

(요즘은 Android를 파는 중이지만...)

 

요청은 하나의 대상 테이블에 대한 엑셀 다운로드 구현이지만

어차피 개발할 거라면 재사용할 수 있도록 개발했다.

 

 

 

 

 

 

 

 

 

 

0. 의존성 추가 (pom.xml)

 


...
		<!-- excel -->
		<dependency>
		    <groupId>org.apache.poi</groupId>
		    <artifactId>poi</artifactId>
		    <version>4.1.1</version>
		</dependency>
		
		<dependency>
		    <groupId>org.apache.poi</groupId>
		    <artifactId>poi-ooxml</artifactId>
		    <version>4.1.1</version>
		</dependency>
...

 

 

 

 

1. SQL (ExcelMapper.xml)

 

엑셀 다운로드용 쿼리를 모아둘 목적으로 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="kr.co.excel.dao.ExcelMapper">
	<resultMap id="excelListDownloadMap" type="java.util.HashMap" >
    	<result column="CONTENT_NO"     	property="번호"    	jdbcType="INTEGER" />
    	<result column="CONTENT_TITLE"     	property="제목"    	jdbcType="VARCHAR" />
    </resultMap>

    <select id="contentListDownload" resultMap="excelListDownloadMap">
		<![CDATA[
			SELECT CONTENT_NO ,CONTENT_TITLE
			FROM TB_CONTENT
			WHERE CONTENT_NO < 100
		]]>
	</select>
</mapper>

 

 

 

2.  DAO (ExcelMapper.java)

 

현재 개발환경에서는 Mapper 를 사용하기 때문에

interface로 간단하게 DAO를 선언할 수 있다.

반환값을 void, 파라미터를 ResultHandler로 선언하면

Mybatis에서 조회되는 값을 바로 핸들링할 수 있다.

 

(ResultHandler 클래스를 잘 짜놓는다면...ㅎㅎㅎ)

ResultHandler는 너무 길어지니 이어지는 게시글에 자세히 적는걸로!

 

import java.util.Map;

import org.apache.ibatis.session.ResultHandler;
import org.springframework.stereotype.Repository;

@Repository
public interface ExcelMapper {
	public void contentListDownload(ResultHandler<Map<String,Object>> ExcelHander);
}

 

 

 

 

 

 

3. Service (ExcelService.java)

 

난 서비스를 ResultHandler과 DAO의 매핑 공간으로 사용했다.

매핑을 해놓으면 해당 메서드를 호출할때

ExcelHandler에 조회 결과가 매핑되는 걸로 보인다.

난 이 ResultHandler를 통해 다운로드 받을 엑셀 형식을 정의하고

다운로드 로직을 ResultHandler 에 넣어놓았다. (excelHandler.download())

 

import java.io.IOException;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import kr.co.excel.dao.ExcelMapper;

@Service
public class ExcelService {
	@Autowired
	private ExcelMapper excelMapper;
	
	ExcelHandler<Map<String,Object>> excelHandler;
	
	public void contentListDownload(
			HttpServletRequest request,
			HttpServletResponse response){
		excelHandler = new ExcelHandler<>(response,"테스트");
		excelMapper.contentListDownload(excelHandler);
		
		try {
			excelHandler.download();
		} catch (IOException e) {
			e.printStackTrace();
			excelHandler.close();
		}
	}
}

 

 

 

 

 

4. Controller (ExcelController.java)

 

View페이지가 별도로 필요없는 RestController를 선언했다. 

엑셀 다운로드시 Response 객체가

필요하기 때문에 파라미터로 선언해놓았다.

Request 객체는 추후에 필요할 듯 싶어서

넣어 놓았을 뿐 당장은 필요없다.

 

import java.util.Map;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import kr.co.excel.service.ExcelService;

@RestController
public class ExcelController {
	
	@Autowired
	ExcelService excelService;
	
	@RequestMapping(value = "/contentListDown.do")
	public void contentListDownload(
			HttpServletRequest request,
			HttpServletResponse response){
		excelService.contentListDownload(request,response);
	}
}

 

 

 

 

도움이 되었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

 

 


 

[SPRING/Mybatis] POI를 이용한 대용량 데이터 추출 (엑셀 다운) -2

 

[SPRING/Mybatis] POI를 이용한 대용량 데이터 추출 (엑셀 다운) -2

[SPRING/Mybatis] POI를 이용한 대용량 데이터 추출 (엑셀 다운) -1 오랜만에 티스토리에 개발관련 기록을 남기는 것같다. 디비의 값을 엑셀로 다운로드을 수 있도록 해달라는 요청이 있었다. (요즘은 A

kyome.tistory.com


 

 

 

 

안녕하세요 ☺️

전 몇 달 전부터 

유튜브 프리미엄을 시작했어요

유튜브 프리미엄을 사용해보니

정말 괜찮아서 추천해볼까 해요!👍

 

 


"Youtube Premium"

 

 

 

 

지인들에게 유튜브 프리미엄을 시작했다고 하면

보통 그냥 광고 보고 영상 보면 되는데

왜 굳이 돈을 쓰냐는 하시는 분들이 많은데

돈을 쓸만하답니다😎

 

 

 

1. 유튜브 프리미엄의 세계

 

 

 

 

유튜브 프리미엄을

그저 광고 없이 유튜브 보기 위해

돈을 쓴다고 생각하시는

분들이 많더라고요

 

유튜브를 광고없이 보는 것뿐 아니라

유튜브 뮤직으로 음악 스트리밍

서비스를 이용할 수 있어요

 

멀티태스킹이 지원되기 때문에

영상을 보다가 잠깐

카톡을 할 수도 있고 

오프라인 저장 기능을

사용할 수도 있답니다.👍

 

유튜브 오리지널도 

추가 비용 없이 볼 수 있어요

K-POP 그룹의 다큐멘터리를

개봉하기도 하고 있는 걸 보면

꾸준히 확장해나갈 것으로 기대가 돼요

 

(현재 저는 유튜브 오리지널을

애용하고 있진 않답니다..😅)

 

링크 참고

(https://www.youtube.com/premium)

 

 

 

 

 

 

 

 

2. 유튜브 프리미엄, 왜 쓸까

1) 절대적인 유튜브 사용량

 

출처 https://it.donga.com/29482/

 

 

 

전 유튜브를 이용하는 시간이

절대적으로 늘었기 때문에 

유튜브 프리미엄이 눈에 들어왔어요

 

저는 주로 출퇴근처럼

이동할 때 핸드폰을 많이 봐요

예전에는 출퇴근 때 노래를 들으며

핸드폰으로 웹서핑을 했지만

 

요즘은 유튜브가 저의 모든

이동시간을 책임지고 있더라고요! 

 

유튜브 광고를 보며 영상을

기다리는 시간도 점점 늘어나면서 

유튜브 프리미엄 가입도

괜찮겠다 싶었어요🤔

 

 

 

 

2) 음악 스트리밍 서비스 대용

 

 

 

하루에 몇 곡 안 듣는데

굳이 매달 음악

스트리밍을 결제해야 할까

라는 생각으로 대안도 없이 가입한

스트리밍 서비스를

해지한 적도 있었어요

 

그리고 알았어요

내가 생각보다 노래도

많이 듣는다는 사실을..😵

 

비록 출퇴근에 노래 듣는 시간은

많이 줄었지만

음악 들을 스트리밍은

필요하더라고요

 

그래서 찾은 답이 유튜브 프리미엄의

유튜브 뮤직이었답니다.

 

 

 

 

 

 

 

 

3) 유튜브 뮤직 따라올 수 없는 음악 추천

 

 

 

유튜브 프리미엄의 진가는

유튜브 뮤직을 만날 때일 것 같아요

 

구글은 추천의 신이라

구글 추천 알고리즘을 경험하면 

유튜브에서 헤어나오 못하고 있지요

 

 

음악도 똑같아요 

유튜브 뮤직에서 듣고 싶은 노래를

하나 찾아서 들으면

추천해주는 플레이리스트가

자동으로 생기고 자연스럽게

다른 곡으로 넘아가면서

계속 노래를 듣게 되더라고요 😮

 

 

예를 들면 피아노곡을 듣고 싶다!

하면 대표적인 피아노곡을

선택해서 플레이하면

비슷한 곡들로 추천해주기 때문에 

나의 플레이리스트를 구성하기 위해

큰 노력이 필요 없어요

정말 진심으로 편하고 좋아요

 

 

 

 

3. 합리적 가격

 

 

 

부가세 포함한 한국 가격

8700원

 

누군가는 너무 비싸다고 하지만

저는 유튜브가 제공하는 서비스에 비해서 

8700원이면 정말 합리적이라고 생각해요

 

자주 듣지 않는 음악 스트리밍에

6000원 쓰는 것보다 

하루 종일도 볼 수 있는 유튜브를 

더 편하고 좋은 환경에서

볼 수 있도록 만드는데

쓰는 8700원이 더 합리적인

소비라고 생각해요!

 

아이폰 사용자는 꼭 PC에서 가입하세요!!

PC로 가입하면 무료 사용기간도 더 길게 주고

아이폰 앱에서 구독 결제하는 것보다 

더 저렴하게 가입할 수 있어요!!

 

 

 

 

4. 저렴하게 하는 방법 (비추)

 

 

유튜브 프리미엄 인도 계정을 만들어서

가입하는 방법이 온라인에서

많이 나오더라고요!

 

인도는 우리나라보다

물가가 낮기 때문에

정말 말도 안 되게 저렴한

가격으로 가입을 할 수 있어요 

 

그렇지만 유튜브 뮤직 감상 중에 

갑자기 인도 노래가 나오고

인도 차트를 추천해준다고 해요.. ㅎㅎ

 

추천이 가장 큰 매력 중에

하나인 유튜브 서비스에서

인도 문화를 추천해준다니...

제가 볼 땐 이러한 부분이 

가장 큰 마이너스 요인이더라고요!

 

그리고 이렇게 인도가 싸서

편법으로 저렴하게

서비스를 이용하고 있다는 걸 

구글이 이미 알고 있고

적극 대응 중이라고 해요

잡히면 계정이 정지당한다고 하니..

 

절대 비추합니다. 😒

 

 

저렴하게 서비스를

이용하려는 노력도 이해는 되지만

서비스의 정당한 금액을 지불하고 

이용하는 것도 올바른 소비자의 길이지

않을까 싶어요ㅎㅎㅎ


 

 

 

흥미로운 추천이었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

안드로이드에서 뷰 객체에

이벤트를 만드는 건
패턴화 되어 있어서 

어렵지 않게 개발할 수 있다.

 

TextView textView = (TextView)findViewById(R.id.textView); 

textView.setOnClickListener(new View.OnClickListener() { 
  @Override 
  public void onClick(View view){ 

  } 
}); 

 


그렇지만 왜 이런 패턴과 구조로

작성해야 하는지 궁금했다.

무작정 외우기보단

이해하고 싶어서

안드로이드 이벤트에

공부해보기로 했다

 

 

 

 


이벤트 쉽게 이해하기

 

 

1. 이해하기 어려운 이유

 

안드로이드 이벤트에 대해 찾아보니
이벤트 소스, 이벤트 핸들러, 

이벤트 리스너 같은 낯선 용어들이 

이해를 더 어렵게 하는 것 같다.
그래서 이런 용어들을 빼고

최소한의 용어들로 이해한 것들을

정리해보았다.

 

 

 

 

 

 

 

2. 뷰 (View)는 누구인가

 

 

안드로이드에서 보여지는 

버튼, 텍스트 등은 다 뷰라는 

이름의 객체이다.

 

 

 

특징 - 1. 뚜렷한 목적

 

뷰들은 다 제각기 

생성된 목적이 뚜렷하다.

 

(Ex, Button은 눌림 (Click) 을 당하기 위해, 

CheckBox는 선택 (Checked) 이 되기 위해..)

 

 

특징 - 2. 다양한 이벤트

또 하나의 특징으로는 

뷰는 다양한 이벤트를 가질 수 있다.

즉, TextView도 클릭이벤트를 

가질 수 있다는 말이다.

 

이처럼 다양한 뷰를

효율적으로 관리하기 위해

뷰는 이벤트를 실행할 때마다 

'on이벤트명()'이라는 메서드가 

실행되도록 불러준다.

 

 


3. 우리가 기대하는 뷰

 

 

사용자는 안드로이드 화면에서 

뷰를 눌렀을 때 기대하는 바가 있다

 

만약, 전송 버튼을 눌렀는데 

버튼이 눌린 시늉만 하고 

아무 일도 생기지 않으면

사용자는 앱이 고장 났다고 

볼 것이다.

 

 

 

 

 

 

 

4. 뷰의 입장 이해하기

 

 

클릭 이벤트의

패턴을 이해하기 위해서

뷰의 입장이 되어 보았다.

뷰의 입장이 돼 보면 클릭이벤트 패턴을

이해하기가 한결 수월해진다.

 

 

 

 

뷰 입장 - 1. 난처한 뷰

 

"세세한 것 하나하나 다 해달라고?

너무한 거 아니야?"

 

[그림] 뷰가 알아야 할 것들이 너무 많아진다

 

뷰 입장에서 생각해보면 

사용자가 버튼을 눌렀으면 

눌린 티만 내면 그만이지 


굳이 눌렸을 때 

어떤 일들이 어떻게 진행되는지 

다 알고 있을 이유가 없다. 

뷰가 수행해야 하는 이벤트가 한 개만

있는 것도 아니고 많은 이벤트에 대한 

세세한 내용까지 뷰가 다 관리하자니

뷰 입장에선 너무 부담스럽다.

 

더 정확히 말하자면,

뷰에 로직을 다 넣으면

무거워지고 의존성이 높아지고 등등..

아무튼 지저분해져서 보고 싶지 

않은 코드가 될 것이다. 

 

 

 

 

뷰 입장 - 2. 뷰의 책임감

 

"그래도 기능을 성실하게

수행해야 하긴 할 텐데.."

 

뷰는 사용자와 직접 소통하는 

역할을 담당하기 때문에 
아무 기능을 하지 않는 뷰는

사실 존재의 가치가 없다고

해도 과언이 아니다. 


이벤트를 효율적으로 처리할

방법이 필요한 상황이다. 

 

 

 

 

 

 

5. 뷰의 묘수

 

[그림] 로직을 대신 수행할 객체를 모집

 

뷰의 묘수 - 1. 대리자 모집

 

세세한 로직을

다 알고 싶지 않은 뷰는 로직을

다 알고 있는 대리자를 모집하기로 한다.

 

그렇다고 아무나 불러서 

일을 수행하라고 할 수는 없으니

최소한의 자격요건을 충족하는 

대리자를 모집한다.

 

 

뷰가 내건 최소한의 자격요건


 " 'on이벤트명' 메서드를 

잘 구현해낸 자"


 

이벤트가 일어날 때마다 

이벤트 이름 앞에 on이 붙은 
'on이벤트명' 메서드를 같이 호출할 테니
이벤트에 함께 호출되고 싶다면

'on이벤트명' 메서드를

작성해와서 등록하라고 한 것이다.

 

 

 

 

 

 

뷰의 묘수 - 2. 관리소장 두기

 

이제 지원한 대리자를 

관리하는 일이 남았다.  

 

뷰는 내부에 대리자 관리소장을

하나 두기로 한다.

 

관리소장은 대리자를 등록하는 

일과 이벤트가 일어났을 때 

대리자를 호출해서 로직을 

수행하도록 하는 일을 한다.

 

 

 

6. 결과 

 

[그림] 클릭 이벤트가 발생시 처리

 

이렇게 구조를 정리해 놓으니

뷰는 이제 이벤트가 발생하면

내부에 관리 소장을 불러

어떤 이벤트가 발생했다는 사실만

알려주면 된다.

 

 


 

제가 공부 후 정리한 것에 대해

혹시 다른 의견이나 수정해주실 부분 있으면

꼭 댓글 달아주세요 

저도 아직 공부하는 중이랍니다.

 

 

 

 

도움이 되는 게시글이었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

 

 

 

 

안녕하세요☺️

상하이를 떠나기로 결정하고, 

날짜를 정하셨다면 

가장 먼저 준비해야 하는 건

비행기겠죠?😌

 

항공편을 준비한 방법을

공유할까 해요😎

 

 


상하이행 비행기

예약하기

 

1. 알찬 여행의 첫 단추 

출발하는 비행기

가장 이른 시간으로,

돌아오는 비행기

가장 늦은 시간으로

예매하는 걸 추천해요

 

왕복으로 미리 구매하는 게

가격면에서도 유리할 수 있고

미리 여행을 계획하기에 좋아요!😌

 

 

 

 

저 같은 경우는 8시 55분이

가장 빠른 비행기였고 

시차로 인해 중국시간

10시에 도착했어요🙊

(한국시간, 11시)

 

중국에서 점심을 먹고 

오후부터 본격적인

일정을 소화했답니다!

 

 

여행의 스타일은

저마다 다르겠지만

일찍 해외로 가고 싶고 

오래 해외에 머물고 싶은 마음은 

비슷할 것 같아요!

 

그러기 위해선 당연히

비행기 시간부터

잘 준비해야겠죠?😉

 

 

 

 

 

 

 

2. 중국 저가항공 OK!

 

중국의 저가 항공을 이용해서 

여행경비를 아끼는 것도 추천해요

당연히 우리나라 항공사가

더 좋고 편하겠죠!

그렇지만 비싸요..!😅

 

 

 

 

인천공항에서 푸동공항까지는 

2시간밖에 걸리지 않는데

가격차이는 대부분

5만원 이상 차이가 나요!

 

이 2시간을 위해서

5만원 이상의

비용을 지출하긴

좀 아까운 것 같아요 🤔

 

 

아, 중국 저가항공의 기내식이 

맛있진 않아요...😭

전 중국동방항공이었는데 

그냥 먹기에 나쁘지 않은 수준??

이었답니다..ㅎㅎ

 

 

 

 

3. 다양한 어플 활용

 

 

 

여행은 정보전인가 봅니다🤓

 더 많은 정보를 모아서

가장 합리적인 선택을 해야겠죠?

 

어느 정도 찾다 보면

비슷한 가격대의

항공권을 찾게 되지만

그래도 조금이라도 더 저렴하고 

나에게 맞는 시간대를

찾기 위해선 노력이 필요해요!

 

 

 

 

 

<추천하는 방법>

 

1) 네이버 활용

 

 

일단 가장 쉽게 찾아볼 수 있는 

네이버 항공권으로 검색해서 

대략의 가격을 알아보기!🤓

 

 

 

2) 각종 어플 활용

 

 

네이버에서 본

가격대를 기준으로 

어플별로 가격을

비교해보면 돼요!😉

 

어플마다 보여주는

방식이 다르기 때문에 

모두 다운받아서 더 편한 어플,

더 합리적인 가격을 제시하는

어플을 이용해서 

예매하면 된답니다!

 

저는 Skyscanner,

Trip.com, 플레이윙즈

이렇게 다운받아서 사용했어요!

 

 

 

 

4. 티켓 구매는 신속하게

어찌 보면 당연한 이야기겠지만

꼭 당부드리고 싶은 정보예요

저도 오늘을

그냥 알아보기만 하고

결제는 내일 해야지!

하다가 그 사이에

가격이 올라버렸어요 😥

 

날짜가 정해졌고 가격이 

충분히 합리적이다! 

하면 바로 결제를

하는 걸 추천해요!

 


 

 

 

 

 

흥미로운 추천이었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

 

 

안녕하세요 🙂

해외여행가기 좋은곳 !

얼마전 상하이를 다녀왔어요 😎

 

여행지로 상하이를 선택한 이유부터

상하이만의 특징을 소개할까해요!

 

 

 


SHANGHAI 

 

 

 

 

 

 

1. 이국적인 특색이 있는 곳

 

비슷한 문화를

공유하고 있는 중국이지만

어딜가더라도 중국스러움을

잃지않는 중국의 특색을

느낄 수 있어요!😌

 

 

(모두모두 중국말로

말하기 때문에 

더더욱 여기가 외국이구나

싶어요..ㅎㅎㅎ)

 

가깝지만 외국을 간

느낌을 확 느끼고 싶다면

중국도 괜찮아요!👍

 

 

 

 

2. 짧은 여행기간

 

 

여행의 기간이 짧다면 

상하이 여행을 추천해요!

상하이는 여행하기 좋은 지역이에요

대중교통이 아주 잘 되어있고

표지판도 친절하기 때문에

여행하면서 큰 어려움이 없어요!🧐

 

무엇보다도 명소들이

넓게 퍼져 있지않고 

지하철역과 가깝기 때문에 

하루에도 여러군데를

둘러볼 수 있답니다!😎

 

 

 

 

 

비행기로 2시간 밖에

걸리지 않고 

공항에서 도심까지 

자기부상열차로 간다면

30분정도만에 갈 수 있어서

이동시간을 아끼고

여행을 만끽할 수 있다는 점이

상하이 여행의 특징이랍니다🤓

 

 

 

 

 

 

 

3. 괜찮은 치안

 

 

여행지가 위험하다면

아무리 예쁘고 아름다워도 

가기가 꺼려지겠죠? 

 

상하이는 치안이

괜찮다는 인상을 받았어요

모든 역마다 공안들이

순찰을 돌고 역에 들어갈때

공항처럼 소지품을 검사해요🙊

 

명소 주변 거리에는

공안같은 분들이

감시하고 있더라고요..ㅎㅎ

무서울정도로 안전해요 😅

 

상하이가 야경이

유명해서인지 

밤에 돌아다녀도 

특별히 불편한 일은 없었어요!

 

 

 

 

4. 먹고 또 먹고, 다양한 먹거리

 

 

중국하면 역시 먹거리지요!

한국에서 먹어볼 수 없는

길거리 음식들이 많아요!

 

요즘은 탕후루,마라탕 등등

한국에서도 맛볼 수 있지만

원조의 느낌은 또 다르겠죠? 😉

 

또 빼놓을 수 없는 메뉴

밀크티와 딤섬도 있답니다.

 

상해는 물가가

부담스러운 정도는 아니에요😲

 

음식점에서 판매하는 식사들은

특별히 엄청 싸다는

느낌은 받지 못했지만

한국물가와 큰차이는 없고

한국보다 조금 저렴한 정도랍니다!

 

 

 

 

 

 

 

5. 다양한 명소, 낭만의 야경

 

상하이는 유럽풍의

야경을 볼 수있는 와이탄,

중국의 느낌을

확 느낄 수 있는 예원,

한국인이라면 더 크게

와닿을 대한민국 임시정부 등등

직접 가서 느끼고 볼 거리가 많아요 !👍

 

특히 야경은 돌아와서

또 생각나는 낭만적인 공간이랍니다!

 

 

 

 

6. 상하이 디즈니랜드 

 

 

상해에는 디즈니랜드가 있어요 

그것도 아시아 최대 규모랍니다.

제대로 다 하나씩 느끼고

체험하려면 2일정도의 계획을

잡아야 할정도예요 🤔

 

 

 

 

 

어트랙션은 디즈니만의 이야기를

놀이기구로 만들었기 때문에 

더 몰입이 돼요!

 

규모에서 놀라고

디테일에서 놀라는

상하이 디즈니랜드! 

강력 추천해요👏👏

 

 


 

 

 

 

흥미로운 추천이었다면

로그인이 필요 없는 공감 버튼 꾹 눌러주세요! 

 

 

 

 

+ Recent posts

"여기"를 클릭하면 광고 제거.