delphi7调用delphi2009生成的dll文件会出现乱码问题。是unicode的原因
终于找到了转换函数
function AnsiToUnicode(s:string):WideString;
var
lpWideChar:PWideChar;
len:Integer;
begin
len := ( Length(s) + 1 ) * 2;
GetMem(lpWideChar, len);
ZeroMemory(lpWideChar, len);
MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,
PChar(s), Length(s),lpWideChar, Len);
Result := lpWideChar;
FreeMem(lpWideChar);
end;
function UnicodeToAnsi(s:WideString):string;
var
lpChar:PChar;
len:integer;
begin
len := Length(s) * 2;
GetMem(lpChar,len);
ZeroMemory(lpChar, len);
WideCharToMultiByte(CP_ACP, WC_COMPOSITECHECK, PWideChar(s),
Length(s),lpChar,Len, nil, nil );
Result := lpChar;
FreeMem(lpChar);
end;
==============================================
Delphi读写UTF-8、Unicode格式文本文件
在研究PvPGN时发现conf配置文件一些为UTF-8和Unicode格式,这样便可良好的支持多语言,从网上查阅资料后,将读写UTF-8、Unicode文件写了几个最精简的函数,更新后加了是否写文件头的功能,以适应更多需要,注意函数未加防错保护。
参数说明:f文件名、s写入或读取的文件内容、hs文件头、b是否读写文件头。
UTF-8文件写入函数
程序代码 程序代码
procedure SaveUTF(f:string;s:string;b:boolean=true);
var
ms:TMemoryStream;
hs:String;
begin
if s='' then exit;
ms:=TMemoryStream.Create;
if b then begin
hs:=#$EF#$BB#$BF;
ms.Write(hs[1],3);
end;
s:=AnsiToUtf8(s);
ms.Write(s[1],Length(s));
ms.Position:=0;
ms.SaveToFile(f);
ms.Free;
end;
UtF-8文件读取函数
程序代码 程序代码
function LoadUTF(f:string;b:boolean=true):string;
var
ms:TMemoryStream;
s,hs:string;
begin
Result:='';
if not FileExists(f) then exit;
ms:=TMemoryStream.Create;
ms.LoadFromFile(f);
if b then begin
SetLength(hs,3);
ms.Read(hs[1],3);
if hs<>#$EF#$BB#$BF then begin ms.Free; exit; end;
SetLength(s,ms.Size-3);
ms.Read(s[1],ms.Size-3);
end else begin
SetLength(s,ms.Size);
ms.Read(s[1],ms.Size);
end;
Result:=Utf8ToAnsi(s);
ms.Free;
end;
Unicode文件写入函数
程序代码 程序代码
procedure SaveUnicode(f:string;s:string;b:boolean=true);
var
ms:TMemoryStream;
hs:string;
ws:WideString;
begin
if s='' then exit;
ms:=TMemoryStream.Create;
if b then begin
hs:=#$FF#$FE;
ms.Write(hs[1],2);
end;
ws:=WideString(s);
ms.Write(ws[1],Length(ws)*2);
ms.Position:=0;
ms.SaveToFile(f);
ms.Free;
end;
Unicode文件读取函数
程序代码 程序代码(宽字符Unicode用这个函数,b赋值false)
function LoadUnicode(f:string;b:boolean=true):string;
var
ms:TMemoryStream;
hs:String;
ws:WideString;
begin
Result:='';
if not FileExists(f) then exit;
ms:=TMemoryStream.Create;
ms.LoadFromFile(f);
if b then begin
SetLength(hs,2);
ms.Read(hs[1],2);
if hs<>#$FF#$FE then begin ms.Free; exit; end;
SetLength(ws,(ms.Size-2) div 2);
ms.Read(ws[1],ms.Size-2);
end else begin
SetLength(ws,ms.Size div 2);
ms.Read(ws[1],ms.Size);
end;
Result:=AnsiString(ws);
ms.Free;
end;