之前我们在使用npoi导入excel表格的时候,往往会遇见那种带有合并单元格的数据在导入的时候出现合并为空的问题,

也就是只有第一条有数据,其余均为空白。在网上翻了半天也没有找到合适的解决方案,最后还是想着静下心来好好研究一番,于是...


我们先来看看通常我们的导入方式,如下图,这是我们的导入模板,带有合并单元格

下面我们来看看对应的npoi所读到的DataTable数据

你会发现,只有第一行有数据,其余我们合并的单元格为空值,那我们导入到数据库必将会出错。

于是去查看代码,发现原来的获取单元格值的时候并没有去判断单元格是否进行了合并。而正好NPOI里面正好

cell.IsMergedCell 的属性,于是我们在检测列的单元格是否合并,并且行数大于1的时候,我们就可以获取值。

一旦检测到单元格合并,并且单元格的值为空值,则让它去取上一行的值。否则直接取单元格的值即可

 //读取每列
for (int j = ; j < row.Cells.Count; j++)
{
ICell cell = row.GetCell(j); //一个单元格
if (cell.IsMergedCell && r > ) //检测列的单元格是否合并
{
//dr[j] = dt.Rows[r - 2][j];
var cellValue = GetCellValue(cell);
if (string.IsNullOrEmpty(cellValue))
{
dr[j] = dt.Rows[r - ][j];
}
else
{
dr[j] = cellValue; //获取单元格的值 if (string.IsNullOrWhiteSpace(dr[j].ToString()) && j > )
{
dr[j] = dr[j - ];
}
}
}
else
{
dr[j] = GetCellValue(cell); //获取单元格的值 if (string.IsNullOrWhiteSpace(dr[j].ToString()) && j > )
{
dr[j] = dr[j - ];
}
}
if (dr[j].ToString() != "")//全为空则不取
{
result = true;
}
}
if (result == true)
{
dt.Rows.Add(dr); //把每行追加到DataTable
}

下面附上完整代码

 /// <summary>
/// 导入Excel 带合并单元格 zhangyu 20200428
/// </summary>
/// <param name="filePath">excel文件路径</param>
/// <returns></returns>
public DataTable ExcelToDataTable(string filePath)
{
System.Web.HttpFileCollection files = System.Web.HttpContext.Current.Request.Files;
if (files.Count > && files[] != null)
{
if (filePath.IndexOf(".xlsx") > )
{
WorkBooks = new XSSFWorkbook(files[].InputStream);
}
else
{
WorkBooks = new HSSFWorkbook(files[].InputStream);
}
}
else
{
FStream = new FileStream(filePath, FileMode.Open, FileAccess.Read);
if (filePath.IndexOf(".xlsx") > )
{
WorkBooks = new XSSFWorkbook(FStream);
}
else
{
WorkBooks = new HSSFWorkbook(FStream);
}
} DataTable dt = new DataTable();
IWorkbook wk = WorkBooks;
//获取后缀名
string extension = filePath.Substring(filePath.LastIndexOf(".")).ToString().ToLower();
//判断是否是excel文件
if (extension == ".xlsx" || extension == ".xls")
{
//获取第一个sheet
ISheet sheet = wk.GetSheetAt();
//获取第一行
IRow headrow = sheet.GetRow();
//创建列
for (int i = headrow.FirstCellNum; i < headrow.Cells.Count; i++)
{
ICell cell = headrow.GetCell(i);
dt.Columns.Add(cell.ToString());
}
//读取每行,从第二行起
for (int r = ; r <= sheet.LastRowNum; r++)
{
bool result = false;
DataRow dr = dt.NewRow();
//获取当前行
IRow row = sheet.GetRow(r);
//读取每列
for (int j = ; j < row.Cells.Count; j++)
{
ICell cell = row.GetCell(j); //一个单元格
if (cell.IsMergedCell && r > ) //检测列的单元格是否合并
{
//dr[j] = dt.Rows[r - 2][j];
var cellValue = GetCellValue(cell);
if (string.IsNullOrEmpty(cellValue))
{
dr[j] = dt.Rows[r - ][j];
}
else
{
dr[j] = cellValue; //获取单元格的值
if (string.IsNullOrWhiteSpace(dr[j].ToString()) && j > )
{
dr[j] = dr[j - ];
}
}
}
else
{
dr[j] = GetCellValue(cell); //获取单元格的值
if (string.IsNullOrWhiteSpace(dr[j].ToString()) && j > )
{
dr[j] = dr[j - ];
}
}
if (dr[j].ToString() != "")//全为空则不取
{
result = true;
}
}
if (result == true)
{
dt.Rows.Add(dr); //把每行追加到DataTable
}
}
}
return dt;
}
/// <summary>
/// 对单元格进行判断取值
/// </summary>
/// <param name="cell"></param>
/// <returns></returns>
private static string GetCellValue(ICell cell)
{
if (cell == null)
return string.Empty;
switch (cell.CellType)
{
case CellType.Blank: //空数据类型 这里类型注意一下,不同版本NPOI大小写可能不一样,有的版本是Blank(首字母大写)
return string.Empty;
case CellType.Boolean: //bool类型
return cell.BooleanCellValue.ToString();
case CellType.Error:
return cell.ErrorCellValue.ToString();
case CellType.Numeric: //数字类型
if (HSSFDateUtil.IsCellDateFormatted(cell))//日期类型
{
return cell.DateCellValue.ToString();
}
else //其它数字
{
return cell.NumericCellValue.ToString();
}
case CellType.Unknown: //无法识别类型
default: //默认类型
return cell.ToString();//
case CellType.String: //string 类型
{
if (cell.IsMergedCell){}
return cell.StringCellValue;
} case CellType.Formula: //带公式类型
try
{
HSSFFormulaEvaluator e = new HSSFFormulaEvaluator(cell.Sheet.Workbook);
e.EvaluateInCell(cell);
return cell.ToString();
}
catch
{
return cell.NumericCellValue.ToString();
}
}
}

最新文章

  1. SQL Server 无法连接到服务器。SQL Server 复制需要有实际的服务器名称才能连接到服务器。请指定实际的服务器名称。
  2. JavaScript检测文件上传的类型与大小
  3. Linq to sql 的语法
  4. 用dom4j解析xml 报java.lang.NoClassDefFoundError:org/jaxen/JaxenException
  5. OSGi 的核心配置、动态化及问题
  6. 解决:安装SQl 2008为SQL Server代理服务提供的凭据无效
  7. 轻量级的内部测试过程r \\ u0026研发团队
  8. 2015十大顶级开源ERP系统点评
  9. Chapter 5. MPEG-4 Visual
  10. enum与typedef enum
  11. 【朝花夕拾】Android性能篇之(一)序言及JVM
  12. 在Linux命令行中以图形化窗口打开文件夹
  13. [原创]CobaltStrike &amp; Metasploit Shellcode一键免杀工具
  14. jsp/servlet学习五之jsp表达式语言初窥
  15. Linux&#160;学习笔记之超详细基础linux命令&#160;Part&#160;14
  16. 【Java】【异常】
  17. Sqlserver2008及以上使用全文索引排除干扰词
  18. 记开发个人图书收藏清单小程序开发(十)DB开发——新增图书信息
  19. Docker Manager for Docker Swarm deploy
  20. postgresql双机热备、高可用方案(采用pacemaker+corosync实现)

热门文章

  1. Maven+JSP+Servlet+JDBC+Redis+Mysql实现的黑马旅游网
  2. vue2.x学习笔记(十七)
  3. bm25算法和tfidf
  4. java 多线--静态代理模式
  5. XSS Cheat Sheet(basics and advanced)
  6. latex-列表环境
  7. Spring Boot @EnableAutoConfiguration和 @Configuration的区别
  8. SpringBoot 集成Swagger2自动生成文档和导出成静态文件
  9. Hard filters (by GATK)
  10. java switch用法