最近在想.net core NPOI 导入导出Excel,一开始感觉挺简单的,后来真的遇到很多坑。所以还是写一篇博客让其他人少走一些弯路,也方便忘记了再重温一遍。好了,多的不说,直接开始吧。

在.Net core 使用NPOI首先必须先安装DotNetCore.NPOI,第一种方法可以在管理Nuget包中安装,如图:

直接搜索DotNet.NPOI 安装就可以了。

第二种方法就是在Nuget程序包控制台中输入 Install-Package DotNetCore.NPOI  然后回车就可以了。

首先导出,和网上很多例子一样,我的想法一开始是用datatable导出到Excel中,但是感觉datatable的性能不够好,而且局限比较大,不过一开始我还是用datatable导出来了。

代码如下:

public static byte[] Output(DataTable dataTable, string[] tableTitle)
{
NPOI.SS.UserModel.IWorkbook workbook = new NPOI.XSSF.UserModel.XSSFWorkbook();
NPOI.SS.UserModel.ISheet sheet = workbook.CreateSheet("sheet");
IRow Title = null;
IRow rows = null;
for (int i = ; i <= dataTable.Rows.Count; i++)
{
//创建表头
if (i - == )
{
Title = sheet.CreateRow();
for (int k = ; k < tableTitle.Length + ; k++)
{
Title.CreateCell().SetCellValue("序号");
Title.CreateCell(k).SetCellValue(tableTitle[k - ]);
}
continue;
}
else
{
rows = sheet.CreateRow(i - );
for (int j = ; j <= dataTable.Columns.Count; j++)
{
rows.CreateCell().SetCellValue(i - );
rows.CreateCell(j).SetCellValue(dataTable.Rows[i - ][j - ].ToString());
}
}
} byte[] buffer = new byte[ * ];
using (MemoryStream ms = new MemoryStream())
{
workbook.Write(ms);
buffer = ms.ToArray();
ms.Close();
}
return buffer;
}

这里返回的是byte数组是为了方便在控制器调用方法后中返回一个文件给客户端。

return File(buffer, "application/ms-excel", "list.xlsx");   

这样客户端直接会有一个文件下载。

当然别忘了引用

using NPOI.SS.UserModel;
using NPOI.XSSF.UserModel;

这里XSSF是适用于xlsx,如果是xls的话引用using NPOI.HSSF.UserModel;

可以发现这里的数据源是datatable类型的,对于使用EF Core 操作数据库不是特别方便,所以还是改用list泛型去导出会方便很多。

代码如下:

public static byte[] OutputExcel(List<T> entitys, string[] title)
{
IWorkbook workbook = new XSSFWorkbook();
ISheet sheet = workbook.CreateSheet("sheet");
IRow Title = null;
IRow rows = null;
Type entityType = entitys[].GetType();
PropertyInfo[] entityProperties = entityType.GetProperties(); for (int i = ; i <= entitys.Count; i++)
{
if (i == )
{
Title = sheet.CreateRow();
for (int k = ; k < title.Length + ; k++)
{
Title.CreateCell().SetCellValue("序号");
Title.CreateCell(k).SetCellValue(title[k - ]);
} continue;
}
else
{
rows = sheet.CreateRow(i);
object entity = entitys[i - ];
for (int j = ; j <= entityProperties.Length; j++)
{
object[] entityValues = new object[entityProperties.Length];
entityValues[j - ] = entityProperties[j - ].GetValue(entity);
rows.CreateCell().SetCellValue(i);
rows.CreateCell(j).SetCellValue(entityValues[j - ].ToString());
}
}
} byte[] buffer = new byte[ * ];
using (MemoryStream ms = new MemoryStream())
{
workbook.Write(ms);
buffer = ms.ToArray();
ms.Close();
} return buffer;
}

因为list<T>可以直接得出行数,所以得出行数很方便,但是不知道list<T>有多少列,这个时候需要用到反射,记得引用System.Reflection 。通过获取list<T>类型去获取list<T>的属性。

然后通过 entityProperties.Length 获取列数。当然这里要留意第一列是表头,第二列才是数据,导出的时候从第二列开始。

通用引用类型去获取一行的值object entity = entitys[i - 1];然后通过属性数组去获取每一列的值,这里的逻辑有点乱,细心一点就一目了然了。object[] entityValues = new object[entityProperties.Length]; entityValues[j - 1] = entityProperties[j - 1].GetValue(entity);

这里是先写到内存中再导出,可能数据多了的时候会慢,不过一般是不会的。导出在这里就结束了。

然后是导入:

还是因为EF Core的原因,感觉还是导入成List<T>最方便,因为是公共方法,所以根据实体类型不同,要兼容所有实体类型,泛型似乎是最好的选择。

在网上找了很多资料,也看到了很多方法实现,不过不是特别适合我现在所在的项目,所以还是自己结合网上的重写,忘了借鉴了的文章的作者,若有侵权,马上删除。

代码如下:

 /// <summary>
/// 导入Excel
/// </summary>
/// <param name="file">导入文件</param>
/// <returns>List<T></returns>
public static List<T> InputExcel(IFormFile file)
{
List<T> list = new List<T> { }; MemoryStream ms = new MemoryStream();
file.CopyTo(ms);
ms.Seek(, SeekOrigin.Begin); IWorkbook workbook = new XSSFWorkbook(ms);
ISheet sheet = workbook.GetSheetAt();
IRow cellNum = sheet.GetRow();
var propertys = typeof(T).GetProperties();
string value = null;
int num = cellNum.LastCellNum; for (int i = ; i <= sheet.LastRowNum; i++)
{
IRow row = sheet.GetRow(i);
var obj = new T();
for (int j = ; j < num; j++)
{
value = row.GetCell(j).ToString();
string str = (propertys[j].PropertyType).FullName;
if (str == "System.String")
{
propertys[j].SetValue(obj, value, null);
}
else if (str == "System.DateTime")
{
DateTime pdt = Convert.ToDateTime(value, CultureInfo.InvariantCulture);
propertys[j].SetValue(obj, pdt, null);
}
else if (str == "System.Boolean")
{
bool pb = Convert.ToBoolean(value);
propertys[j].SetValue(obj, pb, null);
}
else if (str == "System.Int16")
{
short pi16 = Convert.ToInt16(value);
propertys[j].SetValue(obj, pi16, null);
}
else if (str == "System.Int32")
{
int pi32 = Convert.ToInt32(value);
propertys[j].SetValue(obj, pi32, null);
}
else if (str == "System.Int64")
{
long pi64 = Convert.ToInt64(value);
propertys[j].SetValue(obj, pi64, null);
}
else if (str == "System.Byte")
{
byte pb = Convert.ToByte(value);
propertys[j].SetValue(obj, pb, null);
}
else
{
propertys[j].SetValue(obj, null, null);
}
} list.Add(obj);
} return list;
}

数据源是IFormFile类型的,因为前端导入文件的方法是用layui的。所以是IFormFile。

同样我们也不知道list<T>是什么类型的,所以我们还是得用反射去获取类型,而且这里还是写到内存里面,就不会在服务器上留下文件,不过服务器内存小的要注意一下。

然后我们也不知道属性是什么类型的,所以要 string str = (propertys[j].PropertyType).FullName; 

判断属性的数据类型,然后在从Excel中遍历每一个单元格,输出到list中就可以了,记得要留意下数据在第二行开始导入进去。

大致的问题就是这些,重点就是反射去获取list<T>的类型。关于Excel的操作,这里就不详细说明了。

总的来说,NPOI可以脱离微软Excel操作Excel实在是非常方便,oledb真的非常不好用。

最后,我还需要学习的很多,有什么错误请指出。谢谢大家!

最新文章

  1. (转载)Windows常见性能计数器(较好的说明)
  2. EF(Entity Framework)发生错误”正在创建模型,此时不可使用上下文“的解决办法。 正在创建模型,此时不可使用上下文。如果在 OnModelCreating 方法内使用上下文或如果多个线程同时访问同一上下文实例,可能引发此异常。请注意不保证 DbContext 的实例成员和相关类是线程安全的。 临时解决了这个问题,在Context的构造函数中,禁用了自动初始化:
  3. JavaScript常用函数和方法
  4. cocos2dx音乐与音效
  5. TCPIP三次握手详情
  6. Python装饰器与面向切面编程
  7. fs event_socket
  8. vim之pydiction插件
  9. HDOJ 3547 DIY Cube 解题报告
  10. SQL分页语句总结
  11. 基于daridus认证的openvpn部署
  12. zookeeper[2] zookeeper原理(转)
  13. 【Android开发日记】妙用 RelativeLayout 实现3 段布局
  14. javaweb中重定向和请求转发(response.sendRedirect()和request.getRequestDispatcher(rul).forward(request,response)))的区别
  15. js查漏补缺【未完】
  16. DIV内容超出长度显示省略号,鼠标移上自动显示全部内容(EasyUI DataGrid)
  17. K - Find them, Catch them POJ - 1703 (带权并查集)
  18. CentOS6.5安装Maven3.2.5
  19. RichEdit选中文字右键菜单的实现
  20. Linux 安装nginx 及配置

热门文章

  1. uva 10635 LCS转LIS
  2. Spring入门案例 idea创建Spring项目
  3. vue——计算属性和侦听器
  4. Nginx管理(二)
  5. 从零开始的全栈工程师——jQuery
  6. SQL Server2008宝典 全书代码
  7. Vue2自定义指令改变DOM值后未刷新data中绑定属性的值
  8. php连接MySQL分析
  9. Linux --Apache服务搭建
  10. 对 Canal (增量数据订阅与消费)的理解