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

 更新时间:2020年6月25日 11:21  点击:2295

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格式的,但是最新的版本已经支持了。只需要下载并引用下面五个程序集就能使用了。

这里提供一个操作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就不行了。这时候可以强行修改文件的后缀名再打开。

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持猪先飞。

[!--infotagslink--]

相关文章

  • SpringBoot实现excel文件生成和下载

    这篇文章主要为大家详细介绍了SpringBoot实现excel文件生成和下载,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-02-09
  • c#读取excel方法实例分析

    这篇文章主要介绍了c#读取excel方法,实例分析了C#读取excel文件的原理与相关技巧,需要的朋友可以参考下...2020-06-25
  • Python导入数值型Excel数据并生成矩阵操作

    这篇文章主要介绍了Python导入数值型Excel数据并生成矩阵操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-06-09
  • C# winform打开Excel文档的方法总结(必看篇)

    下面小编就为大家带来一篇C# winform打开Excel文档的方法总结(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • C# 导出Excel的6种简单方法实现

    C# 导出 Excel 的6种简单方法:数据表导出到 Excel,对象集合导出到 Excel,数据库导出到 Excel,微软网格控件导出到 Excel,数组导出到 Excel,CSV 导出到 Excel,你都会了吗?需要的朋友们下面随着小编来一起学习学习吧...2020-06-25
  • C#实现Excel表数据导入Sql Server数据库中的方法

    这篇文章主要介绍了C#实现Excel表数据导入Sql Server数据库中的方法,结合实例形式详细分析了C#读取Excel表数据及导入Sql Server数据库的具体操作步骤与相关操作技巧,需要的朋友可以参考下...2020-06-25
  • C#实现将数据导出到word或者Excel中的方法

    这篇文章主要介绍了C#实现将数据导出到word或者Excel中的方法,涉及C#操作word及Excel格式文件的方法,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • C#使用Ado.Net更新和添加数据到Excel表格的方法

    这篇文章主要介绍了C#使用Ado.Net更新和添加数据到Excel表格的方法,较为详细的分析了OLEDB的原理与使用技巧,可实现较为方便的操作Excel数据,需要的朋友可以参考下...2020-06-25
  • C#使用Aspose.Cells创建和读取Excel文件

    这篇文章主要为大家详细介绍了C#使用Aspose.Cells创建和读取Excel文件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-06-25
  • C#使用oledb导出数据到excel的方法

    这篇文章主要介绍了C#使用oledb导出数据到excel的方法,结合实例形式分析了C#操作oledb导出数据的相关技巧与注意事项,需要的朋友可以参考下...2020-06-25
  • C#操作excel打印的示例

    这篇文章主要介绍了C#操作excel打印的示例,帮助大家利用c#打印表格,提高办公效率,感兴趣的朋友可以了解下...2020-12-08
  • C#删除Excel中的图片实例代码

    在本篇文章里小编给大家分享了关于C#删除Excel中的图片的实例代码内容,有兴趣的朋友们参考学习下。...2020-06-25
  • C#使用Aspose.Cells控件读取Excel

    本文介绍Aspose.Cells基础的用法,供大家参考。...2020-06-25
  • C#中将ListView中数据导出到Excel的实例方法

    首先 你需要添加引用Microsoft Excel 11.0 Object Library...2020-06-25
  • python操作mysql、excel、pdf的示例

    这篇文章主要介绍了python操作mysql、excel、pdf的示例,帮助大家更好的理解和学习使用python,感兴趣的朋友可以了解下...2021-03-29
  • python3 循环读取excel文件并写入json操作

    这篇文章主要介绍了python3 循环读取excel文件并写入json操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-14
  • C#将Excel转成PDF的方法

    今天小编就为大家分享一篇关于C#将Excel转成PDF的方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...2020-06-25
  • C# WinForm导出Excel方法介绍

    在.NET应用中,导出Excel是很常见的需求,导出Excel报表大致有以下三种方式:Office PIA,文件流和NPOI开源库,本文只介绍前两种方式...2020-06-25
  • phpexcel导入xlsx文件报错xlsx is not recognised as an OLE file 怎么办

    phpexcel是一款php读写excel的插件了,小编有一个这样的功能要来实现,但是在导入xlsx时发现xlsx is not recognised as an OLE file 了,但是导入xls是没有问题了,碰到这种...2016-11-25
  • 解决python pandas读取excel中多个不同sheet表格存在的问题

    这篇文章主要介绍了解决python pandas读取excel中多个不同sheet表格存在的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-07-14