如何在C++Builder中如何控制Excel (200分)

  • 主题发起人 主题发起人 metbo
  • 开始时间 开始时间
M

metbo

Unregistered / Unconfirmed
GUEST, unregistred user!
我打印出来的东西没有边框!!!!!
怎么办?[:(],也没有标题!!!!!
 
//画线
((RangePtr)ExcelWorksheet1->Cells->get_Item((Variant)(nRow),(Variant)nCol))->Borders->set_LineStyle(xlContinuous);
 

Selection.Borders(xlEdgeTop).LineStyle = xlContinuous
Selection.Borders(xlEdgeTop).Weight = xlMedium
Selection.Borders(xlEdgeTop).ColorIndex = xlAutomatic
加边框
 
贴段别人的代码给你:
-------------------------------------------------------
用C++Builder存取Excel 文件的两种方法。
---- 一.用OLE技术操纵Excel
---- OLE(对象链接与嵌入)是微软提出的标准,是应用程序间交换数据、相互操作的一种方式,MS OFFICE 提供了很强的 OLE 服务功能,VB、Delphi 及C++Builder等语言都可以以客户方式向Excel提交命令,以存取其数据,其中VB和Delphi最为简单,在语法上就采用类VBA的语句,本报九九年第48期的《在Delphi中操作Excel》一文曾介绍了Delphi的做法。我们原以为C++Builder照搬Delphi,结果语法编译都未能通过,查不到这方面具体介绍的资料。经反复试验方知:原来C++Builder采用间接的办法,利用变体类Variant所提供的四个“方法”向OLE服务程序提交操纵命令:
Variant Variant::OlePropertyGet(属性名,参数….);
// 取对象属性
Void Variant::OlePropertySet(属性名,参数….);
// 设置对象属性
Variant Variant::OleFunction(函数名,参数….);
// 运行对象的函数
void Variant::OleProcedure(过程名,参数….);
// 运行对象的过程
其头文件“vcl/utilcls.h”必须嵌在用户程序中,
对于这四个长长的方法名可在程序中用宏语句重新定义一下:
#define PG OlePropertyGet
#define PS OlePropertySet
#define FN OleFunction
#define PR OleProcedure
例如对于VB的提交语句:
Ex.ActiveWorkBook.ActiveSheet.Cells(1,2).Value=3
为便于理解在C++Builder对应的语句可分解为如下四句:
Variant t1=Ex.OlePropertyGet("ActiveWorkBook");
Variant t2=t1.OlePropertyGet("ActiveSheet");
Variant t3=t2.OlePropertyGet("Cells,1,2);
t3.OlePropertySet("Value",3);
可去掉中间变量将这四句合为一句,就是:
Ex.PG("ActiveWorkBook").PG("ActiveSheet").
PG("Cells",1,2).PS("Value",3);
我们将下面的VB程序片段改成对应的C++Builder程序以供参考:
Private Sub Command1_Click() ' Visual Basic程序片段
Dim Ex As Object,Wb As Object, Sh1 As Object
Set Ex = CreateObject("Excel.Application")
Ex.WorkBooks.Open("c:/book1.xls")
Set Wb = Ex.Activeworkbook
Set Sh1 = Wb.ActiveSheet
Text1.Text = Sh1.Cells(1, 1).Value
For i = 1 To 10: For j = 1 To 10
Sh1.Cells(i, j).Value = i * 100 + j
Next j: Next i
Wb.save: Wb.Close: Ex.Quit
End Sub ' - - - - - -- - - - - - - - - - -
#include "Unit1.h" // C++Builder程序片段
#include "vcl/utilcls.h" // util classes实用类说明
// …省写此处原一段代码
// 请在此处插入上面提及的四个宏定义语句
void __fastcall TForm1::Button1Click(TObject *Sender)
{ Variant Ex,Wb,Sh1;
Ex=Variant::CreateObject("Excel.Application");
Ex.PG("WorkBooks").PR("Open","c://book1.xls");
Wb=Ex.PG("ActiveWorkBook");
Sh1=Wb.PG("ActiveSheet");
Edit1->Text=Sh1.PG("Cells",1,1).PG("Value");
for (int i=1;i<=10;i++)
for (int j=1;j<=10;j++)
Sh1.PG("Cells",i,j).PS("Value",i*100+j);
Wb.PR("Save");
Wb.PR("Close");
}
---- 使用这种方法的程序运行时,必须保证系统中同时有MS OFFICE;下面我们再介绍另一种脱离MS OFFICE也能存取Excel表格的方法。
---- 二.用ODBC数据库技术存取Excel
---- ODBC就是开放式数据库链接标准,不同种类的数据库只需提供各自的ODBC 驱动程序就可按相同的命令操纵,微软同样为Excel提供了ODBC驱动程序,我们可在程序中象数据库一样存取Excel表格。定义好ODBC数据源后,实际试验中并没有如想象的那样简单,在控件Table的属性TableName中总是检索不到表名,同样也没有相关的资料可供查阅。通过对Excel的分析,终于发现了问题的关键所在:ODBC的表名并不就是Excel的工作表名(如Sheet1),在Excel 表格中必须对要求操作的行列区域定义一个“名称”作为数据库的表名,该区域的首行各列必须是字段名(否则首行数据会当成字段名),可以定义多个表名。具体操作步骤如下:
---- 1.在Excel上定义“表名”:
---- 运行Excel程序,打开或新建一表格,按下鼠标左键选择一片区域(起始行先填上字段名),再将鼠标位置点到左上角的地址栏,输入一表名如ABC,或者在菜单上选:“插入(I)”-“名称(N)”-“定义(D)”,再输入表名(若已定义, 可在此处删除掉),存盘退出(假定文件名取为c:/Book.xls),若嫌字段名行多余,存盘前可隐藏掉;
---- 2.定义ODBC数据源:
---- 从Windows桌面“我的电脑"进入“控制面板”,双击“32位ODBC”图标,运行“用户DSN”中的“添加(D)"后选“Excel Driver(*.xls)", 再点“完成”便弹出对话框,在“数据源名(N)"右边填一名称,如:excel01,在“版本(V)"上选“Excel97", 点中“选项”取消“只读”,在“选定工作目录”中,选定Excel文件名(本例 c:/Book1.xls),再点“确定”直至退出;
---- 3.设置Database控件避免登录检查:
---- 运行C++Builder,在Form1中加上Data Access的三个控件:Database1、 DataSource1、Table1,加入Database1的目的是为了避免打开数据库时出现登录框,为此双击此控件弹出一对话框,在Alias name中选ODBC数据源名(本例为Excel01),在Name中填上一新的别名(本例取Excel02),再点“Defaults” 出现一批参数缺省值,最后取消Options中的两项“Login prompt"与"Keep inactive connect”,点OK退出;
---- 4.设置其它控件属性:
---- 将Table1的属性DatabaseName选为步骤3中的新别名Excel02,再将另一属性TableName选为步骤1的表名(本例为ABC);将控件DataSource1的属性 DataSet选为Table1;双击Form1,在FormCreate事件子程序内加上一句 Table1- >Open( );
---- 5.查看数据库内容:
---- 为了直接看到Excel数据,在Form1中再加上Data Controls的控件DBGrid1 和DBNavigator1,将两者的属性DataSource都选为DataSource1,双击Table1 的属性Active将值改为true,等待数秒就可看到DBGrid1中出现数据,最后恢复Tabel1- >Active为false;
---- 注意:程序编译前,必须将刚打开的表关闭,即:使Table1- >Active为false, 否则程序运行时报告出错,因为Excel表总是被ODBC以“独占”方式打开;
----------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
靠,你查一下以前有篇Excel2000使用技巧的一篇文章,那里对EXCEL的操作写得非常的清楚
虽然不是很健全,但对于应用已经够啦。你可以把它做成一个DLL然后调用就可以啦/
透漏一个秘密,俺的自动报表设计器就要完成啦(大该还有三天的时间),功能是完全封装
EXCEL,提供数据接口,报表样式定制等等,简单的说就是把EXCEL的操作全部都移植到外面
。DLEPHI真是个好东西啊,做COM简直太方便啦。

 
我在作这个的时候有一个问题:当我调用quit方法退出后,在“任务管理器”中还存
在excel的进程,不知为什么?我的是在nt4.0和2000下运行的
 
如果我们用2K的操作系统,可以发现如果只是用ExcelApp.PR("Quit")后,Excel线程还在,这样,如果应用程序不退出,被调用的EXCEL文件就无法在SHELL下用EXCEL打开。因此,我的BCB调用EXCEL是这样的在最后要加上一点工作,就是把所有的VARIANT变量都要设置为UNASSIGNED,我的一个小例子如下,新建立一个EXCEL文件,显示有多少个表单(sheet),然后添一个表单并命名为test,然后给(1,2)单元赋值。最后删除一个sheet.
==============================================================================
h文件定义宏
#define OPG OlePropertyGet
#define OPS OlePropertySet
#define OFN OleFunction
#define OPR OleProcedure
#define PR Procedure
添加变量:
Variant Axl,Workbook,AxSheet,nms,bef,aft;

cpp函数
void __fastcall TForm1::Button1Click(TObject *Sender)
{
//赋值内容
AnsiString s="hello!";
//文件路径和保存名称
AnsiString filename=GetCurrentDir()+"//test.xls";
//判断文件是否存在
if (FileExists(filename))
{
//询问是否删除
if (MessageDlg("文件已经存在,是否删除?",
mtWarning, TMsgDlgButtons() << mbOK<<mbCancel, 0)==mrOk)
DeleteFile(filename);
else
{ShowMessage("放弃操作");return;}
}
//建立excel连接
Axl= Variant::CreateObject("Excel.Application");
Axl.OPS("Visible",false);
Workbook= Axl.OPG("Workbooks");
//创建一个新的excel工作本(文件)
Workbook.Exec(PR("Add"));
Workbook=Axl.OPG("ActiveWorkbook");
//显示存在的sheet数目
int count=Workbook.OPG("sheets").OPG("count");
ShowMessage(IntToStr(count)+"个表");
//添加一个Sheet,命名为test
aft=Workbook.OPG("sheets",count);
Workbook.OPG("sheets").OPR("Add",bef.NoParam(),aft);
AxSheet=Workbook.OPG("ActiveSheet");
//Rename
AxSheet.OPS("Name","test");
//给单元1,2赋值
AxSheet.OPG("Cells")
.OPG("Item",(Variant)1,(Variant)2)
.OPS("Value",s.c_str());
// .Exec(PropertySet("Value")<< s.c_str());
//以上两种方式都可以
//选择第一张sheet
int sheetnum=1;
AxSheet=Workbook.OPG("sheets",sheetnum);
//选择名字为sheet2的表单
AnsiString shname="sheet2";
AxSheet=Workbook.OPG("sheets",shname.c_str());
//关闭警告提示
Workbook.OPG("Application").OPS("DisplayAlerts",false);
//删除选定表单
AxSheet.OFN("Delete");
//打开警告提示
Workbook.OPG("Application").OPS("DisplayAlerts",false);
//保存文件,两种方式都可以
//Workbook.Exec(PR("SaveAs")<<filename);
Workbook.OPR("SaveAs",filename.c_str());
//结束退出
Workbook.OPR("Close");
Axl.OFN("Quit");
//结束,如果没有如下代码,EXCEL线程直到应用程序退出才结束。
Axl=Unassigned;
Workbook=Unassigned;
AxSheet=Unassigned;
bef=Unassigned;
aft=Unassigned;
nms=Unassigned;

ShowMessage("Welldo
ne boy!");
 
#include <comobj.hpp>
void TxtToExcel(TQuery *Query)
{
Query->DisableControls();
Variant Excel,cell;
int iFieleCount=0;
try
{
Excel=CreateOleObject("Excel.Application");
Excel.Exec(PropertyGet("Workbooks")).Exec(Procedure("Add"));
cell=Excel.Exec(PropertyGet("ActiveCell"));
}
catch(...)
{
ShowMessage("您的系统没有安装Microsoft Excel,请先安装Microsoft Excel!");
return;
}
Excel.Exec(PropertySet("Visible")<<true);
for (int i=0;i<Query->Fields->Count;i++)
{
if(Query->Fields->Fields->Visible)
{
cell.Exec(PropertySet("Value")<<Query->Fields->Fields->DisplayLabel);
cell=cell.Exec(Function("Next"));
++iFieleCount;
}
if(!Query->Fields->Fields->Visible)
goto zhe;
}
zhe: Query->First();
int row=2;
Query->First();
try
{
while (!Query->Eof)
{
cell=Excel.Exec(PropertyGet("Range")<<("A"+IntToStr(row)+":A"+IntToStr(row)));
for (int i=0;i<iFieleCount;i++)
{
cell.Exec(PropertySet("Value")<<Query->Fields->Fields->AsString);
cell=cell.Exec(Function("Next"));
}
Query->Next();
row++;
}
Excel.Exec(Procedure("Quit"));
Query->EnableControls();
}
catch(...)
{
Query->EnableControls();
}
}
 
后退
顶部