I have a text file with specified structure, namely (for each line): char, space, char, space, double value, endline. For instance
q w 1.23
e r 4.56
t y 7.89
What is the proper way to "extract" those values in Free Pascal?
I have a text file with specified structure, namely (for each line): char, space, char, space, double value, endline. For instance
q w 1.23
e r 4.56
t y 7.89
What is the proper way to "extract" those values in Free Pascal?
FreePascal has the function SScanF in SysUtils (you might know if from other languages..)
I've modified RRUZ's example to show how to use it.
uses SysUtils;
type
TData=object
Val1 ,
Val2 : String;
Val3 : Double;
end;
procedure ProcessFile(aFileName:String);
var
F : Text;
LData : TData;
Line : String;
begin
DecimalSeparator:='.';
AssignFile(F,aFileName);
Reset(F);
while not eof(F) do
begin
ReadLn(F,Line);
SScanf(Line,'%s %s %f',[@LData.Val1,@LData.Val2,@LData.Val3]);
//do something with the data
WriteLn(LData.Val1);
WriteLn(LData.Val2);
WriteLn(LData.Val3);
end;
end;
begin
ProcessFile('C:\Bar\Foo\Data.txt');
Writeln('Press Enter to exit');
Readln;
end.
You can use a TStringList
class to load the file and the DelimitedText
property to split the values on another TStringList and then store the values in a record.
Check this sample
{$mode objfpc}{$H+}
uses
Classes, SysUtils;
{$R *.res}
type
TData=record
Val1: Char;
Val2: Char;
Val3: Double;
end;
procedure ProcessFile;
var
LFile : TStringList;
Line : TStringList;
i : Integer;
LData : TData;
LFormat: TFormatSettings;
begin
//set the propert format for the foat values
LFormat:=DefaultFormatSettings;
LFormat.DecimalSeparator:='.';
LFile:=TStringList.Create;
Line :=TStringList.Create;
try
//load the file
LFile.LoadFromFile('C:\Bar\Foo\Data.txt');
Line.Delimiter:=' ';
for i:=0 to LFile.Count-1 do
begin
//read the line and split the result
Line.DelimitedText:=LFile[i];
//some basic check
if Line.Count <> 3 then raise Exception.Create('Wrong data length');
//you can add additional check here
LData.Val1:=Line[0][3];
LData.Val2:=Line[1][4];
LData.Val3:=StrToFloat(Line[2],LFormat);
//do something with the data
WriteLn(LData.Val1);
WriteLn(LData.Val2);
WriteLn(LData.Val3);
end;
finally
Line.Free;
LFile.Free;
end;
end;
begin
try
ProcessFile;
except on E:Exception do Writeln(E.Classname, ':', E.Message);
end;
Writeln('Press Enter to exit');
Readln;
end.
Let's say that we are interested in reading a file, provided by a command line argument after a switch, containing substitution weights for characters.
program WeightFileRead;
uses SysUtils, StrUtils;
var
MyFile : TextFile;
FirstChar, SecondChar, DummyChar : Char;
Weight : Double;
begin
if GetCmdLineArg ('editweights', StdSwitchChars) = ''
then begin
WriteLn ('Syntax: WeightFileRead -editweights filename'); exit
end;
AssignFile (MyFile, GetCmdLineArg ('editweights', StdSwitchChars));
Reset (MyFile);
try
while not EOF (MyFile) do
begin
ReadLn (MyFile, FirstChar, DummyChar, SecondChar, Weight);
WriteLn ('A: ', FirstChar, '; B: ', SecondChar, '; W: ', Weight:0:1);
end
finally
CloseFile (MyFile)
end
end.
In a more general setting, when the first two entries can be longer strings, we can use ExtractWord
that finds n
th whitespace-separated word in a string, (or ExtractSubstr
that treats several whitespaces together as introducing an empty word), and convert the third one into a number.
program WeightFileRead2;
uses SysUtils, StrUtils;
var
MyFile : TextFile;
FileLine : String;
begin
if GetCmdLineArg ('editweights', StdSwitchChars) = ''
then begin
WriteLn ('Syntax: WeightFileRead -editweights filename'); exit
end;
AssignFile (MyFile, GetCmdLineArg ('editweights', StdSwitchChars));
Reset (MyFile);
try
while not EOF (MyFile) do
begin
ReadLn (MyFile, FileLine);
WriteLn ('A: ', ExtractWord (1, FileLine, [' ']),
'; B: ', ExtractWord (2, FileLine, [' ']),
'; W: ', StrToFloat (ExtractWord (3, FileLine, [' '])):0:1);
end
finally
CloseFile (MyFile)
end
end.
Note I use [' ']
rather than StdWordDelims
, since I don't want . or , to be a word delimiter.