web

Spring Boot - Excel Upload & Download

허원철 2017. 5. 5. 12:20
이번 글은 Spring Boot를 이용하여 Excel을 Upload 와 Download을 다루는 예제 글입니다.


자바진영에서 엑셀을 Upload, Download하기 위해 POI를 가장 많이 사용합니다. 엑셀 버전마다 다를 수 있는데, xls의 경우는 poi, xlsx의 경우는 poi-ooxml를 추가하면 됩니다. (xls의 경우, Excel(5.0/95)이하는 불가능한 걸로 알고 있습니다.)
 
 
1. Gradle
  
1
2
3
4
5
6
7
8
9
dependencies {
    compile('org.apache.poi:poi-ooxml:3.16'// .xlsx
    compile('org.apache.poi:poi:3.16'// .xls
 
    compile('org.projectlombok:lombok:1.16.6')
 
    compile('org.springframework.boot:spring-boot-starter-web')
    testCompile('org.springframework.boot:spring-boot-starter-test')
}
cs
  
  
2. Upload
  
- Java 8의 Function, Stream을 이용하여 Component를 추가해보겠습니다.
  
2-1. Component
 
1) MultipartFile과 Function을 이용하여 Component를 만들어줍니다.
    
1
2
3
4
5
6
7
8
@Component
public class ExcelReadComponent {
 
    public <T> List<T> readExcelToList(final MultipartFile multipartFile,
                                       final Function<Row, T> rowFunc) throws IOException, InvalidFormatException {
    // ...
    }
}
cs
 
2) File을 Workbook으로 변환하기 전에 엑셀 파일인지 검증해줍니다.
 
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
public <T> List<T> readExcelToList(final MultipartFile multipartFile,
                                   final Function<Row, T> rowFunc) throws IOException, InvalidFormatException {
 
    final Workbook workbook = readWorkbook(multipartFile);
        
    // ...
}
 
private Workbook readWorkbook(MultipartFile multipartFile) throws IOException, InvalidFormatException {
    verifyFileExtension(multipartFile);
    return multipartFileToWorkbook(multipartFile);
}
 
private void verifyFileExtension(MultipartFile multipartFile) throws InvalidFormatException {
    if!isExcelExtension(multipartFile.getOriginalFilename()) ) {
        throw new InvalidFormatException("This file extension is not verify");
    }
}
 
private boolean isExcelExtension(String fileName) {
    return fileName.endsWith(ExcelConfig.XLS) || fileName.endsWith(ExcelConfig.XLSX);
}
 
private boolean isExcelXls(String fileName) {
    return fileName.endsWith(ExcelConfig.XLS);
}
 
private boolean isExcelXlsx(String fileName) {
    return fileName.endsWith(ExcelConfig.XLSX);
}
cs
 
3) Workbook으로 변환 뒤, Stream을 이용하여 Row 갯수 만큼 Domain로 바꾸고 해당 Stream을 List로 변환해줍니다.
  
1
2
3
4
5
6
7
8
9
10
11
12
public <T> List<T> readExcelToList(final MultipartFile multipartFile,
                                   final Function<Row, T> rowFunc) throws IOException, InvalidFormatException {
 
    final Workbook workbook = readWorkbook(multipartFile);
    final Sheet sheet = workbook.getSheetAt(0);
    final int rowCount = sheet.getPhysicalNumberOfRows();
 
    return IntStream
            .range(0, rowCount)
            .mapToObj(rowIndex -> rowFunc.apply(sheet.getRow(rowIndex)))
            .collect(Collectors.toList());
}
cs
 
 
2-2. Controller & Domain
 
1) 위에서 작성한 Component를 이용하여 JSON 형태로 반환해줍니다.
  
1
2
3
4
5
6
7
8
9
10
11
12
@RestController
@RequestMapping("upload")
public class UploadExcelController {
 
    @Autowired ExcelReadComponent excelReadComponent;
 
    @PostMapping("excel")
    public List<Product> readExcel(@RequestParam("file") MultipartFile multipartFile)
                                                throws IOException, InvalidFormatException {
        return excelReadComponent.readExcelToList(multipartFile, Product::new);
    }
}
cs
   
2) Row를 이용하여 Domain 객체를 만들어줍니다. (간단한 예제이기 때문에 유효성 검사가 없으니 상세한 작업을 원한다면 추가해줘야 할 것 입니다.)
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
@Data
@AllArgsConstructor
public class Product implements Serializable {
 
    private String uniqueId;
 
    private String name;
 
    private String comment;
 
    public Product(Row row) {
        this(row.getCell(0).getStringCellValue(),
            row.getCell(1).getStringCellValue(),
            row.getCell(2).getStringCellValue());
    }
}
cs
  
 
3. Download
 
- spring 4.x 이전에는 AbstractExcelView를 사용하였으나, 이후에는 AbstractXlsView, AbstractXlsxView, AbstractXlsxStreamingView를 이용해야 합니다.
- 각각의 Component로 등록을 해준 뒤, ModelAndView에서 해당 name으로 반환할 때, 인식할 수 있도록 @EnableWebMvc를 추가해줍니다.
- 각각의 AbstractXXXView는 내부적으로 기본 셋팅(Workbook 등등...) 되어 있기 때문에, 공통적으로 buildExcelDocument()를 오버라이딩하면 됩니다.
- 그래서 공통적으로 사용할 ExcelCommonUtil을 만들었습니다. 
 
1
2
3
4
5
6
7
@Override
protected void buildExcelDocument(Map<String, Object> model,
                                  Workbook workbook,
                                  HttpServletRequest request,
                                  HttpServletResponse response) throws Exception {
    new ExcelCommonUtil(workbook, model, response).createExcel();
}
cs
 
- workbook은 해당 Excel형식에 맞는 인스턴스 객체가 됩니다. (XSSFWorkbook, SXSSFWorkbook, HSSFWorkbook 등)
- model은 Controller에서 ModelAndView에서 담은 Map을 뜻합니다.
- response는 클라이언트에게 반환해줄 객체입니다.
 
3-1. ExcelCommonUtil
  
1
2
3
4
5
6
7
8
9
10
public void createExcel() {
    // 1)
    setFileName(response, mapToFileName());
    // 2)
    Sheet sheet = workbook.createSheet();
    // 3)
    createHead(sheet, mapToHeadList());
    // 4)
    createBody(sheet, mapToBodyList());
}
cs
 
1) Download될 Excel 파일에 대해 명명을 해줍니다. 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
private void setFileName(HttpServletResponse response, String fileName) {
    response.setHeader("Content-Disposition",
                       "attachment; filename=\"" + setFileExtension(fileName) + "\"");
}
 
private String setFileExtension(String fileName) {
    if ( workbook instanceof XSSFWorkbook) {
        fileName += ".xlsx";
    }
    if ( workbook instanceof SXSSFWorkbook) {
        fileName += ".xlsx";
    }
    if ( workbook instanceof HSSFWorkbook) {
        fileName += ".xls";
    }
 
    return fileName;
}
cs
  
2) 해당 Excel 파일에 하나의 시트를 만들어줍니다. (단순 예제이기 때문에 시트 하나만 만들어줬습니다.)

3) head에 해당하는 데이터를 넣어줍니다.
 
1
2
3
private void createHead(Sheet sheet, List<String> headList) {
    createRow(sheet, headList, 0);
}
cs
 
4) body에 해당하는 데이터를 넣어줍니다.

1
2
3
4
5
6
private void createBody(Sheet sheet, List<List<String>> bodyList) {
    int rowSize = bodyList.size();
    for (int i = 0; i < rowSize; i++) {
        createRow(sheet, bodyList.get(i), i + 1);
    }
}
cs
 
※ createHead, createBody에서 공통으로 쓰인 메소드
  
1
2
3
4
5
6
7
8
9
private void createRow(Sheet sheet, List<String> cellList, int rowNum) {
    int size = cellList.size();
    Row row = sheet.createRow(rowNum);
 
    for (int i = 0; i < size; i++) {
        row.createCell(i)
            .setCellValue(cellList.get(i));
    }
}
cs

3-2. Controller
 
- ModelAndView를 이용하여 각각의 Component를 호출해줍니다.
  
1
2
3
4
5
6
7
8
9
10
11
@Controller
@RequestMapping("download")
public class DownloadExcelController {
 
    @GetMapping("excel-xls")
    public ModelAndView xlsView() {
        return new ModelAndView("excelXlsView", getDefaultMap());
    }
 
    // ...
}
cs
 
 
참고
 
- 조금 더 자세한 코드 내용은 깃헙을 참고하시기 바랍니다.