http://blog.csdn.net/callmeback/archive/2009/02/18/3906310.aspx
操作系統是Microsoft XP,辦公套裝是Microsoft Office 2003,編程環境是Microsoft Visual Studio 6.0,一切都是Microsoft。
我最近要將數據庫中的內容查詢出來放到excel表格以便打印,所以上網找了這方面的內容,這里主要是拋磚引玉。
從思路上來看,操作excel表格就是將其打開,然后寫入/讀出數據,然后關閉。
首先創建一個程序(我的例子是一個MFC的單文檔程序),在程序的入口處和出口處先作這樣兩個步驟來支持COM庫:
在程序入口處CXXXApp:: InitInstance()函數AfxEnableControlContainer();語句之后加入下面幾行:
if (CoInitialize(NULL) != 0)
{
AfxMessageBox(“初始化COM支持庫失敗!”);
exit(1);
}
假如這個條件不通過就不能運行起程序。
在程序的出口處CXXXApp:: ExitInstance()函數return語句之前加入下面這句話:CoUninitialize();來釋放COM支持庫。
這樣對COM支持庫的代碼已經完成。
下面要從Office的安裝目錄中找到對VC操作excel文件的動態庫,在某些版本下這個文件是Excel8.olb或者Excel9.olb,在我的版本中是excel.exe這個exe也是動態庫的形式,是微軟公司主要的文件結果之一。選擇vc的View菜單里面的ClassWizad命令,會彈出一個對話框;然后點擊Add Class…按鈕選擇From a type library,會彈出一個打開對話框,從這里打開Office安裝目錄下…\Office11\EXCEL.EXE文件,從里面選擇幾個要用到的類:_Application, Workbooks, _Wrokbook, Worksheets, _WorkSheet, Range,點擊OK按鈕。會在程序中生成一個excel.h和excel.cpp文件,這些文件中包含了剛才我們選擇的幾個類的代碼。下面介紹一下這幾個類:
在vc操縱excel的exe動態庫里面有好多個對象模型,就是剛才在創建過程中看到的那個列表,但是經常用到的有這么幾個:_Application, Workbooks, _Wrokbook, Worksheets, _WorkSheet, Range,Charts和_Chart,最后面的兩個是用來操作圖表的,我沒有用到所以這里也就不記錄了。
_Application:這里的Application就是Excel本身,眾所周知,一個Excel可以包含多個工作簿,每個工作簿又可以包含多個工作表,而每個工作表又可以包含多個區域或者圖表,所以這里他們是樹型的結構關系,而application最基本的一個功能就是找到它的子項工作簿。果然,我們在引入我們程序的Application類中看到了這樣的成員函數:GetWorkbooks()。既然application就是excel,那么打開程序,退出程序,顯示/隱藏程序這些基本的操作都可以在這個類的成員函數中找到,果不其然。
Workbooks:這個對象是一個容器對象,它里面存放著所有打開的工作簿。因此,我們可以猜測它一定有添加,查找,打開/關閉工作簿的功能。(本程序中使用excel的一個xlt模板來生成一個xls文件就是使用了這個容器對象的添加功能。)
_Workbook:這是一個工作簿,也就相當于一個xls文件。Excel可以同時打開多個工作簿,所以工作簿之間必定能夠互相切換,每個工作簿可以關聯工作表容器并獲得工作表的索引。
Worksheets:也是一個容器對象,和Workbooks類似。
_Worksheet:這個就是我們看到的工作表,比如Sheet1,sheet2等等。
Rang:就是我們看到的能選中的方框的大小。而我們所要作的操作基本上是以區域為單位進行的。
介紹完這些,就添加一個菜單,來響應操作excel的命令。
然后下面附帶這個函數的內容,注釋還算可以吧,并且附上網上不知道誰寫的但是轉載極多的一個封裝類。
view plaincopy to clipboardprint?
_Application _app;
_Workbook _workBook;
_Worksheet _workSheet;
Worksheets workSheets;
Workbooks workBooks;
Range range;
Range copyFrom;
Range copyTo;
if(!_app.CreateDispatch("Excel.Application", NULL))
{
MessageBox("創建Excel服務失敗!", "信息提示", MB_OK);
return;
}
//利用模板建立新文檔
workBooks.AttachDispatch(_app.GetWorkbooks());
_workBook.AttachDispatch(workBooks.Add(_variant_t("\"C:\\Documents and Settings\\模板.xlt\"")));//你可以自己創建一個模板,并自由設定目錄
//得到worksheets
workSheets.AttachDispatch(_workBook.GetWorksheets());
//得到workSheet
_workSheet.AttachDispatch(workSheets.GetItem(_variant_t("sheet1")));
//得到拷貝的母板
copyFrom.AttachDispatch(_workSheet.GetRange(_variant_t("A3"), _variant_t("Q6")));
copyTo.AttachDispatch(_workSheet.GetRange(_variant_t("A61"), _variant_t("A61")));
//得到全部的cells
range.AttachDispatch(_workSheet.GetCells());
///////////////////////////////////////////////////////////////////////////////////////////////////
// 上邊是頭
/*
中間要做的工作有這兩項:設置數據和拷貝格式
設置數據就是將數據庫中查詢出來的數據寫入表格,拷貝格式就是將表格拷貝到別的地方。
*/
//寫入數據
range.SetItem(_variant_t((long)3), _variant_t((long)1), _variant_t("寫入數據了"));
range.SetItem(_variant_t((long)5), _variant_t((long)1), _variant_t("重新寫入數據了"));
//拷貝一段區域到另外的一段區域
copyFrom.Copy(_variant_t(copyTo));
range.SetItem(_variant_t((long)61), _variant_t((long)1), _variant_t("123"));
//顯示excel表格
_app.SetVisible(TRUE);
//保存為文件
_app.SetDisplayAlerts(FALSE); //隱藏彈出的對話框
_workSheet.SaveAs("d:\\Test.xls",vtMissing,vtMissing,vtMissing,vtMissing,
vtMissing,vtMissing,vtMissing,vtMissing,vtMissing);
_app.Quit();
//下邊是尾
///////////////////////////////////////////////////////////////////////////////////////////////////
copyFrom.ReleaseDispatch();
copyTo.ReleaseDispatch();
range.ReleaseDispatch();
_workSheet.ReleaseDispatch();
workSheets.ReleaseDispatch();
_workBook.ReleaseDispatch();
workSheets.ReleaseDispatch();
_app.ReleaseDispatch();
_Application _app;
_Workbook _workBook;
_Worksheet _workSheet;
Worksheets workSheets;
Workbooks workBooks;
Range range;
Range copyFrom;
Range copyTo;
if(!_app.CreateDispatch("Excel.Application", NULL))
{
MessageBox("創建Excel服務失敗!", "信息提示", MB_OK);
return;
}
//利用模板建立新文檔
workBooks.AttachDispatch(_app.GetWorkbooks());
_workBook.AttachDispatch(workBooks.Add(_variant_t("\"C:\\Documents and Settings\\模板.xlt\"")));//你可以自己創建一個模板,并自由設定目錄
//得到worksheets
workSheets.AttachDispatch(_workBook.GetWorksheets());
//得到workSheet
_workSheet.AttachDispatch(workSheets.GetItem(_variant_t("sheet1")));
//得到拷貝的母板
copyFrom.AttachDispatch(_workSheet.GetRange(_variant_t("A3"), _variant_t("Q6")));
copyTo.AttachDispatch(_workSheet.GetRange(_variant_t("A61"), _variant_t("A61")));
//得到全部的cells
range.AttachDispatch(_workSheet.GetCells());
///////////////////////////////////////////////////////////////////////////////////////////////////
// 上邊是頭
/*
中間要做的工作有這兩項:設置數據和拷貝格式
設置數據就是將數據庫中查詢出來的數據寫入表格,拷貝格式就是將表格拷貝到別的地方。
*/
//寫入數據
range.SetItem(_variant_t((long)3), _variant_t((long)1), _variant_t("寫入數據了"));
range.SetItem(_variant_t((long)5), _variant_t((long)1), _variant_t("重新寫入數據了"));
//拷貝一段區域到另外的一段區域
copyFrom.Copy(_variant_t(copyTo));
range.SetItem(_variant_t((long)61), _variant_t((long)1), _variant_t("123"));
//顯示excel表格
_app.SetVisible(TRUE);
//保存為文件
_app.SetDisplayAlerts(FALSE); //隱藏彈出的對話框
_workSheet.SaveAs("d:\\Test.xls",vtMissing,vtMissing,vtMissing,vtMissing,
vtMissing,vtMissing,vtMissing,vtMissing,vtMissing);
_app.Quit();
//下邊是尾
///////////////////////////////////////////////////////////////////////////////////////////////////
copyFrom.ReleaseDispatch();
copyTo.ReleaseDispatch();
range.ReleaseDispatch();
_workSheet.ReleaseDispatch();
workSheets.ReleaseDispatch();
_workBook.ReleaseDispatch();
workSheets.ReleaseDispatch();
_app.ReleaseDispatch();
別人的代碼:
view plaincopy to clipboardprint?
.h文件:
#include "comdef.h"
#include "excel.h"
class ExcelFile
{
public:
void ShowInExcel(bool bShow);
CString GetCell(int iRow, int iColumn);
int GetCellInt(int iRow, int iColumn);
int GetRowCount();
int GetColumnCount();
bool LoadSheet(int iIndex);
CString GetSheetName(int iIndex);
static void InitExcel();
static void ReleaseExcel();
int GetSheetCount();
bool Open(CString FileName);
ExcelFile();
virtual ~ExcelFile();
protected:
private:
static _Application m_ExcelApp;
Workbooks m_Books;
_Workbook m_Book;
Worksheets m_sheets;
_Worksheet m_sheet;
Range m_Rge;
};
.cpp文件:
ExcelFile::ExcelFile()
{
}
ExcelFile::~ExcelFile()
{
m_Rge.ReleaseDispatch();
m_sheet.ReleaseDispatch();
m_sheets.ReleaseDispatch();
m_Book.ReleaseDispatch();
m_Books.ReleaseDispatch();
}
void ExcelFile::InitExcel()
{
//創建Excel 2000服務器(啟動Excel)
if (!m_ExcelApp.CreateDispatch("Excel.Application",NULL))
{
AfxMessageBox("創建Excel服務失敗!");
exit(1);
}
}
void ExcelFile::ReleaseExcel()
{
m_ExcelApp.ReleaseDispatch();
}
bool ExcelFile::Open(CString FileName)
{
//打開excel文件
//利用模板文件建立新文檔
m_Books.AttachDispatch(m_ExcelApp.GetWorkbooks(),true);
LPDISPATCH lpDis = NULL;
lpDis = m_Books.Add(_variant_t(FileName)); // 如何判斷文件是否打開?
if (lpDis)
{
m_Book.AttachDispatch(lpDis);
//得到Worksheets
m_sheets.AttachDispatch(m_Book.GetWorksheets(),true);
return true;
}
return false;
}
int ExcelFile::GetSheetCount()
{
return m_sheets.GetCount();
}
CString ExcelFile::GetSheetName(int iIndex)
{
_Worksheet sheet;
sheet.AttachDispatch(m_sheets.GetItem(_variant_t((long)iIndex)),true);
CString name = sheet.GetName();
sheet.ReleaseDispatch();
return name;
}
bool ExcelFile::LoadSheet(int iIndex)
{
LPDISPATCH lpDis = NULL;
m_Rge.ReleaseDispatch();
m_sheet.ReleaseDispatch();
lpDis = m_sheets.GetItem(_variant_t((long)iIndex));
if (lpDis)
{
m_sheet.AttachDispatch(lpDis,true);
m_Rge.AttachDispatch(m_sheet.GetCells(), true);
return true;
}
return false;
}
int ExcelFile::GetColumnCount()
{
Range range;
Range usedRange;
usedRange.AttachDispatch(m_sheet.GetUsedRange(), true);
range.AttachDispatch(usedRange.GetColumns(), true);
int count = range.GetCount();
usedRange.ReleaseDispatch();
range.ReleaseDispatch();
return count;
}
int ExcelFile::GetRowCount()
{
Range range;
Range usedRange;
usedRange.AttachDispatch(m_sheet.GetUsedRange(), true);
range.AttachDispatch(usedRange.GetRows(), true);
int count = range.GetCount();
usedRange.ReleaseDispatch();
range.ReleaseDispatch();
return count;
}
CString ExcelFile::GetCell(int iRow, int iColumn)
{
Range range;
range.AttachDispatch(m_Rge.GetItem (COleVariant((long)iRow),COleVariant((long)iColumn)).pdispVal, true);
COleVariant vResult =range.GetValue2();
CString str;
if(vResult.vt == VT_BSTR) //字符串
{
str=vResult.bstrVal;
}
else if (vResult.vt==VT_INT)
{
str.Format("%d",vResult.pintVal);
}
else if (vResult.vt==VT_R8) //8字節的數字
{
str.Format("%f",vResult.dblVal);
//str.Format("%.0f",vResult.dblVal);
//str.Format("%1f",vResult.fltVal);
}
else if(vResult.vt==VT_DATE) //時間格式
{
SYSTEMTIME st;
VariantTimeToSystemTime(vResult.date, &st);
}
else if(vResult.vt==VT_EMPTY) //單元格空的
{
str="(NULL)";
}
range.ReleaseDispatch();
return str;
}
int ExcelFile::GetCellInt(int iRow, int iColumn)
{
Range range;
range.AttachDispatch(m_Rge.GetItem(COleVariant((long)iRow),COleVariant((long)iColumn)).pdispVal, true);
COleVariant vResult =range.GetValue2();
int num;
num = (int)vResult.date;
range.ReleaseDispatch();
return num;
}
void ExcelFile::ShowInExcel(bool bShow)
{
m_ExcelApp.SetVisible(bShow);
}
本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/callmeback/archive/2009/02/18/3906310.aspx