C#通过NPOI操作Excel的实例代码

作者:独钓寒江雪丶 时间:2022-01-20 17:26:29 

C#操作Excel的方法有很多种,常见的有微软官方的OLE Automation,Apache的POI等。这里介绍的是POI翻译成C#的NPOI。

POI是Apache的通过Java操作Office的一个API,可以对Excel,Word,PPT等进行操作,十分的强大。然后就被翻译成C#版本的NPOI了,和log4j与log4net很相似。

好像在NPOI的.net4.0版本之前是不支持office2007及以上的XML格式的,但是最新的版本已经支持了。只需要下载并引用下面五个程序集就能使用了。

C#通过NPOI操作Excel的实例代码

这里提供一个操作Excel的类,类中提供了4个方法,两个导出,两个导入。可以通过DataSet导出拥有多个Sheet的Excel文件,也可以通过DataTable导出拥有一个Sheet的Excel。导入也是一样,通过指定Sheet索引,导出DataTable,或者直接导出所有Sheet返回一个DataSet。


public class ExcelHelper
 {
   /// <summary>
   /// 根据Excel和Sheet返回DataTable
   /// </summary>
   /// <param name="filePath">Excel文件地址</param>
   /// <param name="sheetIndex">Sheet索引</param>
   /// <returns>DataTable</returns>
   public static DataTable GetDataTable(string filePath, int sheetIndex)
   {
     return GetDataSet(filePath, sheetIndex).Tables[0];
   }

/// <summary>
   /// 根据Excel返回DataSet
   /// </summary>
   /// <param name="filePath">Excel文件地址</param>
   /// <param name="sheetIndex">Sheet索引,可选,默认返回所有Sheet</param>
   /// <returns>DataSet</returns>
   public static DataSet GetDataSet(string filePath, int? sheetIndex = null)
   {
     DataSet ds = new DataSet();
     IWorkbook fileWorkbook;
     using (FileStream fs = new FileStream(filePath, FileMode.Open, FileAccess.Read))
     {
       if (filePath.Last() == 's')
       {
         try
         {
           fileWorkbook = new HSSFWorkbook(fs);
         }
         catch (Exception ex)
         {
           throw ex;
         }
       }
       else
       {
         try
         {
           fileWorkbook = new XSSFWorkbook(fs);
         }
         catch
         {
           fileWorkbook = new HSSFWorkbook(fs);
         }
       }
     }

for (int i = 0; i < fileWorkbook.NumberOfSheets; i++)
     {
       if (sheetIndex != null && sheetIndex != i)
         continue;
       DataTable dt = new DataTable();
       ISheet sheet = fileWorkbook.GetSheetAt(i);

//表头
       IRow header = sheet.GetRow(sheet.FirstRowNum);
       List<int> columns = new List<int>();
       for (int j = 0; j < header.LastCellNum; j++)
       {
         object obj = GetValueTypeForXLS(header.GetCell(j) as HSSFCell);
         if (obj == null || obj.ToString() == string.Empty)
         {
           dt.Columns.Add(new DataColumn("Columns" + j.ToString()));
         }
         else
           dt.Columns.Add(new DataColumn(obj.ToString()));
         columns.Add(j);
       }
       //数据
       IEnumerator rows = sheet.GetEnumerator();
       while (rows.MoveNext())
       {
         int j = sheet.FirstRowNum + 1;
         DataRow dr = dt.NewRow();
         bool hasValue = false;
         foreach (int K in columns)
         {
           dr[K] = GetValueTypeForXLS(sheet.GetRow(K).GetCell(K) as HSSFCell);
           if (dr[K] != null && dr[K].ToString() != string.Empty)
           {
             hasValue = true;
           }
         }
         if (hasValue)
         {
           dt.Rows.Add(dr);
         }
         j++;
       }
       ds.Tables.Add(dt);
     }

return ds;
   }

/// <summary>
   /// 根据DataTable导出Excel
   /// </summary>
   /// <param name="dt">DataTable</param>
   /// <param name="file">保存地址</param>
   public static void GetExcelByDataTable(DataTable dt, string file)
   {
     DataSet ds = new DataSet();
     ds.Tables.Add(dt);
     GetExcelByDataSet(ds, file);
   }

/// <summary>
   /// 根据DataSet导出Excel
   /// </summary>
   /// <param name="ds">DataSet</param>
   /// <param name="file">保存地址</param>
   public static void GetExcelByDataSet(DataSet ds, string file)
   {
     IWorkbook fileWorkbook = new HSSFWorkbook();
     int index = 0;
     foreach (DataTable dt in ds.Tables)
     {
       index++;
       ISheet sheet = fileWorkbook.CreateSheet("Sheet" + index);

//表头
       IRow row = sheet.CreateRow(0);
       for (int i = 0; i < dt.Columns.Count; i++)
       {
         ICell cell = row.CreateCell(i);
         cell.SetCellValue(dt.Columns[i].ColumnName);
       }

//数据
       for (int i = 0; i < dt.Rows.Count; i++)
       {
         IRow row1 = sheet.CreateRow(i + 1);
         for (int j = 0; j < dt.Columns.Count; j++)
         {
           ICell cell = row1.CreateCell(j);
           cell.SetCellValue(dt.Rows[i][j].ToString());
         }
       }
     }

//转为字节数组
     MemoryStream stream = new MemoryStream();
     fileWorkbook.Write(stream);
     var buf = stream.ToArray();

//保存为Excel文件
     using (FileStream fs = new FileStream(file, FileMode.Create, FileAccess.Write))
     {
       fs.Write(buf, 0, buf.Length);
       fs.Flush();
     }
   }

/// <summary>
   /// 根据单元格将内容返回为对应类型的数据
   /// </summary>
   /// <param name="cell">单元格</param>
   /// <returns>数据</returns>
   private static object GetValueTypeForXLS(HSSFCell cell)
   {
     if (cell == null)
       return null;
     switch (cell.CellType)
     {
       case CellType.Blank: //BLANK:
         return null;
       case CellType.Boolean: //BOOLEAN:
         return cell.BooleanCellValue;
       case CellType.Numeric: //NUMERIC:
         return cell.NumericCellValue;
       case CellType.String: //STRING:
         return cell.StringCellValue;
       case CellType.Error: //ERROR:
         return cell.ErrorCellValue;
       case CellType.Formula: //FORMULA:
       default:
         return "=" + cell.CellFormula;
     }
   }
 }

这里面可以有一些有意思的操作,比如版本兼容问题。这里通过多态很好的实现了兼容,但是如果是2007版本的xlsm被修改为xsl的后缀怎么办呢,或者2003版本的被修改为xlsm后缀怎么办呢。2003版本改为xlsm还是可以将其视为xls来处理的,但是2007改为xls就不行了。这时候可以强行修改文件的后缀名再打开。

但是上面的代码没有实现这个功能,两个原因:一、这样做不是很安全。二、这时候需要修改系统中其它地方此文件的文件名,放在类中实现不是很方便。

来源:http://www.cnblogs.com/xueyeduling/p/6254799.html

标签:npoi,excel
0
投稿

猜你喜欢

  • Android中AsyncTask与handler用法实例分析

    2023-12-04 00:01:52
  • C#实现xml文件反序列化读入数据到object的方法

    2023-08-20 00:43:08
  • Android中自定义标题栏样式的两种方法

    2022-05-03 09:06:01
  • java spring mvc处理器映射器介绍

    2021-11-22 01:01:35
  • 如何在MyBatis中实现DataSource

    2021-11-29 13:49:48
  • Java中构造、生成XML简明教程

    2021-10-03 09:33:58
  • java StringBuilder类的详解及简单实例

    2022-04-15 01:24:44
  • C#中单问号(?)和双问号(??)的用法整理

    2021-08-27 02:50:53
  • Java实现数据库连接池的方法

    2023-11-28 08:57:30
  • Java中ArrayList和SubList的坑面试题

    2023-09-21 06:56:21
  • Android实用的代码片段 常用代码总结

    2022-02-02 20:29:53
  • Java反射及性能详细

    2023-08-20 20:49:42
  • 详解JAVA动态代理

    2023-11-24 22:52:04
  • dubbo整合springboot新手入门教程详解

    2022-07-29 21:19:00
  • java使用淘宝API读写json实现手机归属地查询功能代码

    2021-05-31 03:27:38
  • Java动态添加view的方法

    2023-06-11 04:20:38
  • java 创建线程的方法总结

    2023-02-25 20:20:30
  • WPF实现3D翻牌式倒计时特效

    2021-11-11 16:08:21
  • 使用C语言编写基于TCP协议的Socket通讯程序实例分享

    2023-07-05 03:30:03
  • Android 处理OnItemClickListener时关于焦点颜色的设置问题

    2021-09-02 01:47:00
  • asp之家 软件编程 m.aspxhome.com