首页  编辑  

在Delphi编程中使用C语言代码

Tags: /超级猛料/Language.Object Pascal/在Delphi编程中使用C语言代码/   Date Created:
在Delphi编程中使用C语言代码
 
  陈经韬 
    Windows下编程的工具有很多,例如VB,Delphi,VC等等.我在这里不想讨论"它们的具体哪个更好一点"这种幼稚的问题.玩过DOS程序设计的人都知道,DOS下很多语言的实质核心还是调用系统提供的汇编中断函数.到了Windows下,它就变成了我们常说的API了.而在Windows下写程序很多时候都是调用API,语言,只不过是一个表达工具而已.
    我现在已经参加工作大约有半年左右,我们公司是用Borland公司的Delphi作为主开发工具.本着未偏袒任何一个工具的立场,我说句公道话:Delphi是目前Win32下开发程序的最快速,最有效率的工具.
    Delphi适合用来开发应用程序,但是有时侯一些底层的东西可以直接使用C语言来开发.我在公司经常开发跟硬件相关的项目,而很多硬件的SDK包是用C来写的.这个时候我一般把它们转换成Delphi(PASCAL)语法的代码.下面谈一下我的个人粗浅经验.因为当时学校教的是Pascal语言,所以我对C语言并不是太熟手.下面的观点或者代码如有错漏之处希望高手们放小弟一马:) 
一:将C语言的程序编译成DLL供Delphi调用.这种方法过于简单,而且需要额外带一个DLL文件,所以不在本文的讨论范围之内.

二:直接转换C语言代码到DELPHI代码


   C语言的函数格式与Delphi不同,它们是函数返回类型在前,函数声明在后.对于没有任何返回类型的函数则定义为VOID类型.
   例如:Delphi中函数function MyFunction:(intIN:integer):Bool;相应的C语言代码就变成Bool MyFunction(int intIN);又例如procedure MyProcedure;====>void MyProcedure;采用这种方法,一般要求对C语言比较熟悉.我一般是采用这种方法.下面是我收集整理的自己常用的Delphi与C之间的类型对应表.其中左边是C类型,右边是对应的Delphi类型:

ABC -> TABC 
ACCEL -> TAccel 
ATOM -> TAtom 
BITMAP -> TBitMap 
BITMAPCOREHEADER -> TBitmapCoreHeader 
BITMAPCOREINFO -> TBitmapCoreInfo 
BITMAPFILEHEADER -> TBitmapFileHeader 
BITMAPINFO -> TBitmapInfo 
BITMAPINFOHEADER -> TBitmapInfoHeader 
BOOL -> Bool 
CBT_CREATEWND -> TCBT_CreateWnd 
CBTACTIVATESTRUCT -> TCBTActivateStruct 
CHAR -> Char 
CHAR* -> PChar 
CLIENTCREATESTRUCT -> TClientCreateStruct 
COLORREF -> TColorRef 
COMPAREITEMSTRUCT -> TCompareItemStruct 
COMSTAT -> TComStat 
CREATESTRUCT -> TCreateStruct 
CTLINFO -> TCtlInfo 
CTLSTYLE -> TCtlStyle 
CTLtype -> TCtltype 
DCB -> TDCB 
DDEAACK -> TDDEAck 
DDEADVISE -> TDDEAdvise 
DDEDATA -> TDDEData 
DDEPOKE -> TDDEPoke 
DEBUGHOOKINFO -> TDebugHookInfo 
DELETEITEMSTRUCT -> TDeleteItemStruct 
DEVMODE -> TDevMode 
DOUBLE -> Double 
DRAWITEMSTRUCT -> TDrawItemStruct 
DWORD -> LongInt 
ENUMLOGFONT -> TEnumLogFont 
EVENTMSG -> TEventMsg 
FARPROC -> TFarProc 
FIXED -> TFixed 
FLOAT -> Single 
GLYPHMETRICS -> TGlyphMetrics 
HANDLE -> THandle 
HANDLETABLE -> THandleTable 
HARDWAREHOOKSTRUCT -> THardwareHookStruct 
HELPWININFO -> THelpWinInfo 
INT -> Integer 
KERNINGPAIR -> TKerningPair 
LOGBRUSH -> TLogBrush 
LOGFONT -> TLogFont 
LOGPALETTE -> TLogPalette 
LOGPEN -> TLogPen 
LONG -> LongInt 
LONG DOUBLE -> Extended 
LONG INT -> LongInt 
LPSTR -> PChar 
LPWSTR -> PWideChar 
MAT2 -> TMat2 
MDICREATESTRUCT -> TMDICreateStruct 
MEASUREITEMSTRUCT -> TMeasureItemStruct 
MENUITEMTEMPLATE -> TMenuItemTemplate 
MENUITEMTEMPLATEHEADER -> TMenuItemTemplateHeader
METAFILEPICT -> TMetaFilePict 
METAHEADER -> TMetaHeader 
METARECORD -> TMetaRecord 
MINMAXINFO -> TMinMaxInfo 
MOUSEHOOKSTRUCT -> TMouseHookStruct 
MSG -> TMsg 
MULTIKEYHELP -> TMultiKeyHelp 
NCCALCSIZE_PARAMS -> TNCCalcSize_Params 
NEWTEXTMETRIC -> TNewTextMetric 
OFSTRUCT -> TOFStruct 
OUTLINETEXTMETRIC -> TOutlineTextMetric 
PAINTSTRUCT -> TPaintStruct 
PALETTEENTRY -> TPaletteEntry 
PANOSE -> TPanose 
PATTERN -> TPattern 
POINTFX -> TPointFX 
PSTR -> PChar 
PWSTR -> PWideChar 
RASTERIZER_STATUS -> TRasterizer_Status 
RGBQUAD -> TRGBQuad 
RGBTRIPLE -> TRGBTriple 
SEGINFO -> TSegInfo 
SHORT -> SmallInt 
SHORT INT -> SmallInt 
SIZE -> TSize 
TEXTMETRIC -> TTextMetric 
TPOINT -> TPoint 
TRECT -> TRect 
TTPOLYCURVE -> TTTPolyCurve 
TTPOLYGONHEADER -> TPolygonHeader 
UINT -> Word 
UNSIGNED -> Word 
UNSIGNED CHAR -> Byte 
UNSIGNED INT -> Word 
UNSIGNED LONG -> LongInt(DWORD) 
UNSIGNED LONG INT -> LongInt 
UNSIGNED SHORT -> Word 
UNSIGNED SHORT INT -> Word 
VOID* -> Pointer 
WINDOWPLACEMENT -> TWindowPlacement 
WINDOWPOS -> TWindowPos 
WNDCLASS -> TWndClass 
WORD -> Word 

三:在Delphi中直接链接C语言的OBJ文件.
   这种方法的好处在于最终EXE不用带任何外部文件.也不用对C语言过于熟悉.
   我们都知道,代码在编译成可执行文件(或DLL,OCX文件,下同)之前,都必须得先生成OBJ文件(DELPHI一般是DCU文件,但也可以通过编辑编译选项生成OBJ文件),然后把OBJ文件和资源文件(*.RES)链接成最终的可执行文件.利用这个方法,我们可以直接把OBJ文件链接到我们的程序里面.
   不过需要注意的是,编译器不同,生成的OBJ文件也不一样.Microsoft的编译器生成的OBJ文件是COFF格式,而Borland的C++Builder生成的是OMF格式.因为我们需要在Delphi中链接,所以必须使用CBC,或者Borland官方站点带的免费编译工具.下面我们通过一个简单的例子来说明具体操作步骤:
   这个例子是简单的提供一个函数,用来判断一个文件是否为Dat格式的VCD文件.头文件声明如下:


/*
文件名称:DatFormat.h
*/
#ifndef DatFormat_H
#define DatFormat_H
#include <windows.h>
#pragma pack(push, 1)//这个与下面的配对,一般用到记录类型的时候需要定义,这里实际不用


#ifdef __cplusplus
extern "C" {
#endif

extern BOOL CheckIsDatFile(const char * FileName,BOOL *IsDatFile);

#ifdef __cplusplus
}
#endif


#pragma pack(pop)

#endif // DatFormat_H

  具体实现代码DatFormat.c如下:

#include "DatFormat.h"
BOOL CheckIsDatFile(const char * FileName,BOOL *IsDatFile)
/*
函数说明:该函数用于判断一个文件是否为Dat文件(即VCD文件)格式.
参数:
IN:
FileName:欲判断的文件名称
IN,OUT:
IsDatFile:是否为Dat格式文件
OUT:
读文件失败返回FALSE,否则返回TRUE.
------------------------------------
作者:陈经韬.2004,01,17. http://www.138soft.com,lovejingtao@21cn.com
*/
{
HANDLE hFile;
DWORD dwBytesRead;
BOOL re;
char MyBuf[4];

*IsDatFile=FALSE;

//建立读文件句柄
hFile = CreateFile(FileName,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
0,
0);


if (hFile == INVALID_HANDLE_VALUE) return FALSE;

//读文件
re = ReadFile(hFile,
&MyBuf,
4,
&dwBytesRead,
NULL);

if (dwBytesRead!=4)
{
CloseHandle(hFile);
return FALSE;
}

//读文件失败的时候
if (re!=TRUE)
{
CloseHandle(hFile);
return FALSE;
}
CloseHandle(hFile);
*IsDatFile=(MyBuf[0]=='R' && MyBuf[1]=='I' && MyBuf[2]=='F' && MyBuf[3]=='F');

return(TRUE);
}

  运行CBC,新建一个工程,然后把DatFormat.c添加到工程里面,编译整个工程,将得到我们需要的OBJ文件:DatFormat.OBJ.然后我们关闭CBC即可,因为下面不再需要用到它了.
  运行Delphi,新建一个工程并保存.然后把DatFormat.OBJ拷贝到它的目录之下.在单元的implementation下面添加如下代码:

{$LINK 'DatFormat.obj'} //链接外部OBJ文件
function _CheckIsDatFile(const FileName:Pchar;IsDatFile:PBool):Bool;cdecl;external;//定义函数.其中cdecl进栈方式说明采用C语言格式传递参数.external说明是个外部声明函数.

  注意函数声明的原形与C定义的不一样.必须在前面添加一个下划线.原因是因为编译器的链接符号中.C与C++是不一样的.因为这个不是本文重点,所以这里不作讨论.请感兴趣的朋友自行参阅相关资料.
  然后我们写如下代码调用此函数:

procedure TFrmMain.Button1Click(Sender: TObject);
var
IsDatFile:Bool;
begin
if OpenDialog1.Execute then
if _CheckIsDatFile(Pchar(OpenDialog1.FileName),@IsDatFile) then
if IsDatFile then ShowMessage('恭喜!该文件是一个Dat格式的视频文件!')
else ShowMessage('不好意思,该文件不是一个Dat格式的视频文件!')
else ShowMessage('读文件错误!');
end;

  编译这个程序,将得到一个干净的可执行EXE文件了.

四:C++Builder中使用Delphi单元

  这个实际是题外话了,不过这里还是提一提:假设我们有一个获取BIOS密码的Delphi单元

unit AwardBiosPas;

{=======================================================
项目: 在Delphi编程中使用C语言代码- 演示程序
模块: 获取BIOS密码单元
描述:
版本:
日期: 2004-01-17
作者: 陈经韬.lovejingtao@21cn.com,http://www.138soft.com
更新: 2004-01-17
=======================================================}

interface
uses
windows, SysUtils;

function My_GetBiosPassword: string;

implementation

function CalcPossiblePassword(PasswordValue: WORD): string;
var
I: BYTE;
C: CHAR;
S: string[8];

begin
I := 0;
while PasswordValue <> 0 do
begin
Inc(I);
if $263 > PasswordValue then
begin
if $80 > PasswordValue then
S[I] := CHAR(PasswordValue)
else if $B0 > PasswordValue then
S[I] := CHAR(PasswordValue and $77)
else if $11D > PasswordValue then
S[I] := CHAR($30 or (PasswordValue and $0F))
else if $114 > PasswordValue then
begin
S[I] := CHAR($64 or (PasswordValue and $0F));
if '0' > S[I] then
S[I] := CHAR(BYTE(S[I]) + 8);
end
else if $1C2 > PasswordValue then
S[I] := CHAR($70 or (PasswordValue and $03))
else if $1E4 > PasswordValue then
S[I] := CHAR($30 or (PasswordValue and $03))
else
begin
S[I] := CHAR($70 or (PasswordValue and $0F));
if 'z' < S[I] then
S[I] := CHAR(BYTE(S[I]) - 8);
end;
end
else
S[I] := CHAR($30 or (PasswordValue and $3));
PasswordValue := (PasswordValue - BYTE(S[I])) shr 2;
end;

S[0] := CHAR(I);
PasswordValue := I shr 1;
while PasswordValue < I do
begin {this is to do because award starts calculating with the last letter}

C := S[BYTE(S[0]) - I + 1];
S[BYTE(S[0]) - I + 1] := S[I];
S[I] := C;
Dec(I);
end;
CalcPossiblePassword := S;
end;

function readcmos(off: byte): byte;
var
value: byte;
begin
asm
xor ax, ax
mov al, off
out 70h, al
in al, 71h
mov value, al
end;
readcmos := value;
end;

function My_GetBiosPassword: string;
var
superpw, userpw: word;
begin
if Win32Platform <> VER_PLATFORM_WIN32_NT then //不是NT
begin
pchar(@superpw)[0] := char(readcmos($1C));
pchar(@superpw)[1] := char(readcmos($1D));
pchar(@userpw)[0] := char(readcmos($64));
pchar(@userpw)[1] := char(readcmos($65));
Result:= ('************BIOS密码**********************')+#13+'超级用户密码为:' + CalcPossiblePassword(superpw) + #13 + '用户密码为:' + CalcPossiblePassword(userpw);
end
else
Result := '用户系统为NT,无法获取BIOS密码!';
end;
end.


  如何直接在CBC中使用它呢?新建一个CBC工程,然后把这个单元加到项目里面去.具体操作为:Add to Project--->文件类型:pascal unit(*.pas),然后Build Demo1.这个时候将在AwardBiosPas.pas的同目录下生成一个AwardBiosPas.hpp文件.把它引用到我们的需要调用的单元.然后直接调用即可:
void __fastcall TFrmMain::Button1Click(TObject *Sender)
{
ShowMessage(My_GetBiosPassword());
}

五:其它方法.当然可以用RES将C语言生成的二进制文件,但这个方法与第一种方法差不多.优点是不怕文件丢失.缺点是很容易被别人直接用资源修改工具打开修改.这个时候可以使用笔者写的自制编程序工具PasAnywhere.不过这已经是另外一个话题了.