@ApiOperation("模板采集到的数据导出excel")
    @PostMapping("/templateDatas/export")
    public void exportImg(FormTemplateVo vo, @ApiIgnore HttpServletRequest request, @ApiIgnore HttpServletResponse response){
 
        AdminUserDto admin = RequestUtils.getLoginUser(request);
        HashMap<String,Object> dataMap = formTemplateService.templateDatasExport(vo, admin);
        LinkedHashMap headerMap = (LinkedHashMap)dataMap.get("header");// 表头列名
        List<HashMap<String, Object>> datas = (List<HashMap<String, Object>>)dataMap.get("list");
 
        for (HashMap<String, Object> m : datas) {
            Set<String> keySet = m.keySet();
            ArrayList<String> imgList = new ArrayList<>();
 
            for (String key : keySet) {
                if (!key.contains("uploadFile")) continue;
                if (StringUtils.isBlank(m.get(key))) continue;;
                //imgList = new ArrayList<>((List<String>) m.get(key));
                imgList = new ArrayList<>(JSON.parseArray(m.get(key).toString(), String.class));
                File[] files = new File[imgList.size()];
                if(imgList.size()>0){
                    for (int i =0 ; i< imgList.size(); i++){
                        String url = imgList.get(i);
                        if (StringUtils.isBlank(url)) continue;
 
                        String filePath = ExcelUtil.saveFile(url, ExcelUtil.path);  //远程服务器的网络图片,所以先保存到本地,如果是本地服务器图片,不需要保存这一步
                        files[i] = new File(filePath);
 
                        // 判断是否是gif格式
                        if (url.contains(".gif")){
                            // gif转二进制流
                            byte[] gifByte = ExcelUtil.GifToJpg(files[i]);
                            // 二进制流转图片
                            if (ExcelUtil.ioToImg(gifByte, i+".png", files[i].getName()) > 0){
                                files[i] = new File(ExcelUtil.path + i + ".png");
                            }
                        }
                    }
                }
                m.put(key, files);
            }
        }
 
        String fileName = "表单采集数据列表" + new Date().getTime() + ".xls";
        response.setContentType("application/vnd.ms-excel");
        response.addHeader("Content-Disposition", "attachment; filename="+fileName);
 
        ExcelUtil.excelOut(headerMap, datas.size(), datas, fileName, response, ExcelUtil.path);
    }
public static final String path= "./template/excelDeleteImage/";
 
    /**
	 * excel工具类,可导出带图片或不带图片的数据,注意数据规则
	 * @date 2022/5/9
	 * @param headerMap 第一行的标题列
	 * @param rows 数据行量
	 * @param maps 装载导出数据的封装了map的list数据集合,注意:此中的map尽量用本类中的方法 javaBean2Map直接生成,或自己拼接;但需与参数titles[]的标题相关数据对应上
	 * @param fileName 导出到本地的文件路径和文件名
	 * @param response response
	 * @param path 保存到本地的图片地址(我这里是为了删除该目录下的图片,因为我是把网络图片保存到到本地的,如果图片已经是本地图片的话就不需要删除)
	 */
	public static void excelOut(LinkedHashMap headerMap, int rows, List<HashMap<String,Object>> maps, String fileName,
								HttpServletResponse response, String path){
 
		OutputStream out = null;
		BufferedImage bufferImg = null;
		HSSFWorkbook wb = null;
 
		try{
			//创建工作sheet
			wb = new HSSFWorkbook();
			HSSFSheet sheet = wb.createSheet(fileName);
			//设置单元格内容水平垂直居中
			HSSFCellStyle style = wb.createCellStyle();
			style.setVerticalAlignment(VerticalAlignment.CENTER);
			style.setAlignment(HorizontalAlignment.CENTER);
			style.setWrapText(true); //设置内容自动换行
 
			//冻结首行
			sheet.createFreezePane(0,1,0,1);
			//画图的顶级管理器,一个sheet只能获取一个(一定要注意这点)
			HSSFPatriarch patriarch = sheet.createDrawingPatriarch();
			HSSFRow row0 = sheet.createRow(0);
			row0.setHeightInPoints(25);
			if (headerMap.size() == 0){
				return ;
			}
			HSSFCell cell = null;
			//第一行、标题行列
			Object[] headerArr = headerMap.keySet().toArray();
			// 每一列占用的单元格数
			int[] cellSize = new int[headerMap.size()];
			for (int i = 0; i < headerMap.size(); i++){
				String value = headerMap.get(headerArr[i]).toString();
				String[] valueArr = value.split("->");
				// 记录单元格数量的值
				cellSize[i] = 1;
 
				// 合并的数量
				int merge = 1;
				// 开始列
				int startLine = Arrays.stream(cellSize).sum() > 1 ? Arrays.stream(cellSize).sum()-1 : 0;
				// 合并单元格
				if (valueArr.length > 1){
					merge = Integer.parseInt(valueArr[1]);
					if (merge > 1){
						CellRangeAddress address = new CellRangeAddress(0, 0, startLine, startLine + merge - 1);
						sheet.addMergedRegion(address);
					}
				}
				cellSize[i] = merge;
 
				cell = row0.createCell(startLine);     //第一个单元格
				cell.setCellValue(valueArr[0]);         //设定值
				cell.setCellStyle(style);
				sheet.setColumnWidth(startLine,6000);
			}
 
			HSSFRow row = null;
			HSSFCell cellRow = null;
			HSSFClientAnchor anchor = null;
 
			for (int i=1;i<=rows;i++){
				int cellColumn = 0;
				//创建行
				row = sheet.createRow(i);
				//设置默认行高
				row.setHeightInPoints(25);
				//行数据处理
				Map<String, Object> stringObjectMap = maps.get(i - 1);
				for (int j = 0; j < headerArr.length; j++) {
					Object header = headerArr[j];
					if (j > 0) cellColumn = sum(cellSize, j);
 
					for(Object value : stringObjectMap.keySet()){
						// 匹配列名和列数据
						if (!header.toString().equals(value)) continue;
						//行单元格
						cellRow = row.createCell(cellColumn);
						cellRow.setCellStyle(style);
						// 设置默认宽度
						//sheet.setColumnWidth(cellColumn,20 * 256);
						//如果行数据中有图片时候的处理
						if (value.toString().contains("uploadFile")){
							File[] file = StringUtils.isBlank(stringObjectMap.get(value)) ? null : (File[]) stringObjectMap.get(value);
							if (file == null || file.length == 0){
								cellRow.setCellValue("");
								continue;
							}else{
								row.setHeightInPoints(50);
 
								for (int x=0;x<file.length;x++){
									ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream();
									//if (x>0){
									//	cellRow = row.createCell(cellColumn);
									//	cellRow.setCellStyle(style);
									//}
									sheet.setColumnWidth(cellColumn,5000);
									//log.error("图片路径"+file[x]);
									// ImageIO.read()方法缺陷,会出现导出图片泛红的问题
									//bufferImg = ImageIO.read(file[x]);
									Image src = Toolkit.getDefaultToolkit().getImage(file[x].getPath());
									bufferImg = toBufferedImage(src);
									ImageIO.write(bufferImg, "jpg", byteArrayOut);
 
									// 计算图片在单元格的位置
									int y = 1023 / file.length;
 
									//anchor = new HSSFClientAnchor(y * x, 0, y * (x + 1), 250,(short) cellColumn, i, (short) cellColumn, i);
									anchor = new HSSFClientAnchor(0, 0, 1023, 250,(short) cellColumn, i, (short) cellColumn, i);
									anchor.setAnchorType(ClientAnchor.AnchorType.byId(0));
									patriarch.createPicture(anchor, wb.addPicture(byteArrayOut.toByteArray(), HSSFWorkbook.PICTURE_TYPE_JPEG));
									cellColumn++;
								}
								continue;
							}
						}
						cellRow.setCellValue(stringObjectMap.get(value).toString());
						//cellColumn ++;
					}
				}
			}
			if(wb!=null){
				out = response.getOutputStream();
				response.setContentType("application/x-msdownload");
				response.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode(fileName, "UTF-8") +".xls");
				// 写入excel文件
				wb.write(out);
			}
		}catch (Exception e){
			e.printStackTrace();
		}finally {
			if(out != null){
				try {
					out.close();
 
					//执行删除生成的图片
					File file = new File(path);//输入要删除文件目录的绝对路径
					deleteFile(file);//由于是保存网络图片到本地服务区,所以画完图片到excel就要删除文件
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
 
		}
	}
 
	public static int sum(int a[], int n){
		return (n<1)?0:sum(a,n-1)+a[n-1];
	}
 
	/**
	 * 删除文件夹目录下的所有文件 (怕到时候本地服务器图片越来越多,占用资源,所以把图片洗完到excel里面就删除)
	 * @date 2022/5/9
	 * @param file
	 */
	public static void deleteFile(File file){
		//判断文件不为null或文件目录存在
		if (file == null || !file.exists()){
			//log.error("文件删除失败,请检查文件路径是否正确");
			return;
		}
		//取得这个目录下的所有子文件对象
		File[] files = file.listFiles();
		//遍历该目录下的文件对象
		for (File f: files){
			//打印文件名
			String name = file.getName();
			//log.error("删除的文件名"+name);
			//判断子目录是否存在子目录,如果是文件则删除
			if (f.isDirectory()){
				deleteFile(f);
			}else {
				f.delete();
			}
		}
		//删除空文件夹  for循环已经把上一层节点的目录清空。
		file.delete();
	}
 
	/**
	 * 保存图片到本地
	 * @date 2022/5/9
	 * @param imageUrl
	 * @param path
	 * @return
	 */
	public static String saveFile(String imageUrl, String path){
		String filename = imageUrl.substring(imageUrl.lastIndexOf("/")+1, imageUrl.length());
		//log.error("图片===="+filename);
		//Random rand = new Random();
		//int s = rand.nextInt(900)+ 100;
		int s = (int) (Math.random() * 10000);
		//log.error("随机数=="+s);
		filename = s + filename;  //这里如果有文件名称重复的,就取一个随机数拼接文件名
		File sf= null;
		OutputStream os = null;
		InputStream is = null;
		try {
			// 构造URL
			URL url = new URL(imageUrl);
			// 打开连接
			URLConnection con = url.openConnection();
			//设置请求超时为5s
			con.setConnectTimeout(5*1000);
			// 输入流
			is = con.getInputStream();
 
			// 1K的数据缓冲
			byte[] bs = new byte[1024];
			// 读取到的数据长度
			int len;
			// 输出的文件流
//		    String path = "E:\\data\\nginxd\\sportsApplets";
//          String path = "/data/nginxd/sportsApplets/excelDeleteImage/";
			sf = new File(path);
			if(!sf.exists()){
				sf.mkdirs();
			}
			os = new FileOutputStream(sf.getPath()+"/"+filename);
			// 开始读取
			while ((len = is.read(bs)) != -1) {
				os.write(bs, 0, len);
			}
		} catch (IOException e) {
			e.printStackTrace();
		} finally {
			// 完毕,关闭所有链接
			try {
				if(os!=null){
					os.close();
				}
				if(is!=null){
					is.close();
				}
			} catch (IOException e) {
				e.printStackTrace();
			}
		}
		return sf.getPath()+"/"+filename;
	}
 
	/**
	 * 将gif格式图片转换为jpg格式文件,并直接返回byte二进制流
	 * @param file
	 * @return
	 */
	public static byte[] GifToJpg(File file){
		BufferedImage bufferedImage;
		ByteArrayOutputStream out = new ByteArrayOutputStream(1024);
		try {
			// read image file
			bufferedImage = ImageIO.read(file);
			// create a blank, RGB, same width and height, and a white
			BufferedImage newBufferedImage = new BufferedImage(
					bufferedImage.getWidth(), bufferedImage.getHeight(),
					BufferedImage.TYPE_INT_RGB);
			// TYPE_INT_RGB:创建一个RBG图像,24位深度,成功将32位图转化成24位
			newBufferedImage.createGraphics().drawImage(bufferedImage, 0, 0, Color.WHITE, null);
			// write to jpeg file
			ImageIO.write(newBufferedImage, "png",out);//转换输出到二进制数组流
			//ImageIO.write(newBufferedImage, "jpg",new File("c:\\java.jpg"));//转换输出到文件
			return out.toByteArray();//二进制流
		} catch (IOException e) {
			//logger.error("***GifToJpg方法报错***");
			e.printStackTrace();
		}
		return null;
	}
 
    public static BufferedImage toBufferedImage(Image image) {
		if (image instanceof BufferedImage) {
			return (BufferedImage) image;
		}
		// 确保加载了图像中的所有像素
		image = new ImageIcon(image).getImage();
		BufferedImage bimage = null;
		GraphicsEnvironment ge = GraphicsEnvironment
				.getLocalGraphicsEnvironment();
		try {
			int transparency = Transparency.OPAQUE;
			GraphicsDevice gs = ge.getDefaultScreenDevice();
			GraphicsConfiguration gc = gs.getDefaultConfiguration();
			bimage = gc.createCompatibleImage(image.getWidth(null),
					image.getHeight(null), transparency);
		} catch (HeadlessException e) {}
 
		if (bimage == null) {
			// 使用默认颜色模型创建缓冲图像
			int type = BufferedImage.TYPE_INT_RGB;
			bimage = new BufferedImage(image.getWidth(null),
					image.getHeight(null), type);
		}
		// 复制图像到缓冲图像
		Graphics g = bimage.createGraphics();
		// 将图像绘制到缓冲图像上
		g.drawImage(image, 0, 0, null);
		g.dispose();
		return bimage;
	}
 
 
	/**
	 * @author P
	 * 将二进制流转换为图片
	 * @param imageBytes  二进制流(byte数组)
	 * @param imgAddress  新图片保存地址【包括图片名称包含后缀】
	 * @param oldAddress  旧的图片保存地址【包括图片名称包含后缀】
	 * @return 返回0(保存失败)返回1(保存成功)
	 *
	 * */
	public static int ioToImg(byte[] imageBytes, String imgAddress, String oldAddress) {
		//二进制转化为图片
		if (imageBytes == null) return 0;
		try (FileOutputStream fileOutputStream = new FileOutputStream(new
				File(path + imgAddress));) {
			fileOutputStream.write(imageBytes);//保存在本地
			File oldFile = new File(path + oldAddress);
			oldFile.delete();
			return 1;
		} catch (IOException e) {
			System.out.println(e);
			return 0;
		}
	}