[fpc-pascal] Feature announcement: Generic functions, procedures and methods
Sven Barth
pascaldragon at googlemail.com
Sat Nov 21 18:32:32 CET 2015
Hello together!
I'm pleased to finally announce the addition of generic functions,
procedures and methods (collectively called "routines") to Free Pascal
which allows writing type safe methods that can be used for multiple types.
Syntax:
The syntax for declaring a generic routine is similar to declaring a
normal routine and also includes support for class methods:
generic [class] (procedure|function)
IDENTIFIER<TYPEARGLIST>[(PARAMETERLIST)][: RESULTTYPE]; MODIFIERS;
For the TYPEARGLIST the same rules apply as for generic types. Type
parameters declared in the TYPEARGLIST might be used in the
PARAMETERLIST, the RESULTTYPE and of course the body of the routine.
Generic routines can be overloaded both by TYPEARGLIST and PARAMETERLIST.
To call a generic routine you use the following syntax:
specialize IDENTIFIER<TYPELIST>[(PARAMETERS)]
For the TYPELIST the same rules apply as for specializing a generic type.
If the routine is part of a type or variable or you're using a unit
name to distinguish a generic routine you need to put that before the
"specialize":
TYPENAME.specialize IDENTIFIER<TYPELIST>[(PARAMETERS)]
VARIABLE.specialize IDENTIFIER<TYPELIST>[(PARAMETERS)]
UNITNAME.specialize IDENTIFIER<TYPELIST>[(PARAMETERS)]
Calls to generic routines are normal factors so they can be used as such
as the following example shows:
=== example begin ===
{$mode objfpc}
generic function Add<T>(aLeft, aRight: T): T;
begin
Result := aLeft + aRight;
end;
begin
Writeln(specialize Add<String>('Generic ', 'routines') + specialize
Add<String>(' with ', 'Free Pascal'));
end.
=== example end ===
Delphi compatibility:
Of course this future is also implemented in a Delphi-compatible way for
the Delphi modes. There the syntax for declaring a generic function is
like this:
[class] (procedure|function) IDENTIFIER<TYPELIST>[(PARAMETERLIST)][:
RETURNTYPE]; MODIFIERS;
So merely the "generic" keyword is missing. This is analogous when
calling a generic routine:
IDENTIFIER<TYPELIST>[(PARAMETERS)]
Because of the missing "specialize" keyword that mark specializations
complex expressions *do not* work in mode Delphi yet. So assignments of
function results are okay, but other than that you'll likely encounter
compiler errors.
Please note that unlike Delphi we *do* support global generic
functions/procedures even in Delphi mode.
Limitations/ToDos:
The feature is not yet finished and there are some limitations that yet
need to be overcome or parts that simply don't work yet, this includes,
but is not necessarily limited to:
- support for complex expressions in Delphi modes (applies to type
specializations as well)
- support for pointers to generic routines (this will currently result
in errors at the best case and internal errors or exceptions at the worst)
- support for the return value in modes without support for "Result"
- support for nested generics, most importantly generic methods inside
generic classes
So please test and report any bugs in the bug tracker. Questions can of
course be asked here on the mailing list (no, this feature won't be part
of FPC 3.0.0).
Regards,
Sven
More information about the fpc-pascal
mailing list