# [C# 開發技巧系列]使用C#操作Word和Excel程序
**一、引言**
在我們日常辦公中,我們經常可能遇到一些重復性的工作的,比如,我們在寫畢業設計的時候,有時候我們寫的過程中不注意,當整篇畢業論文寫完之后,發現在畢業論文中存在很多空白的段落,這是我們就需要人工重新審閱一遍論文,再手動刪除一些空白行,由于畢業論文也不是一篇,有開題報告啊,文獻翻譯等等,這樣就可能需要我們人工都去審閱一篇把一些空白行刪除,這樣既花時間,我們也看的累。然后還有一個例子就是——我們人事部門的MM們,一到月末的時候就需要給本月的壽星員工發送郵件來通知參加生日會,如果員工信息是在Excel中的話,這時候人事的MM就要手動地從中查找本月壽星的郵箱,然后用Outlook一個一個添加郵件地址來給本月壽星發送郵件的,為了讓人事MM們不再那么累,所以就想能不能用程序自動化地完成這一系列的過程呢?答案是肯定的,下面就讓我來實現上面的兩個需求的,使我們(尤其是人事MM們)的辦公更加Easy。
**二、自動刪除Word中的空白行和頁**
在引言部分,我們已經提出了這個需求的。記得當時在寫畢業論文的時候,我也做過這些重復的事情,經常寫完之后會再去審閱一遍畢業論文中的所有文檔,然后手動把一些空白行刪除掉,由于當時并不知道可以對Word來進行自動化編程,所以只能傻傻地做這樣一些重復的事情。但是現在不一樣了,自從接觸了VSTO之后,才知道Office一系列產品都是提供了一些公開的API的,我們可以利用這些對象使我們自定義Office程序和使Office自動化地工作,下面就具體講講如何實現這個小的工具的。
首先,我們先明確下這個工具需要實現的功能——自動移除Word文檔中的空白行。然后向大家解釋下實現該工具的思路:
* 我們打開一個Word文檔,該Word文檔就是一個**Word.Document**對象
* Word文檔中內容都是段落組成的,然而段落在Word對象模型中是**Word.Paragraph**對象
* **空白行或空白頁也就是段落的內容為空**,明白了這點,我們就可以在程序中對段落對象的文本進行判斷,如果段落內容為空,我們就刪除該段落,這樣也就實現了移除空白行的功能了。
有了上面的思路之后,然后大家只需要了解Word中對象模型,然后通過對象模型找到段落對象,然后再判斷它的文本是否為空,為空就刪除段落,不為空就什么都不做。所以思路有了之后,就是要了解Word對象模型了,對于這部分內容,大家可以參考我博客的中的——[創建Word解決方案](http://www.cnblogs.com/zhili/archive/2013/03/09/VSTO_Word.html)。由于代碼中都有注釋,這里就直接看實現該工具的核心代碼:
```
string[] wordPatharray = null;
// 打開需要操作的Word文檔
private void btnOpen_Click(object sender, EventArgs e)
{
using (OpenFileDialog openFileDialog = new OpenFileDialog())
{
openFileDialog.Filter = "Word document(*.doc;*.docx)|*.dox;**.docx|All Files(*.*)|*.*";
// 設置允許選擇多個文件,該屬性默認為false的,即只允許選擇一個文件
openFileDialog.Multiselect = true;
if (openFileDialog.ShowDialog() == DialogResult.OK)
{
txtWordPath.Text = openFileDialog.FileName;
// 獲得所有選定文件的文件名
wordPatharray = openFileDialog.FileNames;
}
}
}
// 移除Word中的所有空頁
private void btnRemove_Click(object sender, EventArgs e)
{
Word.Application wordapp = null;
Word.Document doc = null;
try
{
// 啟動Word應用程序并設置不可見
wordapp = new Word.Application();
// 如果不設置該屬性,就可以看到Word程序的啟動過程,這個和我們手動啟動Word是一樣的
wordapp.Visible = false;
// 遍歷每個文件名
foreach (var wordpath in wordPatharray)
{
doc = wordapp.Documents.Open(wordpath);
// 刪除所有空白頁面
Word.Paragraph paragraph;
Word.Paragraphs paragraphs = doc.Paragraphs;
for (int i = paragraphs.Count; i > 0; i--)
{
paragraph = paragraphs[i];
// 如果段落的文本為空的話,首先選擇該段落,然后再調用Word中Selection對象的Delete方法來刪除
// 不為空什么都不做
if (paragraph.Range.Text.Trim() == string.Empty)
{
paragraph.Range.Select();
wordapp.Selection.Delete();
}
}
if (doc != null)
{
// 先保存所有修改再關閉Word文檔
doc.Save();
((Word._Document)doc).Close();
}
}
MessageBox.Show("刪除空白行成功");
}
catch (Exception ex)
{
MessageBox.Show("異常發生,異常信息為:" + ex.Message);
}
finally
{
// 釋放資源
// 退出Word程序
if (wordapp != null)
{
((Word._Application)wordapp).Quit();
}
doc = null;
wordapp = null;
}
}
```
為了測試該程序的正確性,這里我建立了兩個測試文檔,為了測試,我故意在文檔中刪除了空白行和空白頁面,下面是兩個測試文檔的截圖:

下面就看看該工具的運行效果(效果圖是一段動畫,認為這樣可以更加說明運行效果)
:
**三、人事部門的福音——自動給本月壽星員工發送郵件提醒**
為了幫助大家更好地理解該程序,還是像之前一樣,首先說說實現該程序的思路:
* 我們首先需要打開員工信息表,此時我們可以利用Excel對象模型中的**Excel.Application.Workbooks.Open**方法來獲得一個工作簿對象,關于更多Excel對象模型的內容可以轉向——[創建Excel解決方案](http://www.cnblogs.com/zhili/archive/2013/02/24/ExcelAddin.html)。
* 通過第一步我們已經獲得了工作簿對象了,然后通過遍歷工作簿中的激活表,即表格一(Sheet1),我們可以通過**workbook.ActiveSheet**來獲得表格一對象。
* 遍歷表格一中的所有行來找到生日信息中的月份,如果月份等于當前月份,就給該員工的郵箱進行發郵件。
* 對于自動發送郵件的實現,該實現和我們手動操作Outlook過程是一樣,手動操作時,我們需要手動打開Outlook(**在程序中就是創建Outlook應用程序對象**),然后點擊新建郵件(在程序中就是通過**Applicatin**對象的**CreateItem(Outlook.OlItemType.olMailItem)**方法來創建一個郵件項目),在新建郵件窗口中指定收件人,主題,郵件內容之后,點擊Outlook中的發送郵件按鈕(在程序中就是通過指定 **Outlook.MailItem對象**(即代表一個郵件窗體)的**To(收件人)、Subject(主題)、Body(郵件內容**)屬性,然后再調用Send方法來發送郵件)
明白了思路之后,我們理解代碼會更加容易了,具體實現代碼為:
```
using System;
using System.IO;
using System.Runtime.InteropServices;
using System.Windows.Forms;
// 引用Excel和Outlook的命名空間
using Excel = Microsoft.Office.Interop.Excel;
using Outlook = Microsoft.Office.Interop.Outlook;
string excelpath = string.Empty;
// 打開員工表格
private void btnOpen_Click(object sender, EventArgs e)
{
using (OpenFileDialog openFileDialog = new OpenFileDialog())
{
openFileDialog.Filter = "Excel File(*.xls;*.xlsx)|*.xls;**.xlsx|All Files(*.*)|*.*";
if (openFileDialog.ShowDialog() == DialogResult.OK)
{
txtExcelPath.Text = openFileDialog.FileName;
excelpath = openFileDialog.FileName;
}
}
}
// 自動給本月壽星發郵件通知
private void btnSendEmail_Click(object sender, EventArgs e)
{
if (!File.Exists(txtExcelPath.Text))
{
MessageBox.Show("員工表路徑不存在,請確保輸入正確的文件路徑");
return;
}
if (txbBirthday.Text.Trim() == string.Empty || txbEmail.Text.Trim() == string.Empty)
{
MessageBox.Show("請先輸入員工表中生日信息所在的列和郵箱信息所在的列!");
return;
}
// 輸入信息都正確時開始發送郵件
SendEmail(int.Parse(txbBirthday.Text.Trim()), int.Parse(txbEmail.Text.Trim()));
}
// 發送郵件方法
private void SendEmail(int birthDayColumn,int emailColumn)
{
// 獲得當前月份
int nowmonth = DateTime.Now.Month;
// 發送郵件地址字符串
string toEmailString = string.Empty;
string emailBody="請收到郵件的員工,請本月28號到休閑室來參加生日Party";
Excel.Application excelApp = null;
Excel.Workbook workbook =null;
Excel.Worksheet worksheet = null;
Excel.Range range = null;
try
{
// 新建Excel應用程序被設置它不可見
excelApp = new Excel.Application();
excelApp.Visible = false;
workbook = excelApp.Workbooks.Open(excelpath);
// 獲得打開文件的激活表格
worksheet= workbook.ActiveSheet;
// 遍歷表格中的所有行
for (int row = 2; row < worksheet.UsedRange.Rows.Count + 1; row++)
{
// 因為我的測試表格中第四列是生日信息,在Excel中第一行的下標是從1開始的
// 這里本來需要在頁面設置一個文本框讓用戶填寫生日信息是在那一列的
// 這里為了測試就直接在程序中指定
// 下面的Range就代表生日列中每一個單元格
range = worksheet.Cells[row, birthDayColumn];
// 我們可以通過Range.Value來獲得單元格中的生日信息
// 因為我生日單元格中為日期格式,所以獲得的是日期類型,所以直接通過Month屬性來獲得月份
int month = range.Value.Month;
// 如果我們的Excel文檔中生日時間設置為文本格式的話,這時候就需要通過分割字符串的方式來獲得月份
// 通過Split函數來把生日信息以'/'符號分隔,分隔的數組的第二個就是月份
//int month = Int32.Parse(birthday.Split('/')[1]);
// 如果月份等于當前月的話,就給這個人發郵件
if (month == nowmonth)
{
// 獲得本月生日員工的郵件地址
toEmailString += ";" + ((Excel.Range)worksheet.Cells[row, emailColumn]).Value;
}
}
}
catch (Exception ex)
{
MessageBox.Show("讀取員工表格時出錯,異常信息為:" + ex.Message);
return;
}
finally
{
workbook.Close(Excel.XlSaveAction.xlDoNotSaveChanges);
excelApp.Quit();
if (workbook != null)
{
Marshal.FinalReleaseComObject(workbook);
workbook = null;
}
if (excelApp != null)
{
Marshal.FinalReleaseComObject(excelApp);
excelApp = null;
}
}
if (CreateEmailItem("生日提醒", toEmailString, emailBody))
{
MessageBox.Show("成功給本月壽星發送郵件提醒");
}
}
// 創建郵件項
private bool CreateEmailItem(string subjectEmail,string toEmail,string bodyEmail)
{
Outlook.Application outlookapp = null;
Outlook.MailItem email =null;
try
{
// 創建郵件項,就如你手動點新建郵件一樣
outlookapp = new Outlook.Application();
email = outlookapp.CreateItem(Outlook.OlItemType.olMailItem);
// 指定郵件的主題,收件人和內容,就如你在新建郵件窗體中輸入收件人,主題和內容一樣
email.Subject = subjectEmail;
email.To = toEmail;
email.Body = bodyEmail;
email.Importance = Outlook.OlImportance.olImportanceHigh;
// 發送郵件,就如你點界面上的發送郵件操作一樣
((Outlook._MailItem)email).Send();
}
catch(Exception ex)
{
MessageBox.Show("發送郵件的時候失敗,異常信息為:" + ex.Message);
return false;
}
finally
{
// 釋放資源
((Outlook._Application)outlookapp).Quit();
if (email != null)
{
Marshal.FinalReleaseComObject(email);
email = null;
}
if (outlookapp != null)
{
Marshal.FinalReleaseComObject(outlookapp);
outlookapp = null;
}
}
return true;
}
```
為了測試程序,我新建了一個員工信息表,表格的格式如下(你當然可以根據自己的需要更改格式):

現在就讓我們看看該程序的運行效果:

**四、 小結**
到這里,本專題的內容就和大家介紹完了,在下一個專題中將向大家介紹下如何通過Office提供的API的來遙控幻燈片。如果大家對本專題中兩個工具的實現源碼有任何的疑問,都可以在下面留言給我。覺得不錯的話,幫忙推薦下,感謝大家的支持
- C# 基礎知識系列
- C# 基礎知識系列 專題一:深入解析委托——C#中為什么要引入委托
- C# 基礎知識系列 專題二:委托的本質論
- C# 基礎知識系列 專題三:如何用委托包裝多個方法——委托鏈
- C# 基礎知識系列 專題四:事件揭秘
- C# 基礎知識系列 專題五:當點擊按鈕時觸發Click事件背后發生的事情
- C# 基礎知識系列 專題六:泛型基礎篇——為什么引入泛型
- C# 基礎知識系列 專題七: 泛型深入理解(一)
- C# 基礎知識系列 專題八: 深入理解泛型(二)
- C# 基礎知識系列 專題九: 深入理解泛型可變性
- C#基礎知識系列 專題十:全面解析可空類型
- C# 基礎知識系列 專題十一:匿名方法解析
- C#基礎知識系列 專題十二:迭代器
- C#基礎知識 專題十三:全面解析對象集合初始化器、匿名類型和隱式類型
- C# 基礎知識系列 專題十四:深入理解Lambda表達式
- C# 基礎知識系列 專題十五:全面解析擴展方法
- C# 基礎知識系列 專題十六:Linq介紹
- C#基礎知識系列 專題十七:深入理解動態類型
- 你必須知道的異步編程 C# 5.0 新特性——Async和Await使異步編程更簡單
- 全面解析C#中參數傳遞
- C#基礎知識系列 全面解析C#中靜態與非靜態
- C# 基礎知識系列 C#中易混淆的知識點
- C#進階系列
- C#進階系列 專題一:深入解析深拷貝和淺拷貝
- C#進階系列 專題二:你知道Dictionary查找速度為什么快嗎?
- C# 開發技巧系列
- C# 開發技巧系列 使用C#操作Word和Excel程序
- C# 開發技巧系列 使用C#操作幻燈片
- C# 開發技巧系列 如何動態設置屏幕分辨率
- C# 開發技巧系列 C#如何實現圖片查看器
- C# 開發技巧 如何防止程序多次運行
- C# 開發技巧 實現屬于自己的截圖工具
- C# 開發技巧 如何使不符合要求的元素等于離它最近的一個元素
- C# 線程處理系列
- C# 線程處理系列 專題一:線程基礎
- C# 線程處理系列 專題二:線程池中的工作者線程
- C# 線程處理系列 專題三:線程池中的I/O線程
- C# 線程處理系列 專題四:線程同步
- C# 線程處理系列 專題五:線程同步——事件構造
- C# 線程處理系列 專題六:線程同步——信號量和互斥體
- C# 多線程處理系列專題七——對多線程的補充
- C#網絡編程系列
- C# 網絡編程系列 專題一:網絡協議簡介
- C# 網絡編程系列 專題二:HTTP協議詳解
- C# 網絡編程系列 專題三:自定義Web服務器
- C# 網絡編程系列 專題四:自定義Web瀏覽器
- C# 網絡編程系列 專題五:TCP編程
- C# 網絡編程系列 專題六:UDP編程
- C# 網絡編程系列 專題七:UDP編程補充——UDP廣播程序的實現
- C# 網絡編程系列 專題八:P2P編程
- C# 網絡編程系列 專題九:實現類似QQ的即時通信程序
- C# 網絡編程系列 專題十:實現簡單的郵件收發器
- C# 網絡編程系列 專題十一:實現一個基于FTP協議的程序——文件上傳下載器
- C# 網絡編程系列 專題十二:實現一個簡單的FTP服務器
- C# 互操作性入門系列
- C# 互操作性入門系列(一):C#中互操作性介紹
- C# 互操作性入門系列(二):使用平臺調用調用Win32 函數
- C# 互操作性入門系列(三):平臺調用中的數據封送處理
- C# 互操作性入門系列(四):在C# 中調用COM組件
- CLR
- 談談: String 和StringBuilder區別和選擇
- 談談:程序集加載和反射
- 利用反射獲得委托和事件以及創建委托實例和添加事件處理程序
- 談談:.Net中的序列化和反序列化
- C#設計模式
- UML類圖符號 各種關系說明以及舉例
- C#設計模式(1)——單例模式
- C#設計模式(2)——簡單工廠模式
- C#設計模式(3)——工廠方法模式
- C#設計模式(4)——抽象工廠模式
- C#設計模式(5)——建造者模式(Builder Pattern)
- C#設計模式(6)——原型模式(Prototype Pattern)
- C#設計模式(7)——適配器模式(Adapter Pattern)
- C#設計模式(8)——橋接模式(Bridge Pattern)
- C#設計模式(9)——裝飾者模式(Decorator Pattern)
- C#設計模式(10)——組合模式(Composite Pattern)
- C#設計模式(11)——外觀模式(Facade Pattern)
- C#設計模式(12)——享元模式(Flyweight Pattern)
- C#設計模式(13)——代理模式(Proxy Pattern)
- C#設計模式(14)——模板方法模式(Template Method)
- C#設計模式(15)——命令模式(Command Pattern)
- C#設計模式(16)——迭代器模式(Iterator Pattern)
- C#設計模式(17)——觀察者模式(Observer Pattern)
- C#設計模式(18)——中介者模式(Mediator Pattern)
- C#設計模式(19)——狀態者模式(State Pattern)
- C#設計模式(20)——策略者模式(Stragety Pattern)
- C#設計模式(21)——責任鏈模式
- C#設計模式(22)——訪問者模式(Vistor Pattern)
- C#設計模式(23)——備忘錄模式(Memento Pattern)
- C#設計模式總結
- WPF快速入門系列
- WPF快速入門系列(1)——WPF布局概覽
- WPF快速入門系列(2)——深入解析依賴屬性
- WPF快速入門系列(3)——深入解析WPF事件機制
- WPF快速入門系列(4)——深入解析WPF綁定
- WPF快速入門系列(5)——深入解析WPF命令
- WPF快速入門系列(6)——WPF資源和樣式
- WPF快速入門系列(7)——深入解析WPF模板
- WPF快速入門系列(8)——MVVM快速入門
- WPF快速入門系列(9)——WPF任務管理工具實現
- ASP.NET 開發
- ASP.NET 開發必備知識點(1):如何讓Asp.net網站運行在自定義的Web服務器上
- ASP.NET 開發必備知識點(2):那些年追過的ASP.NET權限管理
- ASP.NET中實現回調
- 跟我一起學WCF
- 跟我一起學WCF(1)——MSMQ消息隊列
- 跟我一起學WCF(2)——利用.NET Remoting技術開發分布式應用
- 跟我一起學WCF(3)——利用Web Services開發分布式應用
- 跟我一起學WCF(3)——利用Web Services開發分布式應用
- 跟我一起學WCF(4)——第一個WCF程序
- 跟我一起學WCF(5)——深入解析服務契約 上篇
- 跟我一起學WCF(6)——深入解析服務契約 下篇
- 跟我一起學WCF(7)——WCF數據契約與序列化詳解
- 跟我一起學WCF(8)——WCF中Session、實例管理詳解
- 跟我一起學WCF(9)——WCF回調操作的實現
- 跟我一起學WCF(10)——WCF中事務處理
- 跟我一起學WCF(11)——WCF中隊列服務詳解
- 跟我一起學WCF(12)——WCF中Rest服務入門
- 跟我一起學WCF(13)——WCF系列總結
- .NET領域驅動設計實戰系列
- .NET領域驅動設計實戰系列 專題一:前期準備之EF CodeFirst
- .NET領域驅動設計實戰系列 專題二:結合領域驅動設計的面向服務架構來搭建網上書店
- .NET領域驅動設計實戰系列 專題三:前期準備之規約模式(Specification Pattern)
- .NET領域驅動設計實戰系列 專題四:前期準備之工作單元模式(Unit Of Work)
- .NET領域驅動設計實戰系列 專題五:網上書店規約模式、工作單元模式的引入以及購物車的實現
- .NET領域驅動設計實戰系列 專題六:DDD實踐案例:網上書店訂單功能的實現
- .NET領域驅動設計實戰系列 專題七:DDD實踐案例:引入事件驅動與中間件機制來實現后臺管理功能
- .NET領域驅動設計實戰系列 專題八:DDD案例:網上書店分布式消息隊列和分布式緩存的實現
- .NET領域驅動設計實戰系列 專題九:DDD案例:網上書店AOP和站點地圖的實現
- .NET領域驅動設計實戰系列 專題十:DDD擴展內容:全面剖析CQRS模式實現
- .NET領域驅動設計實戰系列 專題十一:.NET 領域驅動設計實戰系列總結