Download file in csv format for java (2)

上一遍说到了基于注解式文件下载CSV格式,今天咱就聊聊基于函数式的文件下载,函数式接口是java8提到的新名词,感兴趣的可以查看下源码,可以简化开发代码,自定义流式风格,深度学习请移步至Oracle相关文档,我们这次用到的是:java.util.function包.

直接上道理(代码出现的常量请看上一篇,这里不贴了):

	/**
	 * 导出csv文件数据,基于函数式接口实现
	 * @param fileName 文件名
	 * @param datas 数据源
	 * @return
	 */
	public File exportCsvFile(String fileName, List<T> datas) {
		File f = null;
		OutputStreamWriter fileWriter = null;
		try {
			f = File.createTempFile(fileName, FILE_FORMAT);
			fileWriter = new OutputStreamWriter(new FileOutputStream(f), ENCODINT_GBK);
			fileWriter.append(objectToString(datas));
		} catch (Exception e) {
			// do something
		} finally {
			try {
				fileWriter.flush();
				fileWriter.close();
			} catch (IOException e) {
				// do something
			}
		}
		return f;
	}

注意:我用的是创建临时文件方法,File.createTempFile(String perfix,String suffix)中perfix是有长度要求的;源码片段截图如下:

也就是说参数fileName的长度必须大于3,否则会抛IAE;

	/**
	 * 将对象数据处理成字符串
	 * @param datas 对象数据集合
	 * @return
	 */
	protected String objectToString(List<T> datas) {
		StringBuilder sb = new StringBuilder();
                //获取数据提供者中所要导出的所有不为空的字段名
		List<String> list = providers.stream().map(e -> e.getField()).filter(Objects::nonNull)
				.collect(Collectors.toList());
                //导入时必须显式的Filed
		if (!list.isEmpty()) {
			String filedRow = Joiner.on(COMMA_DELIMITER).join(list);
			sb.append(filedRow).append(NEW_LINE_SEPARATOR);
		}
               //获取字段名标题
		String titleRow = Joiner.on(COMMA_DELIMITER)
				.join(providers.stream().map(e -> e.getTitle()).collect(Collectors.toList()));
		sb.append(titleRow).append(NEW_LINE_SEPARATOR);
		for (T data : datas) {
			for (Provider<T> p : providers) {
				Object o = p.getFunc().apply(data);//获取数据
				sb.append(SPECIAL_SYMBOLS);
				sb.append(Objects.isNull(o) ? BLANK_STRING : o);
				sb.append(COMMA_DELIMITER);
			}
			sb.delete(sb.length() - 1, sb.length());
			sb.append(NEW_LINE_SEPARATOR);
		}
		return sb.toString();
	}

信息提供者: 

class Provider<T> {
	private String field;
	private String title;
	private Function<T, Object> func;

	public Provider(String field, String title, Function<T, Object> func) {
		super();
		this.field = field;
		this.title = title;
		this.func = func;
	}

	public Provider(String title, Function<T, Object> func) {
		super();
		this.title = title;
		this.func = func;
	}

	public String getField() {
		return field;
	}

	public void setField(String field) {
		this.field = field;
	}

	public String getTitle() {
		return title;
	}

	public void setTitle(String title) {
		this.title = title;
	}

	public Function<T, Object> getFunc() {
		return func;
	}

	public void setFunc(Function<T, Object> func) {
		this.func = func;
	}
}

注册数据源: 

	/**
	 * 导入导出使用
	 * @param filed 为null时隐去域名,不可导入,同导出使用
	 * @param title 
	 * @param func
	 * @return
	 */
	public ExportCsvUtils<T> register(String filed, String title, Function<T, Object> func) {
		providers.add(new Provider<T>(filed, title, func));
		return this;
	}

	/**
	 * 仅导出使用
	 * @param title
	 * @param func
	 * @return
	 */
	public ExportCsvUtils<T> register(String title, Function<T, Object> func) {
		providers.add(new Provider<T>(title, func));
		return this;
	}
//数据收集器
private List<Provider<T>> providers = Lists.newArrayList();

测试类:


class Test {
	String a;
	Integer b;

	public String getA() {
		return a;
	}

	public void setA(String a) {
		this.a = a;
	}

	public Integer getB() {
		return b;
	}

	public void setB(Integer b) {
		this.b = b;
	}

	public Test(String a, Integer b) {
		super();
		this.a = a;
		this.b = b;
	}

	@Override
	public String toString() {
		return "Test [a=" + a + ", b=" + b + "]";
	}

}

测试使用:

public static void main(String[] args) {
		List<Test> list = Lists.newArrayList(new Test("c", 1), new Test("d", null), new Test("e", 3));
		ExportCsvUtils<Test> utils = new ExportCsvUtils<Test>();
		File file = utils.register("测试a", a -> a.getA()).register("测试b", a -> Objects.isNull(a.getB()) ? 2 : a.getB())
				.exportCsvFile("测试下载文件", list);//注意:文件名长度大于3
		System.err.println(file);
	}

测试结果:

好了数据导出大功告成,但是按实际需求还没完,代码片段中有提到数据导入时Filed必须是显式的,那接下来咱们就得研究下如何导入CSV文件的数据了.

为什么导出数据中会有"`"符号,是因为导出的csv文件用wps表格或excel表格打开时防止出现科学计数法,日期格式变动(原:yyyy-MM-dd打开后:yyyy/MM/dd)等问题,造成数据错误问题,届时对数据的导入麻烦就大了.

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值