NPOI文件损坏

时间:2019-09-17 12:36:40

标签: c# excel corruption npoi

我制作了一个程序,用于将一个单元格拆分为两个单元格并将它们写在不同的工作表中,但是运行它后,excel文件被损坏。

IWorkbook workbook;
            using(FileStream stream = new FileStream(path, FileMode.Open, FileAccess.Read))
            {
                workbook = new XSSFWorkbook(stream);
            }
            IWorkbook newWorkbook = new XSSFWorkbook();

            ISheet sheet = workbook.GetSheetAt(0);
            ISheet oneWordSheet = newWorkbook.CreateSheet();
            ISheet moreWordsSheet = newWorkbook.CreateSheet();
            IRow tmpRow;
            for(int i = 5; i < 100/*sheet.LastRowNum*/ + 1; i++)
            {
                tmpRow = sheet.GetRow(i);
                string[] strings = tmpRow.GetCell(2).StringCellValue.Split(' ');
                string companyName = strings[0];
                bool parseFailed = true;
                for(int j = 1; parseFailed && j < strings.Length; j++)
                {
                    try
                    {
                        int.Parse(strings[j]);
                        parseFailed = false;
                    }
                    catch (FormatException)
                    {
                        companyName += strings[j];
                        j++;
                    }
                }
                tmpRow.CreateCell(4).SetCellValue(companyName);
                if(companyName.Trim().Split(' ').Length < 2)
                {
                    copyRowToSheet(tmpRow, oneWordSheet);
                }
                else
                {
                    copyRowToSheet(tmpRow, moreWordsSheet);
                }
            }
            using (FileStream stream = new FileStream(path, FileMode.Open, FileAccess.Write))
            {
                newWorkbook.Write(stream);
            }

我做了这样的copyRowToSheet方法。应该是正确的


private static void copyRowToSheet(IRow row, ISheet sheet)
        {
            IRow newRow = sheet.CreateRow(sheet.LastRowNum + 1);
            newRow.CreateCell(0).SetCellValue(row.GetCell(0).NumericCellValue);
            newRow.CreateCell(1).SetCellValue(row.GetCell(1).StringCellValue);
            newRow.CreateCell(2).SetCellValue(row.GetCell(4).StringCellValue);
            newRow.CreateCell(3).SetCellValue(row.GetCell(2).StringCellValue);
            newRow.CreateCell(4).SetCellValue(row.GetCell(3).StringCellValue);
        }

我尝试从工作簿而不是newWorkbook进行写操作,但是它仍然会损坏文件,我还尝试删除了copyRowToSheet方法(只是将if和else case都保留为空,但是结果没有改变...

编辑: 我试着删除程序的整个主体,只剩下这样:

IWorkbook workbook;
            using(FileStream stream = new FileStream(path, FileMode.Open, FileAccess.Read))
            {
                workbook = new XSSFWorkbook(stream);
                stream.Close();
            }
            IWorkbook newWorkbook = new XSSFWorkbook();


            using (FileStream stream = new FileStream(path, FileMode.Open, FileAccess.Write))
            {
                workbook.Write(stream);
                stream.Close();
            }

如果我没记错的话,应该只读取文件,然后将其保存回去而不进行任何编辑,但是它仍然会损坏文件

1 个答案:

答案 0 :(得分:0)

几周前,我刚接触npoi时遇到了同样的问题。在教程和博客中不断重复您正在使用的代码,这是一个非常棘手的问题。

当您创建第二个FileStream以便将电子表格写回到磁盘时,会出现问题。您正在写入与先前阅读的文件相同的文件。

写入现有文件时FileMode.Open的行为是将数据附加到文件末尾。这样一来,您就可以在一个文件中拥有2个excel电子表格,当您打开该文件时,它们会被声明为损坏。

FileMode.Create另一方面会覆盖现有文件,因此更可能是您需要的文件。

using (FileStream stream = new FileStream(path, FileMode.Create, FileAccess.Write))
{
     workbook.Write(stream);
     stream.Close();
}

这是docs文件模式FileFile,因为您可能更喜欢Create的替代方法。

Doc for FileMode

相关问题