目录
EasyExcel是一个基于Java的简单、省内存的读写Excel的开源项目。在尽可能节约内存的情况下支持读写百M的Excel工具。
官网:EasyExcel官方文档 - 基于Java的Excel处理工具 | Easy Excel
github地址:GitHub - alibaba/easyexcel: 快速、简洁、解决大文件内存溢出的java处理Excel工具
Java解析、生成Excel比较有名的框架有Apache poi、jxl,但他们都存在一个严重的问题就是非常的耗内存。
EasyExcel 重写了poi,使一个3M的excel只需要几M内存,并且再大的excel不会出现内存溢出。
64M内存1分钟内读取75M(46W行25列)的Excel。
项目名:zx-test-parent

修改pom文件
<dependencies> <!--测试--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency> </dependencies>
项目名:zx-test-excel

修改pom,添加依赖
<dependencies> <dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>3.1.1</version> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency> </dependencies>
常用的注解:

@Data
@AllArgsConstructor
@NoArgsConstructor
@ColumnWidth(30)
@HeadRowHeight(30)
public class Student1 {
@ExcelProperty("编号")
private String id;
@ExcelProperty("姓名")
@ContentFontStyle(fontName = "楷体",italic = BooleanEnum.TRUE,color = Font.ITALIC)
private String name;
@ExcelProperty({"基本信息","年龄"})
private Integer age;
@ExcelProperty({"基本信息","电话"})
private String phone;
@ExcelProperty({"可选信息","邮件"})
private String Email;
@ExcelProperty({"可选信息","生日"})
@DateTimeFormat("yyyy-MM-dd HH:mm")
private Date birthday;
}
public String path(){
return this.getClass().getResource("/").getPath();
}
excel 属于 office组件一个软件
存在若干版本,大体上划分2种情况,2007前的,2007年后的
2003版:扩展名 xls,内容比较少,最大单元格 IV65536 ,256列(IV)
2007版:扩展名 xlsx,内容较多,最大单元格 XFD1048576,16384列(XFD)
// 模拟数据
public List<Student1> getDate(){
//模拟十条数据
ArrayList<Student1> student1s = new ArrayList<>();
for (int i = 0; i < 10; i++) {
Student1 student1 = new Student1(i+"","土豆"+i,i,"115"+i,"626"+i,new Date());
student1s.add(student1);
}
return student1s;
}
//创建测试写程序
@Test
public void testWrite(){
//写入位置:%classpath%/
//文件名称:student-demo.xls
//表名:土豆
//1,文件位置
String file = path()+"student_demo1.xls";
//2 写操作
EasyExcel.write(file, Student1.class).sheet("土豆").doWrite(getDate());
}
处理类:
处理类需要实现 AnalysisEventListener 接口
package com.czxy.read;
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.czxy.excel.Student1;
public class StudentDataListener1 extends AnalysisEventListener<Student1> {
@Override
public void invoke(Student1 student1, AnalysisContext analysisContext) {
System.out.println(student1);
}
@Override
public void doAfterAllAnalysed(AnalysisContext analysisContext) {
}
}
//测试读出excel程序
@Test
public void testRead(){
String file = path()+"student_demo1.xls";
//读操作
EasyExcel.read(file, Student1.class, new StudentDataListener1()).sheet("土豆").doRead();
}
package com.czxy.excel;
import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.annotation.format.DateTimeFormat;
import com.alibaba.excel.annotation.write.style.ColumnWidth;
import com.alibaba.excel.annotation.write.style.ContentFontStyle;
import com.alibaba.excel.annotation.write.style.HeadRowHeight;
import com.alibaba.excel.enums.BooleanEnum;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.awt.*;
import java.util.Date;
@Data
@AllArgsConstructor
@NoArgsConstructor
@ColumnWidth(30)
@HeadRowHeight(30)
public class Student2 {
@ExcelProperty("编号")
private String id;
@ExcelProperty("姓名")
@ContentFontStyle(fontName = "楷体",italic = BooleanEnum.TRUE,color = Font.ITALIC)
private String name;
@ExcelProperty({"基本信息","年龄"})
private Integer age;
@ExcelProperty({"基本信息","电话"})
private String phone;
@ExcelProperty({"可选信息","邮件"})
private String Email;
@ExcelProperty({"可选信息","生日"})
@DateTimeFormat("yyyy-MM-dd HH:mm")
private Date birthday;
}
//得到根目录路径
public String path(){
return this.getClass().getResource("/").getPath();
}
// 模拟数据
public List<Student1> getDate(){
//模拟十条数据
ArrayList<Student1> student1s = new ArrayList<>();
for (int i = 0; i < 10; i++) {
Student1 student1 = new Student1(i+"","土豆"+i,i,"115"+i,"626"+i,new Date());
student1s.add(student1);
}
return student1s;
}
//创建测试书写多表
@Test
public void testWrite(){
//写入位置:%classpath%/
//文件名称:student-demo.xls
//表名:土豆
//1,文件位置
String file = path()+"student_demo2.xls";
//2 写操作
ExcelWriter excelWriter = EasyExcel.write(file, Student2.class).build();
//写入多个sheet
WriteSheet writeSheet1 = EasyExcel.writerSheet("爱吃豆的").build();
excelWriter.write(getDate(),writeSheet1);
//写入第二个sheet
WriteSheet writeSheet2 = EasyExcel.writerSheet("土豆").build();
excelWriter.write(getDate(),writeSheet2);
excelWriter.finish();
}
具有缓存处理类
package com.czxy.read;
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.czxy.excel.Student2;
import java.util.ArrayList;
import java.util.List;
public class StudentDataListener2 extends AnalysisEventListener<Student2> {
//创建一个集合用于保存学生
private List<Student2> student2List = new ArrayList<>();
//创建一个变量用于判断
private Integer size = 4;
@Override
public void invoke(Student2 student2, AnalysisContext analysisContext) {
student2List.add(student2);
if (student2List.size()>size){
print();
}
}
//重新创建一个方法
public void print(){
student2List.forEach(System.out::println);
System.out.println("============");
//打印完成之后进行清空集合
student2List.clear();
}
//该方法
@Override
public void doAfterAllAnalysed(AnalysisContext analysisContext) {
System.out.println("打印完成");
//如果集合中还有数据进行打印
if (!student2List.isEmpty()){
print();
}
student2List.clear();
}
}
读操作
@Test
public void testRead(){
String file = path()+"student_demo2.xls";
//读操作
ExcelReader excelReader = EasyExcel.read(file, Student2.class, new StudentDataListener2()).build();
ReadSheet readSheet1 = EasyExcel.readSheet("爱吃豆的").build();
excelReader.read(readSheet1);
ReadSheet readSheet2 = EasyExcel.readSheet("土豆").build();
excelReader.read(readSheet2);
excelReader.finish();
}
Student
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Student {
@ExcelProperty("姓名")
private String name;
@ExcelProperty("年龄")
private Integer age;
}
Book
@Data
@NoArgsConstructor
@AllArgsConstructor
@HeadRowHeight(50)
@HeadFontStyle(fontName = "黑体",italic = BooleanEnum.TRUE, color = Font.COLOR_RED, underline = 2)
public class Book {
@ExcelProperty("编号")
private String id;
@ExcelProperty({"作者信息","姓名"})
private String authorName;
@ExcelProperty({"作者信息","年龄"})
private Integer authorAge;
@ExcelProperty({"书籍基本信息","标题"})
private String title;
@ContentFontStyle(fontName = "楷书",italic = BooleanEnum.TRUE, color = Font.COLOR_RED, underline = -1)
@ExcelProperty({"书籍基本信息","价格"})
private Double price;
@ExcelProperty({"书籍基本信息","出版日期"})
@DateTimeFormat("yyyy年MM月dd日")
private Date publishDate;
}
实现
public class TestManyObject {
// 获得当前项目的运行时的根目录
public String getPath() {
return this.getClass().getResource("/").getPath();
}
// 模拟数据
public List<Student> getStudentData() {
List<Student> list = new ArrayList<>();
for (int i = 0; i < 20; i++) {
list.add(new Student("张三" + i, 18 + i));
}
return list;
}
public List<Book> getBookData() {
List<Book> list = new ArrayList<>();
for (int i = 0; i < 20; i++) {
list.add(new Book(i+"" , "张三" + i , 18 +i, "坏蛋是怎么"+i, 998d+i, new Date()));
}
return list;
}
// 遍历map即可
private Map<Class<?>, List<?>> getData() {
Map<Class<?>, List<?>> map = new HashMap<>();
map.put(Student.class, getStudentData());
map.put(Book.class, getBookData());
return map;
}
@Test
public void testManyObject() {
String file = getPath() + "many_object.xlsx";
//1 开始写
ExcelWriter excelWriter = EasyExcel.write(file).build();
//2 依次写每一个对象
for(Map.Entry<Class<?>, List<?>> entry : getData().entrySet()) {
Class<?> clazz = entry.getKey(); //类型
List<?> data = entry.getValue(); //数据
WriteSheet writeSheet = EasyExcel.writerSheet(clazz.getSimpleName()).head(clazz).build();
excelWriter.write(data, writeSheet);
}
//3 写完成
excelWriter.finish();
}
} 我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看rubyzip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',