實現標記datagridview標題並導出Excel的功能

来源:http://www.cnblogs.com/sj951129/archive/2016/10/12/5952190.html
-Advertisement-
Play Games

實現了導入Excel文件,點擊列標題修改並標記後導出Excel的小功能。 ...


最近在學習winform,國慶前被佈置了一個小任務,好不容易大致做出來了,決定記錄下來,以此加深印象。

先說下需求:這是一個導入話單標記後並導出的功能

1. 選擇excel文件

2. 定義欄位
   日期 時間 對方號碼 通話時長 呼叫類型
3. 點擊datagridview 標題 出現下拉菜單 顯示定義的欄位
4. 標記定義欄位列
5. 保存定義欄位數據 到 datatable
6  導出datatable

按照需求一步一步來,先設計界面,需要一個DataGridView和兩個Button,一個導入,一個導出,我加了個Label和TextBox來提示文件路徑。

先在類裡面定義幾個全局變數,下麵的代碼中會用到。

1         int colIndex;//點擊的單元格列索引
2         int rowIndex;//點擊的單元格行索引      
3         Dictionary<int, string> dic = new Dictionary<int, string>();//存放excel標題
4         List<string> list = new List<string>();    //存放標記後的標題
5         DataTable dt;//導入的table
6         string filename = "";//Excel文件名
View Code

第一步:導入Excel預覽,我從網上找了一段excel導入datagridview的代碼,具體如下:

 1         private DataTable ExcelToDataTable(string path)
 2         {
 3 
 4             FileStream fs = File.OpenRead(path);   //打開.xls文件
 5 
 6             HSSFWorkbook wk = new HSSFWorkbook(fs);   //把xls文件中的數據寫入wk中
 7 
 8             var sheet = wk.GetSheetAt(0);   //提取第一個sheet
 9             var headerRow = sheet.GetRow(0);//提取sheet第一行
10             var cellCount = headerRow.LastCellNum;//提取行的最後一列
11             DataTable table = new DataTable();
12             //給table添加一個列
13             for (int i = headerRow.FirstCellNum; i < cellCount; i++)
14             {
15                 DataColumn col = new DataColumn(headerRow.GetCell(i).StringCellValue);
16                 table.Columns.Add(col);
17             }
18             //獲取sheet的行數
19             var rowCount = sheet.LastRowNum;
20             //迴圈逐行將sheet中數據寫入table
21             for (int i = (sheet.FirstRowNum + 1); i < rowCount; i++)
22             {
23                 var row = sheet.GetRow(i);
24                 DataRow datarow = table.NewRow();
25                 for (int j = row.FirstCellNum; j < cellCount; j++)
26                 {
27                     if (row.GetCell(j) != null)
28                     {
29                         datarow[j] = row.GetCell(j).ToString();
30                     }
31                 }
32                 table.Rows.Add(datarow);
33             }
34             wk = null;
35             sheet = null;
36             return table;
37         }
導入方法

這是個導入的方法,雙擊導入按鈕,在事件里添加如下代碼:

 1         private void btnImport_Click(object sender, EventArgs e)
 2         {
 3             OpenFileDialog ofd = new OpenFileDialog();
 4             ofd.InitialDirectory = Environment.GetFolderPath(Environment.SpecialFolder.Desktop);
 5             ofd.Filter = "Excel Files(*.xls)|*.xls";
 6             if (ofd.ShowDialog() != DialogResult.OK)
 7             { return; }
 8             filename = ofd.FileName;
 9             textBox1.Text = ofd.FileName;
10             dt = ExcelToDataTable(filename);
11             dataGridView1.DataSource = dt;//ExcelToDataTable(ofd.FileName);
12 
13             int ColCount = dataGridView1.Columns.Count;
14             //將Excel所有標題存入dic中
15             for (int i = 0; i < ColCount; i++)
16             {
17                 dic.Add(i, dataGridView1.Columns[i].HeaderText);
18             }
19         }
導入預覽

為了方便我將測試的Excel文檔放到桌面,Excel的標題我存到了字典里,下麵會用到。

到這裡基本實現了第一步,點擊導入按鈕,選中excel文件,顯示在datagridview上預覽。

 

第二步、第三步:要求的樣式大概是這樣:

點擊datagridview的標題,彈出下拉菜單,顯示定義的欄位。因此添加一個ContextMenuStrip控制項,將欄位一個個添加進去。

要求點擊標題彈出下拉菜單,我用到了datagridview的CellCilck事件,這個事件里代碼非常簡單,只要彈出contexmenustrip就行了,然後加一個對是否為標題行的判斷。

View Code

第四步比較複雜,大概思路是,下拉菜單的選項是固定的,通過contextMenuStrip1_ItemClicked事件里的e.ItemClicked.Text可以獲取到你選擇的菜單,因此我用了switch case語句來進行判斷。

首先是點擊datagridview一列的標題,然後選中一個菜單,這一列的標題要變成e.itemClicked,並且標記這一列,所以我寫了一個標記的方法表示標記後操作和樣式。

1         private void Mark(string item)
2         {
3             dt.Columns[colIndex].ColumnName = item;
4             dataGridView1.Columns[colIndex].HeaderText = item;
5             dataGridView1.Columns[colIndex].DefaultCellStyle.BackColor = Color.LightSteelBlue;
6             dataGridView1.EnableHeadersVisualStyles = false;
7             dataGridView1.Columns[colIndex].HeaderCell.Style.BackColor = Color.LightSlateGray;
8         }
Mark

因為涉及到匹配的問題,例如日期格式的列無法標記成“呼叫類型”,“通話時間”等,所以加了一些正則判斷,正則表達式以前從沒用過,也是網上現學的,寫得比較差。

又寫了一個判斷後的方法:

 1  private void MatchItem(string match, string str, string item)
 2         {
 3             Match m = Regex.Match(match, str);
 4             if (m.Success)
 5             {
 6                 Mark(item);
 7                 //如果該列已標記,則不添加item到list
 8                 foreach (var i in list)
 9                 {
10                     if (item == i)
11                         return;
12                 }
13                 list.Add(item);
14             }
15             else
16             {
17                 contextMenuStrip1.Hide();
18                 MessageBox.Show("該列不為" + item + "");
19             }
20         }
判斷格式

至於contextmenustrip_itemClicked事件里只要在每個case里調用MatchItem方法就可以了。

 1   private void contextMenuStrip1_ItemClicked(object sender, ToolStripItemClickedEventArgs e)
 2         {
 3             //dt = ExcelToDataTable(filename);
 4             //獲取想要標記的的標題
 5             string item = e.ClickedItem.Text;
 6             //獲取用於判斷的內容
 7             var BeMatch = dataGridView1.Rows[rowIndex + 1].Cells[colIndex].Value.ToString();
 8             //當前列在table是否已經存在
 9             int i = 0;
10             foreach (DataColumn dataCol in dt.Columns)
11             {
12                 if (i != colIndex && dataCol.ColumnName == item)
13                 {
14                     contextMenuStrip1.Hide();
15                     var index = i + 1;
16                     MessageBox.Show("該表第" + index + "列為" + dataCol.ColumnName + ",無法標記");
17                     return;
18                 }
19                 i++;
20             }
21 
22             switch (item)
23             {
24                 case "呼叫類型":
25                     {
26                         string strType = @"[\u4e00-\u9fbb]";
27                         MatchItem(BeMatch, strType, item);
28                     }
29                     break;
30                 case "對方號碼":
31                     {
32                         string strNum = @"0?[1]+[358]+\d{9}";
33                         MatchItem(BeMatch, strNum, item);
34                     }
35                     break;
36                 case "日期":
37                     {
38                         string strDate = @"^2\d{7}$";
39                         MatchItem(BeMatch, strDate, item);
40                     }
41                     break;
42                 case "時間":
43                     {
44                         string strTime = @"^[0-2]\d{1}[0-5]\d{1}[0-5]\d{1}$";
45                         MatchItem(BeMatch, strTime, item);
46                     }
47                     break;
48                 case "通話時長":
49                     {
50                         string strOften = @"\d";
51                         MatchItem(BeMatch, strOften, item);
52                     }
53                     break;
54                 case "取消設置":
55                     {
56                         //遍歷原標題,獲取取消設置的列,移出dt
57                         foreach (var kv in dic)
58                         {
59                             if (kv.Key == colIndex)
60                             {
61                                 list.Remove(dataGridView1.Columns[colIndex].HeaderText);
62                                 //list.Insert(colIndex,kv.Value);
63                                 item = kv.Value;
64                                 dt.Columns[colIndex].ColumnName = item;
65                             }
66                         }
67                         //還原標題和樣式                    
68                         dataGridView1.Columns[colIndex].DefaultCellStyle.BackColor = Color.White;
69                         dataGridView1.Columns[colIndex].HeaderCell.Style.BackColor = DefaultBackColor;
70                     }
71                     break;
72             }
73         }
標記列

第五步 :保存標記後的欄位數據到datatable里,我在網上找了一個方法,附上鏈接:http://www.jb51.net/article/80620.htm

 1    private void btnExport_Click(object sender, EventArgs e)
 2         {
 3             //添加標記的列到table
 4             DataTable table = dt.DefaultView.ToTable(false, list.ToArray());
 5 
 6             try
 7             {
 8                 SaveFileDialog sfd = new SaveFileDialog();
 9                 //sfd.FileName = "測試導出.xls";
10                 sfd.Filter = "Excel Files(*.xls)|*.xls";
11                 sfd.FileName = "測試導出";
12                 if (sfd.ShowDialog() == DialogResult.OK)
13                 {
14                     filename = sfd.FileName;
15                     DataToExcel(table, filename, "話單", false);
16                 }
17             }
18             catch (Exception ex)
19             {
20                 MessageBox.Show(ex.Message);
21             }
導出標記後的列

做完運行的效果是這樣的:

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 問題一: 第一次mysql啟動服務失敗,未返回報錯信息 解決方法: 執行mysqld -console命令,查看error信息,對症下藥 但一般情況下,主要是因為mysql目錄下的data文件夾中內容不正確,解決方法有以下兩個 1.在開啟服務前執行初始化命令 mysqld --initalize 然 ...
  • ...
  • Spark存儲管理(讀書筆記) 轉載請註明出處: "http://www.cnblogs.com/BYRans/" Spark的存儲管理 RDD的存放和管理都是由Spark的存儲管理模塊實現和管理的。本文從架構和功能兩個角度對Spark的存儲管理模塊進行介紹。 架構角度 從架構角度,存儲管理模塊主要 ...
  • Mongodb:修改文檔結構後出現錯誤:Element '***' does not match any field or property of class ***. Mongodb是一種面向文檔的資料庫,即不再有“行”的概念,取而代之的是更為靈活的“文檔”(doucument)模型。在開發中,經 ...
  • 近期又做了個問卷調查,問卷調查一次性要保存一二十個題目和答案!所以嘞,博主為了偷懶,就直接把答卷內容保存成了Json格式! 好處當然是很多啦! 只需一個欄位就能保存整個答卷的內容! 想想都刺激!哈哈~! 好了,閑話少敘,直入正題! 首先,一般不會保存題目,要求的可能是只保存答案!比如第幾題對應的調查 ...
  • 採用web.config配置母版,方便母版的變更,處理方法: 1、在web.config配置如下內容: <configuration> <system.web> <pages masterPageFile="~/Stie.Master(此處可修改)"/> </system.web> </config ...
  • 錯誤內容: 界面顯示內容為: ...
  • 設置忽略xml序列化標簽 ((Newtonsoft.Json.Serialization.DefaultContractResolver)config.Formatters.JsonFormatter.SerializerSettings.ContractResolver).IgnoreSerial ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...