.NetCore NPOI 读取excel内容及单元格内图片

由于数据方提供的数据在excel文件中不止有文字内容还包含图片信息,于是编写相关测试代码,读取excel文件内容及图片信息.

本文使用的是 NPOI-2.6.2 版本,此版本持.Net4.7.2;.NetStandard2.0;.NetStandard2.1;.Net6.0+。

测试文档内容,如下图:

保存后的图片:

打开图片显示正常:

编写读取数据方法,代码如:

cs 复制代码
static public DataTable ReadExcel(string filePath, string _sDirImg)
{
    using (FileStream file = new FileStream(filePath, FileMode.Open, FileAccess.Read))
    {
        IWorkbook workbook = null;
        string sExt = Path.GetExtension(filePath).ToLower();

        if (sExt == ".xlsx")
        {
            workbook = new XSSFWorkbook(file);//新版本的Excel(.xlsx) 
        }
        else
        {
            workbook = new HSSFWorkbook(file);//老版本的Excel(.xls) 

        }

        // 读取第一个 Sheet
        ISheet sheet = workbook.GetSheetAt(0);

        if (sExt == ".xlsx")
        {
            fnReadImageXSSF(sheet, _sDirImg);//新版本的Excel(.xlsx) 

        }
        else
        {
            fnReadImageHSSF(sheet, _sDirImg);//老版本的Excel(.xls) 
        }

        // 创建 DataTable
        DataTable dataTable = new DataTable(sheet.SheetName);

        // 读取表头
        IRow headerRow = sheet.GetRow(0);
        for (int i = 0; i < headerRow.LastCellNum; i++)
        {
            ICell cell = headerRow.GetCell(i);
            dataTable.Columns.Add(cell.ToString(), typeof(string));
        }

        // 读取数据行
        for (int rowNum = 1; rowNum <= sheet.LastRowNum; rowNum++)
        {
            IRow row = sheet.GetRow(rowNum);
            DataRow dataRow = dataTable.NewRow();

            for (int i = 0; i < row.LastCellNum; i++)
            {
                ICell cell = row.GetCell(i);

                if (null == cell) continue;

                dataRow[i] = cell.ToString();
            }

            dataTable.Rows.Add(dataRow);
        }

        return dataTable;
    }
}

编写对应不同版本的excel文件,读取图片方法,读取excel .xlsx文件内图片:

cs 复制代码
/// <summary>
/// 读取excel .xlsx文件内图片
/// </summary>
/// <param name="sheet"></param>
/// <param name="_sDirImg"></param>
static public void fnReadImageXSSF(ISheet sheet, string _sDirImg)
{
    // 读取图像信息
    foreach (XSSFShape shape in ((XSSFDrawing)sheet.DrawingPatriarch).GetShapes())
    {
        if (shape is XSSFPicture)
        {
            XSSFPicture picture = (XSSFPicture)shape;

            // 获取图片所在单元格的行号和列号
            int rowIndex = picture.GetPreferredSize().Row1;
            int colIndex = picture.GetPreferredSize().Col1;

            // 获取图像文件格式
            string imageFormat = picture.PictureData.MimeType switch
            {
                "image/jpeg" => "jpeg",
                "image/png" => "png",
                "image/gif" => "gif",
                "image/bmp" => "bmp",
                _ => "jpg"
            };

            // 保存图像文件
            string outputFileName = _sDirImg + $"{rowIndex}-{colIndex}-{Guid.NewGuid()}.{imageFormat}";
            using (FileStream imageFile = new FileStream(outputFileName, FileMode.Create))
            {
                imageFile.Write(picture.PictureData.Data, 0, picture.PictureData.Data.Length);
            }

            Console.WriteLine($"Saved image: {outputFileName}");
        }
       
    }
}

读取excel .xls文件内图片,代码如下:

cs 复制代码
/// <summary>
/// 读取excel .xls文件内图片
/// </summary>
/// <param name="sheet"></param>
/// <param name="_sDirImg"></param>
static public void fnReadImageHSSF(ISheet sheet, string _sDirImg)
{
    // 读取图像信息
    foreach (HSSFShape shape in ((HSSFPatriarch)sheet.DrawingPatriarch).Children)
    {
        if (shape is HSSFPicture)
        {
            HSSFPicture picture = (HSSFPicture)shape;

            // 获取图片所在单元格的行号和列号
            int rowIndex = picture.GetPreferredSize().Row1;
            int colIndex = picture.GetPreferredSize().Col1;

            // 获取图像文件格式
            string imageFormat = picture.PictureData.MimeType switch
            {
                "image/jpeg" => "jpeg",
                "image/png" => "png",
                "image/gif" => "gif",
                "image/bmp" => "bmp",
                _ => "jpg"
            };

            // 保存图像文件
            string outputFileName = _sDirImg + $"{rowIndex}-{colIndex}-{Guid.NewGuid()}.{imageFormat}";
            using (FileStream imageFile = new FileStream(outputFileName, FileMode.Create))
            {
                imageFile.Write(picture.PictureData.Data, 0, picture.PictureData.Data.Length);
            }

            Console.WriteLine($"Saved image: {outputFileName}");
        }
    }
}

正常处理应该是读取到图片保存成功后,处理datatable图片列的相关地址,如:uploads/xxx/xx.jpg ,返回保存在服务器上的地址,以便前端访问或保存到数据库等,本文并没有处理,有需要的伙伴自行处理吧,希望本文对你有帮助。

相关推荐
棉晗榜15 天前
C# .net core添加单元测试项目,依赖注入接口测试
单元测试·c#·.netcore
时光追逐者15 天前
.NET初级软件工程师面试经验分享
经验分享·面试·职场和发展·c#·.net·.netcore
忧郁的蛋~17 天前
.NET Core 实现缓存的预热的方式
缓存·c#·.net·.netcore
csdn_aspnet18 天前
C# .NET Core 源代码生成器(dotnet source generators)
c#·.netcore
时光追逐者18 天前
C#/.NET/.NET Core技术前沿周刊 | 第 42 期(2025年6.9-6.15)
c#·.net·.netcore
csdn_aspnet19 天前
使用 C# 源生成器(Source Generators)进行高效开发:增强 Blazor 及其他功能
c#·.netcore
lgaof65822@gmail.com21 天前
Asp.Net Core SignalR导入数据
前端·后端·asp.net·.netcore
眸笑丶21 天前
.NET Core 数据库连接字符串加密与解密
数据库·oracle·.netcore
时光追逐者23 天前
C#/.NET/.NET Core技术前沿周刊 | 第 41 期(2025年6.1-6.8)
c#·.net·.netcore
lgaof65822@gmail.com1 个月前
ASP.NET Core SignalR - 部分客户端消息发送
后端·asp.net·.netcore