Delphi+Word解决方案参考

      Delphi+Word解决方案参考

     这是我做项目过程中自己做的几个函数,见到大家都在问Word的问题。现在拿出来和大家共享。(希望有朋友可以进一步添加新的功能,或者做成包或者lib等,更方便大家使用。我自己是没有时间啦,呵呵)

使用前,先根据需要建立一个空的WORD文件作为模板,在模板文件中设置好各种格式和文本。另外,其中的PrnWordTable的参数是TDBGridEh类型的控件,取自Ehlib2.6

其中用到的shFileCopy函数(用于复制文件)和guiInfo函数(用于显示消息框)也是自己编写的,代码也附后。

 

示范代码如下:

 

代码完成的功能:

1. 替换打印模板中的“#TITLE#”文本为“示范代码1”

2. 并且将DBGridEh1控件当前显示的内容插入到文档的末尾

3. 在文档末尾插入一个空行

4. 在文档末尾插入新的一行文本

5. 将文档中的空行去掉

 

if PrnWordBegin('C:/打印模板.DOC','C:/目标文件1.DOC') then

begin

PrnWordReplace('#TITLE#','示范代码1');

PrnWordTable(DBGridEh1);

PrnWordInsert('');

PrnWordInsert('这是新的一行文本');

PrnWordReplace('^p^p','^p',true);

PrnWordSave;

end;

 

源代码如下:

 

//Word打印(声明部分)

 

wDoc,wApp:Variant;

function PrnWordBegin(tempDoc,docName:String):boolean;

function PrnWordReplace(docText,newText:String;bSimpleReplace:boolean=false):boolean;

function PrnWordInsert(lineText:String;bNewLine:boolean=true):boolean;overload;

function PrnWordInsert(var imgInsert:TImage;sBookMark:String=''):boolean;overload;

function PrnWordInsert(var chartInsert:TChart;sBookMark:String=''):boolean;overload;

function PrnWordTable(var dbG:TDBGridEh;sBookMark:String=''):boolean;

procedure PrnWordSave;

procedure PrnWordEnd;

 

//Word打印(实现部分)

 

{

功能:基于模板文件tempDoc新建目标文件docName并打开文件

}

function PrnWordBegin(tempDoc,docName:String):boolean;

begin

result:=false;

//复制模版

if tempDoc<>'' then

if not shFileCopy(tempDoc,docName) then exit;

//连接Word

try

wApp:=CreateOleObject('Word.Application');

except

guiInfo('请先安装 Microsoft Word 。');

exit;

end;

try

//打开

if tempDoc='' then

begin

//创建新文档

wDoc:=wApp.document.Add;

wDoc.SaveAs(docName);

end else begin

//打开模版

wDoc:=wApp.document..Open(docName);

end;

except

guiInfo('打开模版失败,请检查模版是否正确。');

wApp.Quit;

exit;

end;

wApp.Visible:=true;

result:=true;

end;

 

{

功能:使用newText替换docText内容

bSimpleReplace:true时仅做简单的替换,false时对新文本进行换行处理

}

function PrnWordReplace(docText,newText:String;bSimpleReplace:boolean=false):boolean;

var i:Integer;

begin

if bSimpleReplace then

begin

//简单处理,直接执行替换操作

try

wApp.Selection.Find.ClearFormatting;

wApp.Selection.Find.Replacement.ClearFormatting;

wApp.Selection.Find.Text := docText;

wApp.Selection.Find.Replacement.Text :=newText;

wApp.Selection.Find.Forward := True;

wApp.Selection.Find.Wrap := wdFindContinue;

wApp.Selection.Find.Format := False;

wApp.Selection.Find.MatchCase := False;

wApp.Selection.Find.MatchWholeWord := true;

wApp.Selection.Find.MatchByte := True;

wApp.Selection.Find.MatchWildcards := False;

wApp.Selection.Find.MatchSoundsLike := False;

wApp.Selection.Find.MatchAllWordForms := False;

wApp.Selection.Find.Execute(Replace:=wdReplaceAll);

result:=true;

except

result:=false;

end;

exit;

end;

//自动分行

reWord.Lines.Clear;

reWord.Lines.Add(newText);

try

//定位到要替换的位置的后面

wApp.Selection.Find.ClearFormatting;

wApp.Selection.Find.Text := docText;

wApp.Selection.Find.Replacement.Text := '';

wApp.Selection.Find.Forward := True;

wApp.Selection.Find.Wrap := wdFindContinue;

wApp.Selection.Find.Format := False;

wApp.Selection.Find.MatchCase := False;

wApp.Selection.Find.MatchWholeWord := False;

wApp.Selection.Find.MatchByte := True;

wApp.Selection.Find.MatchWildcards := False;

wApp.Selection.Find.MatchSoundsLike := False;

wApp.Selection.Find.MatchAllWordForms := False;

wApp.Selection.Find.Execute;

wApp.Selection.MoveRight(wdCharacter,1);

//开始逐行插入

for i:=0 to reWord.Lines.Count-1 Do

begin

//插入当前行

wApp.Selection.InsertAfter(reWord.Lines[i]);

//除最后一行外,自动加入新行

if i<reWord.Lines.Count-1 then

wApp.Selection.InsertAfter(#13);

end;

//删除替换位标

wApp.Selection.Find.ClearFormatting;

wApp.Selection.Find.Replacement.ClearFormatting;

wApp.Selection.Find.Text := docText;

wApp.Selection.Find.Replacement.Text := '';

wApp.Selection.Find.Forward := True;

wApp.Selection.Find.Wrap := wdFindContinue;

wApp.Selection.Find.Format := False;

wApp.Selection.Find.MatchCase := False;

wApp.Selection.Find.MatchWholeWord := true;

wApp.Selection.Find.MatchByte := True;

wApp.Selection.Find.MatchWildcards := False;

wApp.Selection.Find.MatchSoundsLike := False;

wApp.Selection.Find.MatchAllWordForms := False;

wApp.Selection.Find.Execute(Replace:=wdReplaceAll);

result:=true;

except

result:=false;

end;

end;

 

{

功能:打印TDBGridEh当前显示的内容

基于TDBGridEh控件的格式和内容,自动在文档中的sBookMark书签处生成Word表格

目前能够支持单元格对齐、多行标题(两行)、底部合计等特性

sBookMark:Word中要插入表格的书签名称

}

function PrnWordTable(var dbG:TDBGridEh;sBookMark:String=''):boolean;

var iCol,iLine,i,j,k:Integer;

wTable,wRange:Variant;

iRangeEnd:longint;

iGridLine,iTitleLine:Integer;

getTextText:String;getTextDisplay:boolean;

titleList:TStringList;titleSplit,titleCol:Integer;lastTitleSplit,SubTitle:Integer;lastTitle:String;

begin

result:=false;

try

//计算表格的列数(不包括隐藏的列)

iTitleLine:=1; //始终默认为1

iCol:=0;

for i:=0 to dbG.Columns.Count-1 Do

begin

if dbG.Columns[i].Visible then

begin

iCol:=iCol+1;

end;

end;

//计算表格的行数(不包括隐藏的列)

if dbG.DataSource.DataSet.Active then

iLine:=dbG.DataSource.DataSet.RecordCount

else

iLine:=0;

iGridLine:=iLine+iTitleLine+dbG.FooterRowCount;

//定位插入点

if sBookMark='' then

begin

//在文档末尾

iRangeEnd:=wDoc.Range.End-1;

if iRangeEnd<0 then iRangeEnd:=0;

wRange:=wDoc.Range(iRangeEnd,iRangeEnd);

end else begin

//在书签处

wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);

end;

wTable:=wDoc.Tables.Add(wRange,iGridLine,iCol);

wTable.Columns.AutoFit;

//标题行

k:=1;

for j:=1 to dbG.Columns.Count Do

begin

if dbG.Columns[j-1].Visible then

begin

if dbG.UseMultiTitle then

begin

titleList:=strSplit(dbG.Columns[j-1].Title.Caption,'|');

wTable.Cell(1,k).Range.InsertAfter(titleList.Strings[0]);

end else

wTable.Cell(1,k).Range.InsertAfter(dbG.Columns[j-1].Title.Caption);

//设置单元格对齐方式

if dbG.Columns[j-1].Title.Alignment=taCenter then

wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter

else if dbG.Columns[j-1].Title.Alignment=taRightJustify then

wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight

else if dbG.Columns[j-1].Title.Alignment=taLeftJustify then

wTable.Cell(1,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;

k:=k+1;

end;

end;

//填写每一行

if iLine>0 then

begin

dbG.DataSource.dataset.DisableControls;

dbG.DataSource.DataSet.First;

for i:=1 to iLine Do

begin

k:=1;

for j:=1 to dbG.Columns.Count Do

begin

if dbG.Columns[j-1].Visible then

begin

if dbG.Columns[j-1].FieldName<>'' then //避免由于空列而出错

begin

//如果该列有自己的格式化显示函数,则调用显示函数获取显示串

getTextText:='';

if Assigned(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FieldName).OnGetText) then

begin

dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FieldName).OnGetText(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FieldName),getTextText,getTextDisplay);

wTable.Cell(i+iTitleLine,k).Range.InsertAfter(getTextText);

end else begin

//使用数据库内容显示

wTable.Cell(i+iTitleLine,k).Range.InsertAfter(dbG.DataSource.DataSet.FieldByName(dbG.Columns[j-1].FieldName).AsString);

end;

end;

//设置单元格对齐方式

if dbG.Columns[j-1].Alignment=taCenter then

wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter

else if dbG.Columns[j-1].Alignment=taRightJustify then

wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight

else if dbG.Columns[j-1].Alignment=taLeftJustify then

wTable.Cell(i+iTitleLine,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;

k:=k+1;

end;

end;

dbG.DataSource.DataSet.Next;

end;

end;

//结尾行

for i:=1 to dbG.FooterRowCount Do

begin

k:=1;

for j:=1 to dbG.Columns.Count Do

begin

if dbG.Columns[j-1].Visible then

begin

wTable.Cell(iLine+1+i,k).Range.InsertAfter(dbG.GetFootervalue(i-1,dbG.Columns[j-1]));

//设置单元格对齐方式

if dbG.Columns[j-1].Footer.Alignment=taCenter then

wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphCenter

else if dbG.Columns[j-1].Footer.Alignment=taRightJustify then

wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphRight

else if dbG.Columns[j-1].Footer.Alignment=taLeftJustify then

wTable.Cell(iLine+1+i,k).Range.ParagraphFormat.Alignment:=wdAlignParagraphJustify;

k:=k+1;

end;

end;

end;

//处理多行标题

if dbG.UseMultiTitle then

begin

//先分割单元格,再逐个填入第二行

k:=1;

titleCol:=1;

lastTitleSplit:=1;

SubTitle:=0;

lastTitle:='';

for j:=1 to dbG.Columns.Count Do

begin

if dbG.Columns[j-1].Visible then

begin

titleList:=strSplit(dbG.Columns[j-1].Title.Caption,'|');

if titleList.Count>1 then

begin

//处理第二行以上的内容

wTable.Cell(1,k-SubTitle).Range.Cells.Split(titleList.Count,1,false);

for titleSplit:=1 to titleList.Count-1 Do

begin

wTable.Cell(titleSplit+1,titleCol).Range.InsertAfter(titleList.Strings[titleSplit]);

end;

titleCol:=titleCol+1;

//处理第一行合并

if (lastTitleSplit=titleList.Count) and (lastTitle=titleList.Strings[0]) then

begin

//内容相同时,合并单元格

wTable.Cell(1,k-SubTitle).Range.Copy;

wRange:=wDoc.Range(wTable.Cell(1,k-SubTitle-1).Range.Start,wTable.Cell(1,k-SubTitle).Range.End);

wRange.Cells.Merge;

wRange.Paste;

SubTitle:=SubTitle+1;

end;

end;

lastTitle:=titleList.Strings[0];

lastTitleSplit:=titleList.Count;

titleList.Clear;titleList.Free;

k:=k+1;

end;

end;

end;

//自动调整表格

wTable.AutoFitBehavior(1);//根据内容自动调整表格wdAutoFitContent

wTable.AutoFitBehavior(2);//根据窗口自动调整表格wdAutoFitWindow

result:=true;

except

result:=false;

end;

try

dbG.DataSource.dataset.EnableControls;

except

end;

end;

 

{

功能:在Word文件中插入文本(能够自动进行换行处理)

lineText:要插入的文本

bNewLine:true时新起一行,false时在当前行插入

}

function PrnWordInsert(lineText:String;bNewLine:boolean=true):boolean;

var i:Integer;

begin

try

if bNewLine then

wDoc.Range.InsertAfter(#13);

//自动分行

reWord.Lines.Clear;

reWord.Lines.Add(lineText);

//开始逐行插入

for i:=0 to reWord.Lines.Count-1 Do

begin

//插入当前行

wDoc.Range.InsertAfter(reWord.Lines[i]);

//除最后一行外,自动加入新行

if i<reWord.Lines.Count-1 then

wDoc.Range.InsertAfter(#13);

end;

result:=true;

except

result:=false;

end;

end;

 

{

功能:在Word文件的sBookMark书签处插入TImage控件包含的图片

}

function PrnWordInsert(var imgInsert:TImage;sBookMark:String=''):boolean;

var wRange:Variant;iRangeEnd:Integer;

begin

try

if sBookMark='' then

begin

//在文档末尾

iRangeEnd:=wDoc.Range.End-1;

if iRangeEnd<0 then iRangeEnd:=0;

wRange:=wDoc.Range(iRangeEnd,iRangeEnd);

end else begin

//在书签处

wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);

end;

if imgInsert.Picture.Graphic<>nil then

begin

Clipboard.Assign(imgInsert.Picture);

wRange.Paste;

end else begin

wRange.InsertAfter('照片');

end;

result:=true;

except

result:=false;

end;

end;

 

{

功能:在书签sBookMark处插入TChart控件包含的图表

}

function PrnWordInsert(var chartInsert:TChart;sBookMark:String=''):boolean;

var wRange:Variant;iRangeEnd:Integer;

begin

try

if sBookMark='' then

begin

//在文档末尾

iRangeEnd:=wDoc.Range.End-1;

if iRangeEnd<0 then iRangeEnd:=0;

wRange:=wDoc.Range(iRangeEnd,iRangeEnd);

end else begin

//在书签处

wRange:=wDoc.Range.Goto(wdGoToBookmark,,,sBookMark);

end;

chartInsert.CopyToClipboardBitmap;

wRange.Paste;

result:=true;

except

result:=false;

end;

end;

 

{

功能:保存Word文件

}

procedure PrnWordSave;

begin

try

wDoc.Save;

except

end;

end;

 

{

功能:关闭Word文件

}

procedure PrnWordEnd;

begin

try

wDoc.Save;

wDoc.Close;

wApp.Quit;

except

end;

end;

 

附:shFileCopy源代码

 

{

功能:安全的复制文件

srcFile,destFile:源文件和目标文件

bDelDest:如果目标文件已经存在,是否覆盖

返回值:true成功,false失败

}

function shFileCopy(srcFile,destfile&:String;bDelDest:boolean=true):boolean;

begin

result:=false;

if not FileExists(srcFile) then

begin

guiInfo ('源文件不存在,不能复制。'+#10#13+srcFile);

exit;

end;

if srcFile=destFile then

begin

guiInfo ('源文件和目标文件相同,不能复制。');

exit;

end;

if FileExists(destFile) then

begin

if not bDelDest then

begin

guiInfo ('目标文件已经存在,不能复制。'+#10#13+destFile);

exit;

end;

FileSetAttr(destFile,FileGetAttr(destFile) and not $00000001);

if not DeleteFile(PChar(destFile)) then

begin

guiInfo ('目标文件已经存在,并且不能被删除,复制失败。'+#10#13+destFile);

exit;

end;

end;

if not CopyFileTo(srcFile,destFile) then

begin

guiInfo ('发生未知的错误,复制文件失败。');

exit;

end;

//目标文件去掉只读属性

FileSetAttr(destFile,FileGetAttr(destFile) and not $00000001);

result:=true;

end;

 

附:guiInfo源代码

 

{

功能:封装了各种性质的提示框

sMsg:要提示的消息

}

procedure guiInfo(sMsg:String);

begin

MessageDlg(sMsg,mtInformation,[mbOK],0);

end;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
//文件操作部分起 //拷贝一个文件,封装CopyFile procedure FileCopyFile(const sSrcFile, sDstFile: string); //给定路径复制文件到同一目录下 bRecursive:true所有 procedure FileCopyDirectory(sDir, tDir: string; bRecursive: Boolean);overload; //给定路径原样复制文件 ,自编 procedure FileCopyDirectory(sDir, tDir: string);overload; //给定路径原样复制文件 ,用WinAPI ,若原目录下有相同文件则再生成一个 procedure FileCopyDirectory(sDir, tDir:string;AHandle:Thandle);overload; //移动文件夹 procedure FileMoveDirectory(sDir, tDir:string;AHandle:Thandle); //删除给定路径及以下的所有路径和文件 procedure FileDeleteDirectory(sDir: string);overload; //删除给定路径及以下的所有路径和文件 用WinApi procedure FileDeleteDirectory(AHandle: THandle;const ADirName: string);overload; //删除给定路径及以下的所有路径和文件 到回收站 procedure FileDeleteDirectoryToCycle(AHandle: THandle;const ADirName: string); //取得指定文件的大小 function FileGetFileSize(const Filename: string): DWORD; //在Path下取得唯一FilenameX文件 function FileGetUniqueFileName(const Path: string; Filename: string): string; //取得临时文件 function FileGetTemporaryFileName: string; //取得系统路径 function PathGetSystemPath: string; //取得Windows路径 function PathGetWindowsPath: string; //给定文件名取得在系统目录下的路径,复制时用 function PathSystemDirFile(const Filename: string): string; //给定文件名取得在Windows目录下的路径,复制时用 function PathWindowsDirFile(const Filename: string): string; //给定文件名取得在系统盘下的路径,复制时用 function PathSystemDriveFile(const Filename: string): string; //路径最后有'/'则去'/' function PathWithoutSlash(const Path: string): string; //路径最后没有'/'则加'/' function PathWithSlash(const Path: string): string; //取得两路径的不同部分,条件是前半部分相同 function PathRelativePath(BaseDir, FilePath: string): string; //取得去掉属性的路径,文件名也作为DIR function PathExtractFileNameNoExt(Filename: string): string; //判断两路径是否相等 function PathComparePath(const Path1, Path2: string): Boolean; //取得给定路径的父路径 function PathParentDirectory(Path: string): string; //分割路径,Result=根(如d:)sPath = 除根外的其他部分 function PathGetRootDir(var sPath: string): string; //取得路径最后部分和其他部分 如d:\aa\aa result:=aa sPath:=d:\aa\ function PathGetLeafDir(var sPath: string): string; //取得当前应用程序的路径 function PathExeDir(FileName: string = ''): string; //文件操作部分止 //系统处理起 //提示窗口 procedure MsgBox(const Msg: string); //错误显示窗口 procedure MsgErrBox(const Msg: string); //询问窗口 带'是','否'按钮 function MsgYesNoBox(const Msg: string): Boolean; //询问窗口 带'是','否,'取消'按钮//返回值smbYes,smbNo,smbCancel function MsgYesNoCancelBox(const Msg: string): Integer; //使鼠标变忙和恢复正常 procedure DoBusy(Busy: Boolean); //显示错误信息 procedure ShowLastError(const Msg: string = 'API Error'); //发出错误信息 procedure RaiseLastError(const Msg: string = 'API Error'); //释放Strings连接的相关资源 procedure FreeStringsObjects(SL: TStrings); //系统处理止 //时间处理起 //整数到时间 function TimeT_To_DateTime(TimeT: Longint): TDateTime; //转化为秒 function TimeToSecond(const H, M, S: Integer): Integer; //秒转化 procedure TimeSecondToTime(const secs: Integer; var H, M, S: Word); //秒转化 function TimeSecondToTimeStr(secs: Integer): string; //时间处理止 //控件处理起 //设置控件是否能使用 procedure ConEnableControl(AControl: TControl; Enable: Boolean); //设置控件是否能使用,包子控件 procedure ConEnableChildControls(AControl: TControl; Enable: Boolean); procedure ConEnableClassControl(AControl: TControl; Enable: Boolean; ControlClass: TControlClass); procedure ConFree(aCon: TWinControl);//释放aCon上的控件 //从文件本中导入,类似LoadfromFile procedure ConLoadTreeViewFromTextFile(Nodes: TTreeNodes; Filename: string); //存为文本,类似SaveToFile procedure ConSaveTreeViewToTextFile(Nodes: TTreeNodes; Filename: string); //在控件上写文本 procedure ConWriteText(aContr: TControl;sText: string); //控件处理止 //字符串处理起 //取以Delimiters分隔的字符串 bTrail如果为True则把第index个后的也取出来 function StrGetToken(const S: string; index: Integer; bTrail: Boolean = False; Delimiters: TSysCharSet = DEFAULT_DELIMITERS): string; //取以Delimiters分隔的字符串的个数 function StrCountWords(S: string; Delimiters: TSysCharSet = DEFAULT_DELIMITERS): Integer; //用NewToken替换S中所有Token bCaseSensitive:=true大小写敏感 function StrReplaceString(var S: string; const Token, NewToken: string; bCaseSensitive: Boolean): Boolean; //从第Index个起以Substr替换Count个字符 procedure StrSimple_ReplaceString(var S: string; const Substr: string; index, Count: Integer); //去掉S中的回车返行符 procedure StrTruncateCRLF(var S: string); //判定S是否以回车返行符结束 function StrIsContainingCRLF(const S: string): Boolean; //把SL中的各项数据转化为以Delimiter分隔的Str function StrCompositeStrings(SL: TStrings; const Delimiter: string): string; //封装TStrings的LoadFromFile function StrSafeLoadStrings(SL: TStrings; const Filename: string): Boolean; //封装TStrings的SaveToFile procedure StrSafeSaveStrings(SL: TStrings; const Filename: string); //字符串处理止 //字体处理起 procedure StringToFont(sFont: string; Font: TFont; bIncludeColor: Boolean = True); function FontToString(Font: TFont; bIncludeColor: Boolean = True): string; //字体处理止 //网络起 //判定是否在线 function NetJudgeOnline:boolean; //得到本机的局域网Ip地址 Function NetGetLocalIp(var LocalIp:string): Boolean; //通过Ip返回机器名 Function NetGetNameByIPAddr(IPAddr: string; var MacName: string): Boolean ; //获取网络中SQLServer列表 Function NetGetSQLServerList(var List: Tstringlist): Boolean; //获取网络中的所有网络类型 Function NetGetNetList(var List: Tstringlist): Boolean; //获取网络中的工作组 Function NetGetGroupList(var List: TStringList): Boolean; //获取工作组中所有计算机 Function NetGetUsers(GroupName: string; var List: TStringList): Boolean; //获取网络中的资源 Function NetGetUserResource(IpAddr: string; var List: TStringList): Boolean; //映射网络驱动器 Function NetAddConnection(NetPath: Pchar; PassWord: Pchar;LocalPath: Pchar): Boolean; //检测网络状态 Function NetCheckNet(IpAddr:string): Boolean; //检测机器是否登入网络 Function NetCheckMacAttachNet: Boolean; //判断Ip协议有没有安装 这个函数有问题 Function NetIsIPInstalled : boolean; //检测机器是否上网 Function NetInternetConnected: Boolean; //网络止 //窗口起 function FormCreateProcessFrm(MsgTitle: string):TForm; //窗口止 //EMail起 function CheckMailAddress(Text: string): boolean; //EMail止

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值