C#中的受管代码和非受管代码
在.net公用语言框架内运行的程序为受管代码。受管代码在程序中所有类型都受到严格检
查,没有指针,对内存的管理完全由运行系统控制。受控状态下,编写程序更为容易,且更少出错,我们可以花更多的时间在解决实际问题上而不是在计算机语言问
题上。相对而言,那些在.NET框架外运行的程序为非受管代码。比如:COM组件、ActiveX组件、Win32
API函数、指针运算等。C#编程中在某些特定情况下,需要运用非受管代码,例如,要利用一个成熟的COM组件,或者调用一个API函数,或者用指针去编
写实时/高效程序等。
Visual C#中调用Excel的COM组件
一个.NET组件事实上是一个.NET下的DLL,它包含的不仅是运行程序本身,更重要的是包含这个DLL的描述信息(Meta
Data,即元数据),而一个COM组件是用其类库(TLB)储存其描述信息。这些COM组件都是非受管代码,要在Visual
C#中使用这些非受管代码的COM组件,就必须把他们转换成受管代码的.NET组件。所以在用Visual
C#调用Excel表格之前,必须完成从COM组件的非受管代码到受管代码的类库的转换。
1、将Excel的COM组件转换为.NET组件
在项目中打开Add Reference对话框,选择COM栏,之后在COM列表中找到"Microsoft Excel 9.0 Object
Library"(Office
2000),然后将其加入到项目的References中即可。Visual C#.NET会自动产生相应的.NET组件文件,以后即可正常使用。
这个转换形成.NET组件不能单独使用,它不过是以前的COM组件的一个外层包装,在.NET中可以通过这个外层包装去发现原来的COM组件并调用其相应的界面函数。所以它必须与原来的COM组件一起起作用。
2、Visual C#打开Excel表格
事实上,在C#中使用一个经转换的COM组件和使用任何一个其它.NET组件完全一样。可以用new关键字创建一个经转换的COM组件,然后再像使用任何一个其它C#对象一样使用这个组件对象。
在转换后的.NET组件中定义了一个命名空间Excel,在此命名空间中封装了一个类Application,这个类和启动Excel表格有非常重要的关系,在Visual
C#中,只需要下列三行代码就可以完成打开Excel表格的工作,具体如下:
Excel.Application excel = new
Excel.Application ();//引用Excel对象 excel.Application.Workbooks.Add ( true );//引用Excel工作簿 excel.Visible = true ;//使Excel可视 |
Excel.Application excel = new
Excel.Application () ; excel.Application.Workbooks.Add ( true ) ; excel.Cells[ 1 , 1 ] = "First Row First Column" ; excel.Cells[ 1 , 2 ] = "First Row Second Column" ; excel.Cells[ 2 , 1 ] = "Second Row First Column" ; excel.Cells[ 2 , 2 ] = "Second Row Second Column" ; excel.Visible = true ; |
4、实例
下面实例在C#中连接Oracle数据库(Name),从表(TableName)中读取数据,并写入Excel。
string
cnString="Provider=msdaora.1;Data source=Name; "; cnString=cnString+"user id=UserName;password=Password"; try { OleDbConnection cn=new OleDbConnection (cnString); cn.Open (); try { string s="select * from Name.TableName"; OleDbCommand cmd=new OleDbCommand (s,cn); OleDbDataReader dr=cmd.ExecuteReader (); Excel.Application xlApp = new Excel.Application(); if(xlApp==null){MessageBox.Show ("Can‘t open Excel!");return;} xlApp.Application .Workbooks .Add (true); int row=2,fieldcount; fieldcount=dr.FieldCount ; for(int col=0;col<fieldcount;col++) xlApp.Cells [1,col+1]=dr.GetName(col); while (dr.Read ()) { for(int col=0;col<fieldcount;col++) xlApp.Cells [row,col+1]=dr.GetValue(col).ToString(); row++; } xlApp.Visible =true; xlApp=null; } catch(Exception ex ){MessageBox.Show (ex.Message );} finally {cn.Close();} } catch(Exception ex){MessageBox.Show (ex.Message );} } } |
| ||||||
前些日子,有很多朋友说需要C#导出到Excel的代码,现共享给大家 /// <summary> /// <summary> 这种方法目前最有效,如果有不明白的地方可以来信交流
|
首先将excel.exe copy 到 ..\Microsoft Visual Studio .NET 2003\SDK\v1.1\Bin目录下
利用.net 中带的工具在命令提示符下执行tlbimp
excel.exe.这样就不会因为你的Excel是xp或2000的不同要去找不同的*.olb文件,还有一点就是因为在2000以后的版本中没有了excel9.olb这个文件了。
通过执行tlbimp excel.exe后我们会得到excel.dll文件。
只要有了这个Excel.dll,现在我们就能使用Excel的各种操作函数了。
下面就让我们具体看看C#是如何使用这些东东吧。
1.
创建一个新Excel的Application:
Application exc = new Application();
if (exc == null) {
Console.WriteLine("ERROR: EXCEL couldn‘t be started");
return 0;
}
2. 让这个工程可见:
exc.set_Visible(0, true);
3. 获取WorkBooks集合:
Workbooks
workbooks = exc.Workbooks;
4. 加入新的WorkBook:
_Workbook workbook =
workbooks.Add(XlWBATemplate.xlWBATWorksheet, 0);
5. 获取WorkSheets集合:
_Worksheet worksheet = (_Worksheet) sheets.get_Item(1);
if (worksheet ==
null) {
Console.WriteLine ("ERROR in worksheet == null");
}
6.
给单元格设置变量:
例程:
现在我们来看看如何使用数组,他有些类似于设置单元格。仅仅需要的改变只是args2[0] = array2;
大家需要了解Tlbimp这个工具的使用啊:)这个东东很有用,可以将普通Win32程序移植到.Net下面来:)
如果操作的excel的格式很简单,就是一般的表的结构,那么其实操作EXCEL文件跟操作ACCESS数据库文件的方法几乎一样。
(需要注意的地方就是,1、程序会把EXCLE表中的第一行记录作为列名;2、在使用EXCLE表的时候,
要在表名后面加上符号$)
下面,我给你帖一段如何连接和读取EXCEL文件的代码吧:
如果是在asp.net 下使用的话,要记得在 <system.web>中添加<identity
impersonate="true"/>
否则就会出现 “异常详细信息: System.UnauthorizedAccessException:
拒绝访问“。
=======================================================
一、调用Excel的方法:一般情况下有两种方法调用Excel:
1、
直接使用Delphi自带的组件:在Form中分别放入ExcelApplication,
ExcelWorkbook和ExcelWorksheet。
2、动态创建Excel文件:首先创建 Excel
对象,使用ComObj,Excel2000:
var ExcelApp: Variant;
ExcelApp := CreateOleObject(
‘Excel.Application‘ );
二、导入数据:在程序中,我们可以将查询到的数据(SQL、Access、)导入到Excel中。例如:用Adoquery查询Access中的数据:
1、先查到所需的数据;
2、导入:i:=1;
Adoquery.First;
while not Adoquery.Eof do
Begin
ExcelApp.WorkSheets[1].Cells[i,1].Value := i;//添加序号的值
ExcelApp.WorkSheets[1].Cells[i,2].Value :=
Adoquery.FieldByName(‘cp_name‘).AsString;
……
Inc(i);
Adoquery.Next;
End;
当然也可以把Adotable、Adoquery、Table、Query等组件的数据导入到Excel中。
三、Excel的处理:如果在你已知Excel格式的情况下,可以控制Excel,如下:
1、 显示当前窗口:ExcelApp.Visible :=
True;
2、 更改 Excel 标题栏:ExcelApp.Caption := ‘标题内容‘;
3、
添加新工作簿:ExcelApp.WorkBooks.Add;
4、
设置第2个工作表为活动工作表:ExcelApp.WorkSheets[2].Activate;
5、
给单元格赋值:ExcelApp.Cells[1,1].Value := ‘第一行第一列‘;
6、
设置指定列的宽度(单位:字符个数),以第一列为例:
ExcelApp.ActiveSheet.Columns[1].ColumnsWidth :=
5;
7、 设置指定行的高度(单位:磅)(1磅=0.035厘米),以第二行为例:
ExcelApp.ActiveSheet.Rows[2].RowHeight := 1/0.035; // 1厘米
8、文字水平居中:Excelid.worksheets[1].Rows[1].HorizontalAlignment := $FFFFEFF4;
文字垂直居中:Excelid.worksheets[1].Rows[1].VerticalAlignment := $FFFFEFF4;
9、 插入一行或一列:a. ExcelApp.ActiveSheet.Rows[2].Insert;
b.
ExcelApp.ActiveSheet.Columns[1].Insert;
10、 删除一行或一列:a.
ExcelApp.ActiveSheet.Rows[2].Delete;
b.
ExcelApp.ActiveSheet.Columns[1].Delete;
11、合并单元格:ExcelApp.worksheets[1].range[A1:F8‘].Merge(abc);注:要声明变量abc:
Variant;
12、竖行显示文字:ExcelApp.worksheets[1].Cells.Item[1,1].Orientation:=
xlVertical;
13、单元格加边线:ExcelApp.worksheets[1].Range[A1:F8].Borders.LineStyle
:= 1;
14、在第8行之前插入分页符:ExcelApp.WorkSheets[1].Rows[8].PageBreak := 1;
15、在第4列之前删除分页符:ExcelApp.ActiveSheet.Columns[4].PageBreak := 0;
16、指定边框线宽度:ExcelApp.ActiveSheet.Range[ ‘B3:D4‘ ].Borders[2].Weight := 3;
1-左
2-右 3-顶 4-底 5-斜( \ ) 6-斜(
/ )
17、拷贝操作:a.拷贝整个工作表:ExcelApplication1.ActiveSheet.Used.Range.Copy;
b.拷贝指定区域:ExcelApplication1.ActiveSheet.Range[ ‘A1:E2‘ ].Copy;
c.从A1位置开始粘贴:ExcelApplication1.ActiveSheet.Range.[ ‘A1‘ ].PasteSpecial;
d.从文件尾部开始粘贴:ExcelApplication1.ActiveSheet.Range.PasteSpecial;
18、清除第一行第四列单元格公式:ExcelApp.ActiveSheet.Cells[1,4].ClearContents;
19、工作表保存:if
not ExcelApp.ActiveWorkBook.Saved then
ExcelApp.ActiveSheet.PrintPreview;
20、工作表另存为:ExcelApp.SaveAs( ‘C:\Excel\Demo1.xls‘ );
21、放弃存盘:ExcelApp.ActiveWorkBook.Saved := True;
22、关闭工作簿:ExcelApp.WorkBooks.Close;
23、退出 Excel:ExcelApp.Quit;
下面是有关打印页面控制的语句:
24、设置第一行字体属性:ExcelApp.ActiveSheet.Rows[1].Font.Name := ‘隶书‘;
ExcelApp.ActiveSheet.Rows[1].Font.Color := clBlue;
ExcelApp.ActiveSheet.Rows[1].Font.Bold := True;
ExcelApp.ActiveSheet.Rows[1].Font.UnderLine := True;
ExcelApp.ActiveSheet.Rows[1].Font.size:=10;
25、进行页面设置:a.页眉:ExcelApp.ActiveSheet.PageSetup.CenterHeader := ‘报表演示‘;
b.页脚:ExcelApp.ActiveSheet.PageSetup.CenterFooter := ‘共&N页 第&P页‘;
c.页眉到顶端边距2cm:ExcelApp.ActiveSheet.PageSetup.HeaderMargin := 2/0.035;
d.页脚到底端边距3cm:ExcelApp.ActiveSheet.PageSetup.HeaderMargin := 3/0.035;
e.顶边距2cm:ExcelApp.ActiveSheet.PageSetup.TopMargin := 2/0.035;
f.底边距2cm:ExcelApp.ActiveSheet.PageSetup.BottomMargin := 2/0.035;
g.左边距2cm:ExcelApp.ActiveSheet.PageSetup.LeftMargin := 2/0.035;
h.右边距2cm:ExcelApp.ActiveSheet.PageSetup.RightMargin := 2/0.035;
i.页面水平居中:ExcelApp.ActiveSheet.PageSetup.CenterHorizontally := 2/0.035;
j.页面垂直居中:ExcelApp.ActiveSheet.PageSetup.CenterVertically := 2/0.035;
k.打印单元格网线:ExcelApp.ActiveSheet.PageSetup.PrintGridLines := True;
26、打印预览工作表:ExcelApp.ActiveSheet.PrintPreview;
27、打印输出工作表:ExcelApp.ActiveSheet.PrintOut;
对Excel的其他控制:
28、excel的多单元格合计功能:ExcelApp..Cells[ARow, ACol].Formula
:= ‘=
SUM($+IntToStr(BeginRow) +:$ + IntToStr(EndRow) +‘);
注:声明变量ARow, ACol:
Integer;
29、打开已经存在的Excel文件: ExcelApplication1.Workbooks.Open (c:\a.xls
EmptyParam,EmptyParam,EmptyParam,EmptyParam,
EmptyParam,EmptyParam,EmptyParam,EmptyParam,
EmptyParam,EmptyParam,EmptyParam,EmptyParam,0);
举例来说
二、选择excel表‘按钮,用于打开EXCEL文件,其代码如下:
procedure
TForm1.Button1Click(Sender: TObject);
var i,j:integer;
begin
opendialog1.InitialDir:=ExtractFileDir(paramstr(0));//文件的打存放初始路径
opendialog1.Execute;
Try
ExcelApplication1.Connect;//EXCEL应用程序
Except
MessageDlg(‘Excel may not be installed‘,mtError, [mbOk], 0);
Abort;
End;
ExcelApplication1.Visible[0]:=True;
ExcelApplication1.Caption:=‘Excel Application‘;
try
excelapplication1.Workbooks.Open(opendialog1.FileName,
null,null,null,null,null,null,null,null,null,null,null,null,0);//打开指定的EXCEL
文件
except
begin
ExcelApplication1.Disconnect;//出现异常情况时关闭
ExcelApplication1.Quit;showmessage(‘请选择EXCEL电子表格!‘);
exit;
end;
end;
ExcelWorkbook1.ConnectTo(ExcelApplication1.Workbooks[1]);//ExcelWorkbook1与Eexcelapplication1建立连接
ExcelWorksheet1.ConnectTo(ExcelWorkbook1.Worksheets[1] as
_Worksheet);//Excelworksheet1与Excelworkbook1建立连接
//开始从EXCEL中取数,放到stringgrid1中,取完数后关闭EXCEL
for i:=1 to 1000 do//最大取值1000
for
j:=1 to 6 do
begin
if trim(excelworksheet1.cells.item[i+1,1])<>‘‘
then
begin
stringgrid1.rowCount:=i+1;
stringgrid1.Cells[j,i]:=ExcelWorksheet1.Cells.Item[i+1,j];
end
else
begin
label3.caption:=inttostr(i-1);
ExcelApplication1.Disconnect;
ExcelApplication1.Quit;
//将第一条数据赋给编辑框
edit2.text:=stringgrid1.Cells[1,1];
edit1.text:=stringgrid1.Cells[2,1];
edit3.text:=stringgrid1.Cells[3,1];
edit4.text:=stringgrid1.Cells[4,1];
edit5.text:=stringgrid1.Cells[5,1];
exit;
end;
end;
end;
三、‘下一条记录‘按钮,完成记录向下移动,代码如下:
procedure TForm1.Button2Click(Sender:
TObject);
var x:integer;
begin
x:=stringgrid1.row+1;
if x<>
stringgrid1.RowCount then
begin
stringgrid1.row:=stringgrid1.row+1;
label1.caption:=inttostr(x);
edit2.text:=stringgrid1.Cells[1,x];
edit1.text:=stringgrid1.Cells[2,x];
edit3.text:=stringgrid1.Cells[3,x];
edit4.text:=stringgrid1.Cells[4,x];
edit5.text:=stringgrid1.Cells[5,x];
exit;
end
else
showmessage(‘已到第一条记录!‘);
end;
四、‘上一条记录‘,完成记录上移,代码如下:
var x:integer;
begin
x:=stringgrid1.row-1;
if
x<>0 then
begin
stringgrid1.row:=stringgrid1.row-1;
label1.caption:=inttostr(x);
edit2.text:=stringgrid1.Cells[1,x];
edit1.text:=stringgrid1.Cells[2,x];
edit3.text:=stringgrid1.Cells[3,x];
edit4.text:=stringgrid1.Cells[4,x];
edit5.text:=stringgrid1.Cells[5,x];
exit;
end
else
showmessage(‘已到最后一条记录!‘);
end;
五、stringgrid中上下移动时代码:
procedure TForm1.StringGrid1Click(Sender: TObject);
var i:integer;
begin
i:=stringgrid1.Row;
label1.caption:=inttostr(i);
edit1.text:=stringgrid1.Cells[2,i];
edit2.text:=stringgrid1.Cells[1,i];
edit3.text:=stringgrid1.Cells[3,i];
edit4.text:=stringgrid1.Cells[4,i];
edit5.text:=stringgrid1.Cells[5,i];
end;
图片插入到EXCEL中
uses:clipbrd
function
begin
var
MyFormat:Word;
AData:THandle;
//临时句柄变量。
APalette:HPALETTE; //临时变量。
Stream1:TMemoryStream;//TBlobStream
xx:tbitmap;
Stream1:= TMemoryStream.Create;
TBlobField(query.FieldByName(‘存储图片的字段‘)).SaveToStream(Stream1);
Stream1.Position :=0;
xx:=tbitmap.Create ;
xx.LoadFromStream(Stream1);
xx.SaveToClipboardFormat(MyFormat,AData,APalette);
ClipBoard.SetAsHandle(MyFormat, AData);
myworksheet1.Range[‘g3‘,‘h7‘].select;//myworksheet1是当前活动的sheet页
myworksheet1.Paste;
end;
============================================================
各位兄弟:读取EXCEL数据是有的麻烦,经过研究,已搞掂,C#的代码如下,至于写数据则较简单:
Excel.ApplicationClass excel=new Excel .ApplicationClass
();
excel.Workbooks .Add(path);//打开麻烦,增加一个较好处理
Excel.Worksheet
worksheet=(Excel.Worksheetexcel.Worksheets.get_Item (1);
for(int i=1;i<=row;i++)
{
Excel.Range r=worksheet.get_Range ("A"+i.ToString
(),"A"+i.ToString ());
r.Select ();
labcellname.Text =excel.ActiveCell.Text .ToString
().Trim ();
Application.DoEvents ();
labnum.Text =i.ToString ();
Application.DoEvents ();
}
excel.Quit
();
C#操作Excel的方法
一. 直接调用COM组件
(如excel 2003)引用COM组件,添加excel的com对象Microsoft Excel 11.0
Object,然后在引用中可以看到
Microsoft.Office.Core,Excel,VBIDE三个对象。此时在程序中需要引入:
using System.Reflection;
using Microsoft.Office.Core;
using
Microsoft.Office.Interop.Excel;
基本的操作方式:
Application
excel = new
ApplicationClass();
excel.Visible =
false;
Workbook wb = excel.Workbooks._Open(modelFile, Missing.Value, Missing.Value,
Missing.Value,
Missing.Value
, Missing.Value, Missing.Value, Missing.Value,
Missing.Value
, Missing.Value, Missing.Value, Missing.Value, Missing.Value);
Worksheet xSheet = (Worksheet)wb.Sheets[1];
//Sheets sts =
wb.Worksheets;
//_Worksheet st =
(_Worksheet)sts.get_Item(1);
//st.Cells[3, 5] = "111";
//直接在cell上写值
//st.Cells[2, 5] =
"hahaha";
Range range = xSheet.get_Range("A3",
"H3");
object[] objLines = { a200.Date, a200.PreviousClosePrice, a200.OpenPrice,
a200.High, a200.Low, a200.Close, a200.Change, a200.ChangeRate
};
range.set_Value(Missing.Value, objLines);
wb.SaveAs(fileName, Missing.Value, Missing.Value, Missing.Value,
Missing.Value, Missing.Value,
XlSaveAsAccessMode.xlNoChange,
Missing.Value, Missing.Value, Missing.Value, Missing.Value, Missing.Value);
//wb.Close(false, Missing.Value, Missing.Value);
//以下步骤必须进行,否则Excel在进程里不能自动释放
NAR(range);
NAR(xSheet);
wb.Close(false, Missing.Value,
Missing.Value);
NAR(wb);
excel.Quit();
NAR(excel);
System.GC.Collect();
//以往的做法是将进程里所以的Excel进程Kill掉,不推荐!
private void NAR(object o)
{
try
{
System.Runtime.InteropServices.Marshal.ReleaseComObject(o);//强制释放一个对象
}
catch
{ }
finally
{
o = null;
}
}
二. 通过OLEDB操作Excel
OleDbConnection conn =
null;
try
{
//fileName
表示要操纵的Excel的文件路径,如果excel不存在,现创建它,可以通过模版文件复制创建。
string
strConn;
strConn = "Provider=Microsoft.Jet.OLEDB.4.0;"
+
"Data Source=" + fileName + ";"
+
"Extended Properties=‘Excel 8.0;HDR=no;IMEX=0‘";
conn = new
OleDbConnection(strConn);
conn.Open();
System.Data.OleDb.OleDbCommand cmd = new
System.Data.OleDb.OleDbCommand();
cmd.Connection =
conn;
//在Excel的Sheet1的A3到H3处插入数据
cmd.CommandText = "insert into [Sheet1$A3:H3] (F1,F2,F3,F4,F5,F6,F7,F8)
values(‘" + a200.Date +
"‘,‘"
+ a200.PreviousClosePrice + "‘,‘" + a200.OpenPrice + "‘,‘" + a200.High + "‘,‘"
+ a200.Low + "‘,‘" + a200.Close + "‘,‘"
+
a200.Change + "‘,‘" + a200.ChangeRate +
"‘)";
cmd.ExecuteNonQuery();
conn.Close();
}
catch
(Exception
e)
{
if (conn !=
null)
conn.Close();
Console.WriteLine(e.ToString());
}
注:
1)使用 Excel 工作簿时,默认情况下,区域中的第一行是标题行(或字段名称)。如果第一个区域不包含标题,您可以在连接字符串的扩展属性中指定
HDR=NO。
如果您在连接字符串中指定 HDR=NO,Jet OLE DB 提供程序将自动为您命名字段(F1 表示第一个字段,F2
表示第二个字段,依此类推);
2)IMEX=1将所有读入数据
看作字符,其他值(0、2)请查阅相关帮助文档;3)如果出现“找不到可安装的isam”错误,一般是连接字符串错误。
3、从excel文件读取数据
string sql = "select * from
[sheet1$]";
DoOleSql(sql,"test.xls");
4、更新excel文件中的数据
string sql = "update [sheet1$] set FieldName1=‘333‘ where
FieldName2=‘b3‘";
DoOleSql(sql,"test.xls");
5、向excel文件插入数据
string sql = "insert into
[sheet1$](FieldName1,FieldName2,…)
values(‘a‘,’b’,…)";
DoOleSql(sql,"test.xls");
6、删除excel文件中的数据:不提倡使用这种方法
7、对于非标准结构的excel表格,可以指定excel中sheet的范围
1)读取数据:string
sql = "select * from [sheet1$A3:F20]";
2)更新数据:string sql = "update
[sheet1$A9:F15] set FieldName=‘333‘ where
AnotherFieldName=‘b3‘";
3)插入数据:string sql = "insert into
[sheet1$A9:F15](FieldName1,FieldName2,…)
values(‘a‘,’b’,…)";
4)删除数据:不提倡
注:1)代码根据需要可以自行修改;2)如果出现“操作必须使用一个可更新的查询”错误,可能sql语句中对excel文件中的“字段”引用有错误,或对excel文件不
具有“修改”权限;3)如果出现“不能扩充选定范围”错误,可能是对excel文件引用的“范围”有错误。
原文:http://www.cnblogs.com/net-study/p/3708558.html