在最近的一个项目中,有大量的数据源来至Excel,转成JSON供前台使用。Excel数据是人工录入的,难免会有错误,所以中间会有逻辑检查。在C#中读取Excel的方式有很多,网上一搜一大堆,这里我也贴出一个ExcelHelper,提供根据Excel文件获取所有Sheet名称和获取Sheet内容两个方法。使用的时候记得注册AccessDatabaseEngine,该驱动有64位、32位两个版本,请根据自己的环境选择正确的版本。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using System.Data.OleDb;
  5. using System.Linq;
  6.  
  7. namespace Math.Library.Helper
  8. {
  9. public class ExcelHelper
  10. {
  11. public static DataTable GetExcelContent(String filePath, string sheetName)
  12. {
  13. if (sheetName == "_xlnm#_FilterDatabase")
  14. return null;
  15. DataSet dateSet = new DataSet();
  16. String connectionString = String.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0};Extended Properties='Excel 12.0;HDR=NO;IMEX=1;'", filePath);
  17. String commandString = string.Format("SELECT * FROM [{0}$]", sheetName);
  18. using (OleDbConnection connection = new OleDbConnection(connectionString))
  19. {
  20. connection.Open();
  21. using (OleDbCommand command = new OleDbCommand(commandString, connection))
  22. {
  23. OleDbCommand objCmd = new OleDbCommand(commandString, connection);
  24. OleDbDataAdapter myData = new OleDbDataAdapter(commandString, connection);
  25. myData.Fill(dateSet, sheetName);
  26. DataTable table = dateSet.Tables[sheetName];
  27.  
  28. for (int i = ; i < table.Rows[].ItemArray.Length; i++)
  29. {
  30. var cloumnName = table.Rows[].ItemArray[i].ToString();
  31. if (!string.IsNullOrEmpty(cloumnName))
  32. table.Columns[i].ColumnName = cloumnName;
  33. }
  34. table.Rows.RemoveAt();
  35. return table;
  36. }
  37. }
  38. }
  39.  
  40. public static List<string> GetExcelSheetNames(string filePath)
  41. {
  42. OleDbConnection connection = null;
  43. System.Data.DataTable dt = null;
  44. try
  45. {
  46. String connectionString = String.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0};Extended Properties='Excel 12.0;HDR=YES;IMEX=1;'", filePath);
  47. connection = new OleDbConnection(connectionString);
  48. connection.Open();
  49. dt = connection.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, null);
  50.  
  51. if (dt == null)
  52. {
  53. return new List<string>();
  54. }
  55.  
  56. String[] excelSheets = new String[dt.Rows.Count];
  57. int i = ;
  58. foreach (DataRow row in dt.Rows)
  59. {
  60. excelSheets[i] = row["TABLE_NAME"].ToString().Split('$')[];
  61. i++;
  62. }
  63. return excelSheets.Distinct().ToList();
  64. }
  65. catch (Exception ex)
  66. {
  67. LogHelper.Logger.Error(ex);
  68. return new List<string>();
  69. }
  70. finally
  71. {
  72. if (connection != null)
  73. {
  74. connection.Close();
  75. connection.Dispose();
  76. }
  77. if (dt != null)
  78. {
  79. dt.Dispose();
  80. }
  81. }
  82. }
  83.  
  84. }
  85. }

假如我的Excel数据如下,有班级和学生两张表(实际上有上百个Sheet,这里只做演示之用)。

当然我们不可能为每一个Sheet都对应做一个数据Model,不仅繁琐,还很容易出错。这时我想到了Newtonsoft.Json,Newtonsoft.Json是.NET下开源的JSON格式序列化和反序列化的类库。其中Newtonsoft.Json.Linq提供了对LINQ支持,支持动态对象、数组的序列化。

将整个Excel转换为一个JSON文件,每一个Sheet Name作为Key,Content就是Value,Value以数组形式存在,最终得到数据格式如下:

对应的代码如下,JArray和JObject是Newtonsoft.Json中的对象,支持动态属性和方法,表名和列名就是这样插入JSON中的。

  1. class Program
  2. {
  3. static void ExcelToJson()
  4. {
  5. List<string> tableNames = ExcelHelper.GetExcelSheetNames("test.xlsx");
  6. var json = new JObject();
  7. tableNames.ForEach(tableName =>
  8. {
  9. var table = new JArray() as dynamic;
  10. DataTable dataTable = ExcelHelper.GetExcelContent("test.xlsx", tableName);
  11. foreach (DataRow dataRow in dataTable.Rows)
  12. {
  13. dynamic row = new JObject();
  14. foreach (DataColumn column in dataTable.Columns)
  15. {
  16. row.Add(column.ColumnName, dataRow[column.ColumnName].ToString());
  17. }
  18. table.Add(row);
  19. }
  20. json.Add(tableName, table);
  21. });
  22. Console.WriteLine(json.ToString());
  23. Console.WriteLine(json.ToString(Formatting.None));
  24. }
  25.  
  26. static void Main(string[] args)
  27. {
  28. ExcelToJson();
  29. }
  30. }

当然,为了减少前后端传输数据的流量,可以使用ToString(Formatting.None),这样生成出来的数据就没有格式了。

Excel动态生成JSON的更多相关文章

  1. C# 如何在Excel 动态生成PivotTable

    Excel 中的透视表对于数据分析来说,非常的方便,而且很多业务人员对于Excel的操作也是非常熟悉的,因此用Excel作为分析数据的界面,不失为一种很好的选择.那么如何用C#从数据库中抓取数据,并在 ...

  2. JS 动态生成JSON对象

    JS 动态生成JSON对象,一般用到JSON传递参数的时候,会用到. function onGeneratedRow(columnsResult) { var jsonData = {}; colum ...

  3. node读取excel文件生成JSON

    当前的目录结构 excel的数据如下: node识别excel,先得安装  node-xlsx,用npm或yarn都可以 npm install  node-xlsx 或 yarn add node- ...

  4. js动态生成JSON树

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  5. C 简单处理excel 转成 json

    引言 工作中常需要处理excel转json问题. 希望这篇博文能简单描述这个问题.并提供一种解决思路.提升感悟. 今天我们处理的事就是为了把 xlsm => json. 一种方式是. 去 goo ...

  6. 分享我基于NPOI+ExcelReport实现的导入与导出EXCEL类库:ExcelUtility (续3篇-导出时动态生成多Sheet EXCEL)

    ExcelUtility 类库经过我(梦在旅途)近期不断的优化与新增功能,现已基本趋向稳定,功能上也基本可以满足绝大部份的EXCEL导出需求,该类库已在我们公司大型ERP系统全面使用,效果不错,今天应 ...

  7. java动态生成带下拉框的Excel导入模板

    在实际开发中,由于业务需要,常常需要进行Excel导入导出操作.以前做一些简单的导入时,先准备一个模板,再进行导入,单有十几. 二十几个导入模板时,往往要做十几.二十几个模板.而且,当在模板中需要有下 ...

  8. Jquery解析Json字符串,并且动态生成数据表格Table

    //ajax获得后台传来的json字符串 $.post("UserInfo.ashx", function (data) { //假设data="{T1:[{User_I ...

  9. ArcGIS Server 10.2 实战(一)Asp.net MVC与JSON数据妙用实现动态生成要素图层

    今年7月刚刚发布的ArcGIS 10.2为GIS的web开发带来了一个很实在的功能,JSON转要素.以往GIS图层外部数据(如文本数据,数据库数据)动态地写入地图服务中的图层是一件不可想象的事情,如今 ...

随机推荐

  1. *windows文件显示后缀名

  2. PCL—综述—三维图像处理

    点云模型与三维信息 三维图像是一种特殊的信息表达形式,其特征是表达的空间中三个维度的数据.和二维图像相比,三维图像借助第三个维度的信息,可以实现天然的物体-背景解耦.除此之外,对于视觉测量来说,物体的 ...

  3. Django admin的一些有用定制

    Model实例,myapp/models.py: from django.db import models class Blog(models.Model): name = models.CharFi ...

  4. (贪心5.2.1)UVA 10026 Shoemaker's Problem(利用数据有序化来进行贪心选择)

    /* * UVA_10026.cpp * * Created on: 2013年10月10日 * Author: Administrator */ #include <iostream> ...

  5. IOS ARC与非ARC混合编译

    要开启ARC的:-fobjc-arc不开启ARC的:-fno-objc-arc 是否使用arc: 在build setting里找automatic reference counting,YES/NO

  6. UVA 11916 Emoogle Grid(同余模)

    题目链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  7. chrome控制台小技巧

    对于大多数开发人员来说,chrome控制台最常用的命令就是 console.log()了,然后还有一些其他类似的命令,如: console.info()   提示信息 console.error() ...

  8. CodePage代码,MultiByteToWideChar

    Identifier .NET Name Additional information 37 IBM037 IBM EBCDIC US-Canada 437 IBM437 OEM United Sta ...

  9. LeetCode Number of Islands 岛的数量(DFS,BFS)

    题意:0代表水,1代表陆地,那么被水围起来的就是岛了,给一个01矩阵,问有多少个岛? 思路:DFS还是比较短,实现了一下.如果一个点已经被遍历过了,那就将其置为0就行了,不要去搜0的. class S ...

  10. Java [Leetcode 278]First Bad Version

    题目描述: You are a product manager and currently leading a team to develop a new product. Unfortunately ...