一个用微软官方的OpenXml读写Excel 目前网上不太普及的方法。
新版本的xlsx是使用新的存储格式,貌似是处理过的XML。
传统的excel处理方法,我真的感觉像屎。用Oldeb不方便,用com组件要实际调用excel打开关闭,很容易出现死。
对于OpenXML我网上搜了一下,很多人没有介绍。所以我就这里推荐下,相信会成为信息系统开发的必备。
先写出个例子,会发现如此的简介:
using System;
using System.Collections.Generic;
using System.Text;
using XFormular.config;
using System.IO;
using com.xtar.amfx;
using System.Runtime.Serialization.Formatters.Binary;
using System.Data; namespace XFormular.test
{
class Class1
{
public void test()
{
DataTable table = new DataTable("");
table.Columns.Add("");
for (int i = ; i < ; i++)
{
DataRow row = table.NewRow();
row[] = i;
table.Rows.Add(row);
} List<DataTable> lsit = new List<DataTable>();
lsit.Add(table); OpenXmlSDKExporter.Export(AppDomain.CurrentDomain.BaseDirectory + "\\excel.xlsx", lsit);
}
}
}
写出代码:
using System;
using System.IO;
using System.Windows.Forms;
using DocumentFormat.OpenXml;
using DocumentFormat.OpenXml.Packaging;
using DocumentFormat.OpenXml.Spreadsheet;
using DocumentFormat.OpenXml.Extensions;
using System.Collections.Generic;
using System.Data;
using System.Text.RegularExpressions; namespace XFormular
{
//http://simpleooxml.codeplex.com/
//http://www.pin5i.com/showtopic-21817.html
//http://blog.csdn.net/lbj147123/article/details/6603942
class OpenXmlSDKExporter
{
private static string[] Level = {"A", "B", "C", "D", "E", "F", "G",
"H", "I", "G", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T",
"U", "V", "W", "X", "Y", "Z" }; public static List<DataTable> Import(string path)
{
List<DataTable> tables = new List<DataTable>(); if (path.EndsWith(ExcelHelper.POSTFIX_SVN))
return tables; using (MemoryStream stream = SpreadsheetReader.StreamFromFile(path))
{
using (SpreadsheetDocument doc = SpreadsheetDocument.Open(stream, true))
{
foreach (Sheet sheet in doc.WorkbookPart.Workbook.Descendants<Sheet>())
{
DataTable table = new DataTable(sheet.Name.Value); WorksheetPart worksheet = (WorksheetPart)doc.WorkbookPart.GetPartById(sheet.Id); List<string> columnsNames = new List<string>(); foreach (Row row in worksheet.Worksheet.Descendants<Row>())
{
foreach (Cell cell in row)
{
string columnName = Regex.Match(cell.CellReference.Value, "[a-zA-Z]+").Value; if (!columnsNames.Contains(columnName))
{
columnsNames.Add(columnName);
} }
} columnsNames.Sort(CompareColumn); foreach (string columnName in columnsNames)
{
table.Columns.Add(columnName);
} foreach (Row row in worksheet.Worksheet.Descendants<Row>())
{
DataRow tableRow = table.NewRow();
table.Rows.Add(tableRow); foreach (Cell cell in row)
{
string columnName = Regex.Match(cell.CellReference.Value, "[a-zA-Z]+").Value;
tableRow[columnName] = GetValue(cell, doc.WorkbookPart.SharedStringTablePart);
}
} if (table.Rows.Count <= )
continue;
if (table.Columns.Count <= )
continue; tables.Add(table);
}
}
} return tables;
} public static String GetValue(Cell cell, SharedStringTablePart stringTablePart)
{ if (cell.ChildElements.Count == ) return null; //get cell value String value = cell.CellValue.InnerText; //Look up real value from shared string table if ((cell.DataType != null) && (cell.DataType == CellValues.SharedString)) value = stringTablePart.SharedStringTable .ChildElements[Int32.Parse(value)] .InnerText; return value; } public static void Export(string path, List<DataTable> tables)
{
using (MemoryStream stream = SpreadsheetReader.Create())
{
using (SpreadsheetDocument doc = SpreadsheetDocument.Open(stream, true))
{
SpreadsheetWriter.RemoveWorksheet(doc, "Sheet1");
SpreadsheetWriter.RemoveWorksheet(doc, "Sheet2");
SpreadsheetWriter.RemoveWorksheet(doc, "Sheet3"); foreach (DataTable table in tables)
{
WorksheetPart sheet = SpreadsheetWriter.InsertWorksheet(doc, table.TableName);
WorksheetWriter writer = new WorksheetWriter(doc, sheet); SpreadsheetStyle style = SpreadsheetStyle.GetDefault(doc); foreach (DataRow row in table.Rows)
{
for (int i = ; i < table.Columns.Count; i++)
{
string columnName = SpreadsheetReader.GetColumnName("A", i);
string location = columnName + (table.Rows.IndexOf(row) + );
writer.PasteText(location, row[i].ToString(), style);
}
} writer.Save();
}
SpreadsheetWriter.StreamToFile(path, stream);//保存到文件中
}
}
} private static int CompareColumn(string x, string y)
{
int xIndex = Letter_to_num(x);
int yIndex = Letter_to_num(y);
return xIndex.CompareTo(yIndex);
} /// <summary>
/// 数字26进制,转换成字母,用递归算法
/// </summary>
/// <param name="value"></param>
/// <returns></returns>
private static string Num_to_letter(int value)
{
//此处判断输入的是否是正确的数字,略(正在表达式判断)
int remainder = value % ;
//remainder = (remainder == 0) ? 26 : remainder;
int front = (value - remainder) / ;
if (front < )
{
return Level[front - ] + Level[remainder];
}
else
{
return Num_to_letter(front) + Level[remainder];
}
//return "";
} /// <summary>
/// 26进制字母转换成数字
/// </summary>
/// <param name="letter"></param>
/// <returns></returns>
private static int Letter_to_num(string str)
{
//此处判断是否是由A-Z字母组成的字符串,略(正在表达式片段)
char[] letter = str.ToCharArray(); //拆分字符串
int reNum = ;
int power = ; //用于次方算值
int times = ; //最高位需要加1
int num = letter.Length;//得到字符串个数
//得到最后一个字母的尾数值
reNum += Char_num(letter[num - ]);
//得到除最后一个字母的所以值,多于两位才执行这个函数
if (num >= )
{
for (int i = num - ; i > ; i--)
{
power = ;//致1,用于下一次循环使用次方计算
for (int j = ; j < i; j++) //幂,j次方,应该有函数
{
power *= ;
}
reNum += (power * (Char_num(letter[num - i - ]) + times)); //最高位需要加1,中间位数不需要加一
times = ;
}
}
//Console.WriteLine(letter.Length);
return reNum;
} /// <summary>
/// 输入字符得到相应的数字,这是最笨的方法,还可用ASIICK编码;
/// </summary>
/// <param name="ch"></param>
/// <returns></returns>
private static int Char_num(char ch)
{
switch (ch)
{
case 'A':
return ;
case 'B':
return ;
case 'C':
return ;
case 'D':
return ;
case 'E':
return ;
case 'F':
return ;
case 'G':
return ;
case 'H':
return ;
case 'I':
return ;
case 'J':
return ;
case 'K':
return ;
case 'L':
return ;
case 'M':
return ;
case 'N':
return ;
case 'O':
return ;
case 'P':
return ;
case 'Q':
return ;
case 'R':
return ;
case 'S':
return ;
case 'T':
return ;
case 'U':
return ;
case 'V':
return ;
case 'W':
return ;
case 'X':
return ;
case 'Y':
return ;
case 'Z':
return ;
}
return -;
}
}
}
using System;
using System.Collections.Generic;
using System.Text;
using System.Data;
using System.Data.OleDb; namespace xtar_biz_codegen
{
class ExcelHelper
{
public static string POSTFIX_97 = "XLS"; public static string POSTFIX_03 = "XLSX";
}
}
最后附送一个项目文件,可以自己测试:
http://pan.baidu.com/s/1msFuY
一个用微软官方的OpenXml读写Excel 目前网上不太普及的方法。的更多相关文章
- #应用openxml读写excel代码
这个例子比较简单,没有考虑格式之类的问题. using DocumentFormat.OpenXml.Packaging; using DocumentFormat.OpenXml.Spreadshe ...
- Python: 读写Excel(openpyxl / win32com.client)
项目周报汇报的时候要做数据汇总,总是要从不同的excel文件中去获取数据最后汇总到一个excel表里面,所以决定用python直接写个自动化脚本来自动执行. 用python来读写excel,目前找了2 ...
- 微软官方的Excel android 移动版的折腾
微软官方的Excel android 移动版,有重大bug.害我折腾了一天多时间.最终确认是Excel自身的问题. 现象描述:手机上新建或是保存excel后.放到电脑上,不能打开.提示”Excel在B ...
- 微软官方 Win 11 “体检工具”太烂了?开发者自己做了一个
1.Win 10 免费升级到 Win 11 最近微软官方终于宣布了 Windows 11,不仅带来了全新的 UI,而且还有很多新功能:比如支持 Android 应用. 虽然微软官方已说明 Win 10 ...
- 【原创】.NET读写Excel工具Spire.Xls使用(1)入门介绍
在.NET平台,操作Excel文件是一个非常常用的需求,目前比较常规的方法有以下几种: 1.Office Com组件的方式:这个方式非常累人,微软的东西总是这么的复杂,使用起来可能非常不便,需要安装E ...
- 【原创】.NET读写Excel工具Spire.Xls使用(3)单元格控制
本博客所有文章分类的总目录:http://www.cnblogs.com/asxinyu/p/4288836.html .NET读写Excel工具Spire.Xls使用文章 ...
- .NET读写Excel工具Spire.Xls使用(1)入门介绍
原文:[原创].NET读写Excel工具Spire.Xls使用(1)入门介绍 在.NET平台,操作Excel文件是一个非常常用的需求,目前比较常规的方法有以下几种: 1.Office Com组件的方式 ...
- 使用NPOI读写Excel、Word
NPOI 是 POI 项目的 .NET 版本.POI是一个开源的Java读写Excel.WORD等微软OLE2组件文档的项目. 使用 NPOI 你就可以在没有安装 Office 或者相应环境的机器上对 ...
- 【原创】.NET读写Excel工具Spire.Xls使用(4)对数据操作与控制
本博客所有文章分类的总目录:http://www.cnblogs.com/asxinyu/p/4288836.html .NET读写Excel工具Spire.Xls使用文章 ...
随机推荐
- Groovy 处理 XML
1. Parsing XML 1.1. XmlParser and XmlSlurper The most commonly used approach for parsing XML with Gr ...
- select制作分层级目录,让select显示和可下拉选择的"不一样"
今天遇到一个特殊的select问题,需求是这样的:每次点击这个select时,根据选择的option的值做出相应的处理并返回新的select,option内容.所以大致思路是给这个select绑定ch ...
- pd name与comment互换,或者code互换,总之互换
1 PowerDesigner中批量根据对象的name生成comment的脚本 执行方法:Open PDM -- Tools -- Execute Commands -- Run Script Opt ...
- 面向对象程序设计 第二次作业<1>
Github链接:https://github.com/zora02/object-oriented/tree/master/1001.A%2BB%20Format%20(20) 一.解题 题目 解题 ...
- C# 微信v3退款
1.退款需要退款证书.C#都是用p12的证书,双击证书导入,密码是mch_id(商户号) 2.调用微信退款接口进行退款操作 string resp=string.Empty string cert = ...
- load与initialize
NSObject类有两种初始化方式load和initialize load + (void)load; 对于加入运行期系统的类及分类,必定会调用此方法,且仅调用一次. iOS会在应用程序启动的时候调用 ...
- 三大基础排序算法BubbleSort、SelectSort、InsertSort
public class Strategy { public static void main(String[] args) { int [] array=new int[]{26,25,15,42, ...
- 子进程 已安装 pre-removal 脚本 返回了错误号 1或2 与 子进程 已安装 post-installation 脚本 返回了错误号 1或2
今天在ubuntu kylin上安装了virtualbox, 后来我想删除了再装个新一点的,结果正常的情况下删除不了,我就把找到的virtualbox的目录全部都删除了, 再通过apt-get rem ...
- 【转】测试LibreOffice SDK 开发环境配置(Windows)
原文:http://www.aqcoder.com/blog/detail/id/7441186b-93fd-482c-b4d7-0facd1ee498d 下载与安装 LibreOffice 主页:h ...
- Linux网关配置(centos6)
1.找寻本地机器网关的位置,点击红圈位置 打开后样式 2.双击点开打开网络和共享中心,找到查看活动网络---->点击右边的本地连接 3.点击详细信息 4.查看如下 5.对照配置,进入linux命 ...