导出写Excel文件,Sheet名与导出文件名相同

news/2024/7/21 4:57:47 标签: excel, stream, integer, constructor, string, delphi
使用方法是
Var  MyExcel : TDS2Excel
Begin
  :  :  :  :
    MyExcel := TDS2Excel.Create(aDataSet: TDataSet;aDBGrid:TDBGrid)
    Save2File(XLS文件名, WillWriteHead); //文件名,字段名做表格列头。
    // Save2Files(WillWriteHead: Boolean); 该过程会自动弹出文件对话框,供用户自己选择文件名
  :  :  :  :
End ;

实际上Express的cxGride控件也是采用这种方法实现数据表格导出到Excal的。
下面是实现的类
===============================
DELPHI 写EXCEL的XLS格式文件
===============================
unit ObjectUnit;

interface

Uses
DB, Classes, Dialogs,DBGrids,Controls;
var
CXlsBof: array[0..5] of Word = ($809, 8, 0, $10, 0, 0);
CXlsEof: array[0..1] of Word = ($0A, 00);
CXlsLabel: array[0..5] of Word = ($204, 0, 0, 0, 0, 0);
CXlsNumber: array[0..4] of Word = ($203, 14, 0, 0, 0);
CXlsRk: array[0..4] of Word = ($27E, 10, 0, 0, 0);
CXlsBlank: array[0..4] of Word = ($201, 6, 0, 0, $17);


Type
TDS2Excel = Class(TObject)
Private
  FCol: word;
  FRow: word;
  FDataSet: TDataSet;
  FDbGrid  :TDbGrid;
  Stream: TStream;
  FWillWriteHead: boolean;
  FBookMark: TBookmark;
  procedure IncColRow;
  procedure WriteBlankCell;
  procedure WriteFloatCell(const AValue: Double);
  procedure WriteIntegerCell(const AValue: Integer);
  procedure WriteStringCell(const AValue: string);
  procedure WritePrefix;
  procedure WriteSuffix;
  procedure WriteTitle;
  procedure WriteDataCell;

  procedure Save2Stream(aStream: TStream);
Public
  procedure Save2File(FileName: string; WillWriteHead: Boolean);
  procedure Save2Files(WillWriteHead: Boolean);
  Constructor Create(aDataSet: TDataSet;aDBGrid:TDBGrid);
end;

implementation

uses SysUtils;

Constructor TDS2Excel.Create(aDataSet: TDataSet;aDBGrid:TDBGrid);
begin
inherited Create;
FDataSet := aDataSet;
FDbGrid  :=aDbGrid;
end;


procedure TDS2Excel.IncColRow;
begin
if FDbGrid <>nil then
begin
if FCol = FDbGrid.Columns.Count - 1 then
begin
  Inc(FRow);
  FCol :=0;
end
else
  Inc(FCol);
end else
begin
if FCol = FDataSet.FieldCount - 1 then
begin
  Inc(FRow);
  FCol :=0;
end
else
  Inc(FCol);
end;
end;

procedure TDS2Excel.WriteBlankCell;
begin
CXlsBlank[2] := FRow;
CXlsBlank[3] := FCol;
Stream.WriteBuffer(CXlsBlank, SizeOf(CXlsBlank));
IncColRow;
end;

procedure TDS2Excel.WriteFloatCell(const AValue: Double);
begin
CXlsNumber[2] := FRow;
CXlsNumber[3] := FCol;
Stream.WriteBuffer(CXlsNumber, SizeOf(CXlsNumber));
Stream.WriteBuffer(AValue, 8);
IncColRow;
end;

procedure TDS2Excel.WriteIntegerCell(const AValue: Integer);
var
V: Integer;
begin
CXlsRk[2] := FRow;
CXlsRk[3] := FCol;
Stream.WriteBuffer(CXlsRk, SizeOf(CXlsRk));
V := (AValue shl 2) or 2;
Stream.WriteBuffer(V, 4);
IncColRow;
end;

procedure TDS2Excel.WriteStringCell(const AValue: string);
var
L: Word;
begin
L := Length(AValue);
CXlsLabel[1] := 8 + L;
CXlsLabel[2] := FRow;
CXlsLabel[3] := FCol;
CXlsLabel[5] := L;
Stream.WriteBuffer(CXlsLabel, SizeOf(CXlsLabel));
Stream.WriteBuffer(Pointer(AValue)^, L);
IncColRow;
end;

procedure TDS2Excel.WritePrefix;
begin
Stream.WriteBuffer(CXlsBof, SizeOf(CXlsBof));
end;

procedure TDS2Excel.WriteSuffix;
begin
Stream.WriteBuffer(CXlsEof, SizeOf(CXlsEof));
end;

procedure TDS2Excel.WriteTitle;
var
n: word;
begin
if FDbGrid <> nil then
for n := 0 to FDBGrid.Columns.Count - 1 do
  WriteStringCell(FDBGrid.Columns[n].Title.Caption)
else
for n := 0 to FDataSet.FieldCount - 1 do
  WriteStringCell(FDataSet.Fields[n].FieldName);
end;

procedure TDS2Excel.WriteDataCell;
var
n: word;
begin

WritePrefix;
if FWillWriteHead then WriteTitle;
FDataSet.DisableControls;
FBookMark := FDataSet.GetBookmark;
FDataSet.First;

if FDbGrid=nil then
begin
  while not FDataSet.Eof do
  begin
  for n := 0 to FDataSet.FieldCount - 1 do
  begin
    try
    if FDataSet.Fields[n].IsNull then
      WriteBlankCell
    else begin
      case FDataSet.Fields[n].DataType of
        ftSmallint, ftInteger, ftWord, ftAutoInc, ftBytes:
            WriteIntegerCell(FDataSet.Fields[n].AsInteger);
        ftFloat, ftCurrency, ftBCD:
            WriteFloatCell(FDataSet.Fields[n].AsFloat);
        ftTypedBinary:
      else
        WriteStringCell(FDataSet.Fields[n].AsString);
      end;
    end;
    except
      WriteBlankCell;
    end;
  end;
  FDataSet.Next;
  end;
end

else

begin
  while not FDbGrid.DataSource.DataSet.Eof do
  begin

for n := 0 to FDbGrid.Columns.Count - 1 do
  begin

    if FDbGrid.Columns[n].Field.IsNull then
      WriteBlankCell
    else begin
      case FDbGrid.Columns[n].Field.DataType of
        ftSmallint, ftInteger, ftWord, ftAutoInc, ftBytes:
            WriteIntegerCell(FDbGrid.Columns[n].Field.AsInteger);
        ftFloat, ftCurrency, ftBCD:
            WriteFloatCell(FDbGrid.Columns[n].Field.AsFloat);
      else
        WriteStringCell(FDbGrid.Columns[n].Field.AsString);
      end;
    end;
  end;
  FDbGrid.DataSource.DataSet.Next
  end;
end;


WriteSuffix;
if FDataSet.BookmarkValid(FBookMark) then FDataSet.GotoBookmark(FBookMark);
FDataSet.EnableControls;
end;

procedure TDS2Excel.Save2Stream(aStream: TStream);
begin
FCol := 0;
FRow := 0;
Stream := aStream;
WriteDataCell;
end;

procedure TDS2Excel.Save2File(FileName: string; WillWriteHead: Boolean);
var
aFileStream: TFileStream;
begin
FWillWriteHead := WillWriteHead;
if FileExists(FileName) then DeleteFile(FileName);
aFileStream := TFileStream.Create(FileName, fmCreate);
Try
  Save2Stream(aFileStream);
Finally
  aFileStream.Free;
end;
end;

procedure TDS2Excel.Save2FileS(WillWriteHead: Boolean);
var
SaveDialog11: TSaveDialog;
begin
SaveDialog11 := TSaveDialog.Create(nil);
Try
  SaveDialog11.Filter := 'Excel文档|*.xls';
  SaveDialog11.InitialDir := 'D:/';
  SaveDialog11.FileName:='*.xls';
  if not SaveDialog11.Execute then exit;
  if FileExists(SaveDialog11.FileName) then DeleteFile(SaveDialog11.FileName);
  Save2File(SaveDialog11.FileName, WillWriteHead);
Finally
  SaveDialog11.Free;
end;
end;

end.  

http://www.niftyadmin.cn/n/1412719.html

相关文章

Docker (二) :Centos中安装Docker

Docker (二) &#xff1a;Centos中安装Docker 文章目录Docker (二) &#xff1a;Centos中安装DockerCentos安装 Docker准备工作系统要求卸载旧版本安装必要的一些系统工具安装 Docker CE建立 docker 用户组启动 Docker CE,测试 Docker 是否安装正确镜像加速九 Docker 镜像加速器…

HTTP原理

https://www.cnblogs.com/klb561/p/9221754.html转载于:https://www.cnblogs.com/zeenzhou/p/11149931.html

实现动态页面传值 options动态显示数据

href"../css/index.css" rel"stylesheet" type"text/css"> 选择组 未选人员 已选人员 姚明刘翔李小双刘德华黎明张学友王晶张艺谋郑秀文杰克逊 << >> <script language"javascript" type"text…

让webstorm全副武装!

不喜欢说太多废话&#xff0c;直接上攻略大全&#xff0c;下面是使用的版本&#xff1a; 一、改变先从脸开始 1. 更换主题&#xff0c;这个看脸的世界啊&#xff01; file>settings>appearance&behavior>appearance,阿狸现在用的Dracula,经典黑色&#xff0c;这样…

day54 线程队列,协程基础

目录 线程队列QueueLifoQueuePriorityQueue运算符重载协程单线程实现并发处理Greenlet模块Gevent协程的定义注意事项&#xff1a;线程队列 Queue 普通的容器&#xff0c;不具备IPC的能力用法和进程类似&#xff0c;有join方法&#xff0c;原理等同于joinableQueueLifoQueue 后进…

Ubuntu Server 18.04 配置固定IP和DNS以及主机名

Ubuntu Server 18.04 配置固定IP和DNS以及主机名 作者&#xff1a; 霍英俊 huo920live.com 文章目录Ubuntu Server 18.04 配置固定IP和DNS以及主机名修改 hostname配置固定 IP配置固定 DNS方法一方法二修改 hostname 在同一局域网中主机名不应该相同&#xff0c;所以我们需要…

如何使父div的范围包含浮动的子div

方法就是在子div中加入一个特殊的div,再设置其样式<body><div class"father"><div class"son1">Box-1<div> //son div都是设置了float属性的<div class"son2">Box-2<div><div class"son3">…

wechat-php-sdk

wechat-php-sdk 微信公众平台php版开发包 支持消息加解密方式的明文模式、兼容模式、安全模式支持自动接入微信公众平台&#xff08;步骤&#xff09;功能模块 Wechat &#xff08;处理自动接入、获取与回复微信消息&#xff09;(使用说明) 接收普通消息/事件推送被动回复&…