a:integer;
d:double;
begin
...
d:=a; //Implicit
d:=double(a); //Explicit
But! I am still thinking about the syntax for declaring new implicit or explicit operators: Delphi or Free Pascal like. :)
среда, 22 декабря 2010 г.
YAR: Enhance syntax of [ ] array operator
Now support [.. , ..]
A sample
integerarrarr:Myrecord[6][3][5][3][3][6];
integerarrarr[1,1,2][1][1,2].c[1]:=abc().a;
A sample
integerarrarr:Myrecord[6][3][5][3][3][6];
integerarrarr[1,1,2][1][1,2].c[1]:=abc().a;
вторник, 7 декабря 2010 г.
YAR and static arrays
Working on support static arrays in YAR.
The shortest sample
program Helloworld;
integerarr:integer[6];
begin
integerarr[2]:=1;
end
=>
.assembly extern mscorlib { auto }
.assembly project {}
.field public static int32* integerarr
.method static public void main() cil managed
{
.entrypoint
ldc.i4 28
localloc
stsfld int32* integerarr
ldsfld int32* integerarr
ldc.i4 2
ldc.i4 4
mul
add
ldc.i4 1
stind.i4
ret
}
The shortest sample
program Helloworld;
integerarr:integer[6];
begin
integerarr[2]:=1;
end
=>
.assembly extern mscorlib { auto }
.assembly project {}
.field public static int32* integerarr
.method static public void main() cil managed
{
.entrypoint
ldc.i4 28
localloc
stsfld int32* integerarr
ldsfld int32* integerarr
ldc.i4 2
ldc.i4 4
mul
add
ldc.i4 1
stind.i4
ret
}
воскресенье, 5 декабря 2010 г.
YAR Update: Importing external function
Yar supports now for
import external functions
sample
program Helloworld;
function importedfunc(var a:integer):integer;external library=mydll name=importfunc;
GField:integer;
begin
GField:=1;
write importedfunc(GField);
end
=>
.assembly extern mscorlib { auto }
.assembly project {}
.field public static int32 GField
.method static public void main() cil managed
{
.entrypoint
ldc.i4 1
stsfld int32 GField
ldsflda int32 GField
call int32 importedfunc(int32& a)
call void [mscorlib]System.Console::Write(int32)
ret
}
.method public static pinvokeimpl("mydll.dll" as "importfunc" winapi)
int32 importedfunc(int32& a) cil managed preservesig
{}
import external functions
sample
program Helloworld;
function importedfunc(var a:integer):integer;external library=mydll name=importfunc;
GField:integer;
begin
GField:=1;
write importedfunc(GField);
end
=>
.assembly extern mscorlib { auto }
.assembly project {}
.field public static int32 GField
.method static public void main() cil managed
{
.entrypoint
ldc.i4 1
stsfld int32 GField
ldsflda int32 GField
call int32 importedfunc(int32& a)
call void [mscorlib]System.Console::Write(int32)
ret
}
.method public static pinvokeimpl("mydll.dll" as "importfunc" winapi)
int32 importedfunc(int32& a) cil managed preservesig
{}
пятница, 3 декабря 2010 г.
YAR: records with methods
Syntax:
program Helloworld;
Myrecord=record
c:integer;
function abc(a:integer):integer;
begin
self.c:=a;
result:=self.c;
end;
end;
mr:Myrecord;
function Func(var a:Myrecord):integer;
begin
a.c:=4;
write mr.c;
a.abc(5);
write a.c;
result:=6;
end;
begin
Func(mr);
end
.NET BackEnd:
.assembly extern mscorlib { auto }
.assembly project {}
.class public sealed Myrecord extends [mscorlib]System.ValueType
{
.field public static int32 c
.method public instance int32 abc(int32 a) cil managed
{
ldarg.0
ldarg a
stfld int32 .this::c
ldarg.0
ldfld int32 .this::c
ret
}
}
.field public static valuetype Myrecord mr
.method static public void main() cil managed
{
.entrypoint
ldsflda valuetype Myrecord mr
call int32 Func( valuetype Myrecord& a)
pop
ret
}
.method static public int32 Func( valuetype Myrecord& a) cil managed
{
ldarg a
ldc.i4 4
stfld int32 Myrecord::c
ldsflda valuetype Myrecord mr
ldfld int32 Myrecord::c
call void [mscorlib]System.Console::Write(int32)
ldarg a
ldc.i4 5
call instance int32 Myrecord::abc(int32 a)
pop
ldarg a
ldfld int32 Myrecord::c
call void [mscorlib]System.Console::Write(int32)
ldc.i4 6
ret
}
program Helloworld;
Myrecord=record
c:integer;
function abc(a:integer):integer;
begin
self.c:=a;
result:=self.c;
end;
end;
mr:Myrecord;
function Func(var a:Myrecord):integer;
begin
a.c:=4;
write mr.c;
a.abc(5);
write a.c;
result:=6;
end;
begin
Func(mr);
end
.NET BackEnd:
.assembly extern mscorlib { auto }
.assembly project {}
.class public sealed Myrecord extends [mscorlib]System.ValueType
{
.field public static int32 c
.method public instance int32 abc(int32 a) cil managed
{
ldarg.0
ldarg a
stfld int32 .this::c
ldarg.0
ldfld int32 .this::c
ret
}
}
.field public static valuetype Myrecord mr
.method static public void main() cil managed
{
.entrypoint
ldsflda valuetype Myrecord mr
call int32 Func( valuetype Myrecord& a)
pop
ret
}
.method static public int32 Func( valuetype Myrecord& a) cil managed
{
ldarg a
ldc.i4 4
stfld int32 Myrecord::c
ldsflda valuetype Myrecord mr
ldfld int32 Myrecord::c
call void [mscorlib]System.Console::Write(int32)
ldarg a
ldc.i4 5
call instance int32 Myrecord::abc(int32 a)
pop
ldarg a
ldfld int32 Myrecord::c
call void [mscorlib]System.Console::Write(int32)
ldc.i4 6
ret
}
понедельник, 8 ноября 2010 г.
Added @ and ^ operators
I have just added.
begin
pinteger:=@mr.c;
a:=pinteger^;
mr.c:=@a^;
end
=>
.entrypoint
ldsflda valuetype Myrecord mr
ldflda int32 Myrecord::c
stsfld int32* pinteger
ldsfld int32* pinteger
ldobj int32
stsfld int32 a
ldsflda valuetype Myrecord mr
ldsflda int32 a
ldobj int32
stfld int32 Myrecord::c
ret
begin
pinteger:=@mr.c;
a:=pinteger^;
mr.c:=@a^;
end
=>
.entrypoint
ldsflda valuetype Myrecord mr
ldflda int32 Myrecord::c
stsfld int32* pinteger
ldsfld int32* pinteger
ldobj int32
stsfld int32 a
ldsflda valuetype Myrecord mr
ldsflda int32 a
ldobj int32
stfld int32 Myrecord::c
ret
среда, 3 ноября 2010 г.
Test YAR Type inference
Todays working sample
function abc < T,U > (a:T):^U;
begin
end
pinteger:^integer;
d:double;
pinteger:=abc(d);
=> T=double
=> U=integer
So generic operators like @, [], sizeof .. to come very soon.
function abc < T,U > (a:T):^U;
begin
end
pinteger:^integer;
d:double;
pinteger:=abc(d);
=> T=double
=> U=integer
So generic operators like @, [], sizeof .. to come very soon.
вторник, 2 ноября 2010 г.
YAR and Type inference
Working on support Type Inference in YAR.
A sample:
mr:Myrecord;
a,b:integer;
d:double;
function GenericFunc < T, U , Z> (var a:T;b:U):Z;
begin
end;
..
begin
GenericFunc(a,a):=d; //I know this function result is not a lvalue.
//I relax the assign op type checking rules
//to test of infering of result type
A sample:
mr:Myrecord;
a,b:integer;
d:double;
function GenericFunc < T, U , Z> (var a:T;b:U):Z;
begin
end;
..
begin
GenericFunc(a,a):=d; //I know this function result is not a lvalue.
//I relax the assign op type checking rules
//to test of infering of result type
среда, 27 октября 2010 г.
Процесс рефакторинга
Сам по себе процесс продвигался достаточно безболезненно, однако долго.
Приятно, что планируемые изменения в целом концептуально гладко уложились с текущей структурой компилятора. Однако переименование типов и небольшое изменение их зоны отвественности("утрясание" иерархий классов) отняли большую часть времени. Но в целом остался доволен, хотя и утомительное это занятие заниматься не делом.
Приятно, что планируемые изменения в целом концептуально гладко уложились с текущей структурой компилятора. Однако переименование типов и небольшое изменение их зоны отвественности("утрясание" иерархий классов) отняли большую часть времени. Но в целом остался доволен, хотя и утомительное это занятие заниматься не делом.
вторник, 12 октября 2010 г.
Время для рефакторинга
Достаточно сложно считать на несколько шагов вперед, хотя и очень интересно. Не всегда удается выбрать баланс между выразительностью и скоростью на ранних этапах, да и просто порой принятые решения по проектированию на ранних этапах в меньшей степени предусмотрены для задач для более поздних. Хотя технически работать можно и в рамках текущих проектных решений, однако это требует лишних движений, что отвлекает от решения текущих подзадач.
четверг, 7 октября 2010 г.
Added Procedural Types
Added procedural Types to YAR Language
Extend .NET codegen by MSIL calli instruction
Sample
TMyFunc=function(a:integer):integer;
TXfunc=function(var a:double):TMyFunc;
function abc(a:integer):integer;
begin
result:=a;
end;
function abc(var a:double):TMyFunc;
begin
result:=abc;
end;
e:double;
f:integer;
g:TXfunc;
begin
g:=abc;
f:=g(e)(f);
end
=>
.method static public void main() cil managed
{
.entrypoint
ldftn native int abc(float64& a)
stsfld native int g
ldsfld int32 f
ldsflda float64 e
ldsfld native int g
calli native int (float64& a)
calli int32 (int32 a)
stsfld int32 f
ret
}
Extend .NET codegen by MSIL calli instruction
Sample
TMyFunc=function(a:integer):integer;
TXfunc=function(var a:double):TMyFunc;
function abc(a:integer):integer;
begin
result:=a;
end;
function abc(var a:double):TMyFunc;
begin
result:=abc;
end;
e:double;
f:integer;
g:TXfunc;
begin
g:=abc;
f:=g(e)(f);
end
=>
.method static public void main() cil managed
{
.entrypoint
ldftn native int abc(float64& a)
stsfld native int g
ldsfld int32 f
ldsflda float64 e
ldsfld native int g
calli native int (float64& a)
calli int32 (int32 a)
stsfld int32 f
ret
}
вторник, 5 октября 2010 г.
Supporting dot operator for .NET backend
A sample
program Helloworld;
MyrecordStuff=record
a:integer;
b:integer;
c:boolean;
d:double;
f:double;
end;
Myrecord=record
a,b,c:integer;
private
d:integer;
f:integer;
public
pp:MyrecordStuff;
end;
function abc(var a:integer):integer;
begin
a:=1;
result:=5;
end;
function abc(var a:MyrecordStuff):integer;
begin
a.a:=7;
result:=5;
end;
a,b,c,d:Myrecord;
e:double;
f:integer;
begin
abc(a.a);
abc(a.pp.a);
e:=a.pp.d;
a.pp.a:=f;
a.pp.a:=a.pp.b;
write abc(a.pp);
write a.pp.a;
end
Generated MSIL code
.assembly extern mscorlib { auto }
.assembly project {}
.class public sealed MyrecordStuff extends [mscorlib]System.ValueType
{
.field public static int32 b
.field public static float64 f
.field public static int32 a
.field public static bool c
.field public static float64 d
}
.class public sealed Myrecord extends [mscorlib]System.ValueType
{
.field private static int32 f
.field private static int32 d
.field public static int32 b
.field public static int32 a
.field public static int32 c
.field public static valuetype MyrecordStuff pp
}
.field public static valuetype Myrecord b
.field public static float64 e
.field public static valuetype Myrecord a
.field public static int32 f
.field public static valuetype Myrecord c
.field public static valuetype Myrecord d
.method static public void main() cil managed
{
.entrypoint
ldsflda valuetype Myrecord a
ldflda int32 Myrecord::a
call int32 abc(int32& a)
pop
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldflda int32 MyrecordStuff::a
call int32 abc(int32& a)
pop
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld float64 MyrecordStuff::d
stsfld float64 e
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldsfld int32 f
stfld int32 MyrecordStuff::a
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld int32 MyrecordStuff::b
stfld int32 MyrecordStuff::a
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
call int32 abc( valuetype MyrecordStuff& a)
call void [mscorlib]System.Console::Write(int32)
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld int32 MyrecordStuff::a
call void [mscorlib]System.Console::Write(int32)
ret
}
.method static public int32 abc(int32& a) cil managed
{
ldarg a
ldc.i4 1
stind.i4
ldc.i4 5
ret
}
.method static public int32 abc( valuetype MyrecordStuff& a) cil managed
{
ldarg a
ldc.i4 7
stfld int32 MyrecordStuff::a
ldc.i4 5
ret
}
program Helloworld;
MyrecordStuff=record
a:integer;
b:integer;
c:boolean;
d:double;
f:double;
end;
Myrecord=record
a,b,c:integer;
private
d:integer;
f:integer;
public
pp:MyrecordStuff;
end;
function abc(var a:integer):integer;
begin
a:=1;
result:=5;
end;
function abc(var a:MyrecordStuff):integer;
begin
a.a:=7;
result:=5;
end;
a,b,c,d:Myrecord;
e:double;
f:integer;
begin
abc(a.a);
abc(a.pp.a);
e:=a.pp.d;
a.pp.a:=f;
a.pp.a:=a.pp.b;
write abc(a.pp);
write a.pp.a;
end
Generated MSIL code
.assembly extern mscorlib { auto }
.assembly project {}
.class public sealed MyrecordStuff extends [mscorlib]System.ValueType
{
.field public static int32 b
.field public static float64 f
.field public static int32 a
.field public static bool c
.field public static float64 d
}
.class public sealed Myrecord extends [mscorlib]System.ValueType
{
.field private static int32 f
.field private static int32 d
.field public static int32 b
.field public static int32 a
.field public static int32 c
.field public static valuetype MyrecordStuff pp
}
.field public static valuetype Myrecord b
.field public static float64 e
.field public static valuetype Myrecord a
.field public static int32 f
.field public static valuetype Myrecord c
.field public static valuetype Myrecord d
.method static public void main() cil managed
{
.entrypoint
ldsflda valuetype Myrecord a
ldflda int32 Myrecord::a
call int32 abc(int32& a)
pop
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldflda int32 MyrecordStuff::a
call int32 abc(int32& a)
pop
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld float64 MyrecordStuff::d
stsfld float64 e
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldsfld int32 f
stfld int32 MyrecordStuff::a
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld int32 MyrecordStuff::b
stfld int32 MyrecordStuff::a
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
call int32 abc( valuetype MyrecordStuff& a)
call void [mscorlib]System.Console::Write(int32)
ldsflda valuetype Myrecord a
ldflda valuetype MyrecordStuff Myrecord::pp
ldfld int32 MyrecordStuff::a
call void [mscorlib]System.Console::Write(int32)
ret
}
.method static public int32 abc(int32& a) cil managed
{
ldarg a
ldc.i4 1
stind.i4
ldc.i4 5
ret
}
.method static public int32 abc( valuetype MyrecordStuff& a) cil managed
{
ldarg a
ldc.i4 7
stfld int32 MyrecordStuff::a
ldc.i4 5
ret
}
среда, 29 сентября 2010 г.
Deferred assign of type constructor
Working on support syntax like:
TFUNC < T > =function (a:TFUNC < Integer > ):T;
TFUNC < T > =function (a:TFUNC < Integer > ):T;
вторник, 21 сентября 2010 г.
Code generation for YAR polymorphic methods
Added for .NET backend.
All process consists of :
YAR => semantic tree => call codegen(Semantic tree, .NET backend) => IL ASM
Here is a sample:
YAR
function abc < T,U,Z > (a:integer;b:U):Z;
c,d:T;
e:Z;
f:U;
begin
c:=d;
f:=b;
result:=e;
end;
=>
IL ASM
.method static public !!Z abc < T,U,Z > (int32 a,!!U b) cil managed
{
.locals init(!!U f,!!Z e,!!T c,!!T d)
ldloc d
stloc c
ldarg b
stloc f
ldloc e
ret
}
All process consists of :
YAR => semantic tree => call codegen(Semantic tree, .NET backend) => IL ASM
Here is a sample:
YAR
function abc < T,U,Z > (a:integer;b:U):Z;
c,d:T;
e:Z;
f:U;
begin
c:=d;
f:=b;
result:=e;
end;
=>
IL ASM
.method static public !!Z abc < T,U,Z > (int32 a,!!U b) cil managed
{
.locals init(!!U f,!!Z e,!!T c,!!T d)
ldloc d
stloc c
ldarg b
stloc f
ldloc e
ret
}
понедельник, 13 сентября 2010 г.
Обновление синтаксического анализатора YAR и проверка типов
Разбор YAR построен на восходящем синтаксическом анализаторе. И при использовании
символов < и > для указания параметров типов создаются конфликты в состояниях анализатора. После некоторых манипуляций в синтаксический анализатор YAR добавлен механизм для устранения таких конфликтов. То есть выражения вида TList < tlist < t,z,u >,TList < t,z,tlist < z > > > и A < 5 будут успешно разрешены.
Механизм проверки типов также корректно проверят unbound параметры, считая их только самосовместимыми.
function abc < t,u,z >(a:integer;b:U):Z;
c,d:T;
e:Z;
f:U;
begin
c:=d;
f:=b;
result:=e;
end;
Кодогенератор и BackEnd для .NET будет расширен, как только будет закончена работа в front-end части.
символов < и > для указания параметров типов создаются конфликты в состояниях анализатора. После некоторых манипуляций в синтаксический анализатор YAR добавлен механизм для устранения таких конфликтов. То есть выражения вида TList < tlist < t,z,u >,TList < t,z,tlist < z > > > и A < 5 будут успешно разрешены.
Механизм проверки типов также корректно проверят unbound параметры, считая их только самосовместимыми.
function abc < t,u,z >(a:integer;b:U):Z;
c,d:T;
e:Z;
f:U;
begin
c:=d;
f:=b;
result:=e;
end;
Кодогенератор и BackEnd для .NET будет расширен, как только будет закончена работа в front-end части.
вторник, 24 августа 2010 г.
понедельник, 23 августа 2010 г.
суббота, 14 августа 2010 г.
Обновление YAR
-Добавлена оптимизация в работе операторов +, *, -, <, <=, =, <>, >=, > при операндах константах.
Код вычислений не генерируется.
-Оптимизировано вычисление предиката конструкции if и генерация кода аналогично.
Автоматически распространится на все Backend'ы.
Код вычислений не генерируется.
-Оптимизировано вычисление предиката конструкции if и генерация кода аналогично.
Автоматически распространится на все Backend'ы.
среда, 4 августа 2010 г.
пятница, 9 июля 2010 г.
Working with .NET unmanaged API.
I started translating cor.h corhdr.h to pas analogues yesterday.
Today I'm testing the results. 5 min a normal fly. :)
Today I'm testing the results. 5 min a normal fly. :)
понедельник, 31 мая 2010 г.
четверг, 27 мая 2010 г.
Working on Virtual Machine
So just few steps:
1. Read some stuff
2. JVM or CLR Like?
I prefer MSIL for its universal untype load and store operations.
3. Implement it.
1. Read some stuff
2. JVM or CLR Like?
I prefer MSIL for its universal untype load and store operations.
3. Implement it.
воскресенье, 25 апреля 2010 г.
вторник, 6 апреля 2010 г.
пятница, 2 апреля 2010 г.
Just Any type Delphi Case statement
Just Any type Delphi Case statement
Here is the implementation.
TPAIRTYPE[T]=record
Value:T;
Proc:TProc;
end;
CaseAnyTypeClassSupport[T]=class
private
class function GetCaseOption(Value:T;Action:TProc):TPAIRTYPE[T];static;
public
class procedure MyCase(const Value:T;const Pairs:array of TPAIRTYPE[T];ElseProc:TProc=nil);static;
class property CaseOption[Value:T;Action:TProc]:TPAIRTYPE[T] read GetCaseOption;default;
end;
uses generics.defaults;
class procedure CaseAnyTypeClassSupport[T].MyCase(const Value:T;const Pairs:array of TPAIRTYPE[T];ElseProc:TProc=nil);
var Pair:TPAIRTYPE[T];
Comparer:IComparer[T];
begin
Comparer:=TComparer[T].Default;
for Pair in Pairs do
if Comparer.Compare(Value,Pair.Value)=0 then
begin
Pair.Proc();
exit;
end;
if Assigned(ElseProc) then ElseProc();
end;
class function CaseAnyTypeClassSupport.GetCaseOption(Value:T;Action:TProc):TPAIRTYPE[T];
begin
Result.Value:=Value;
Result.Proc:=action;
end;
and a usage
procedure TForm2.FormCreate(Sender: TObject);
var Stuff:CaseAnyTypeClassSupport[string];
begin
Stuff.MyCase('2',
[
Stuff['4',procedure
begin
showmessage('Option 1');
end],
Stuff['2',procedure
begin
showmessage('Option 2');
end]
],procedure
begin
showmessage('Else option');
end);
end;
Because of restriction of Delphi compiler you could not write usage like this
with stuff do
MyCase('2',
[
['4',procedure
begin
showmessage('Option 1');
end],
['2',procedure
begin
showmessage('Option 2');
end]
],procedure
begin
showmessage('Else option');
end);
end;
You can enhance MyCase for supports variance types(type with subtypes) the same way.
Here is the implementation.
TPAIRTYPE[T]=record
Value:T;
Proc:TProc;
end;
CaseAnyTypeClassSupport[T]=class
private
class function GetCaseOption(Value:T;Action:TProc):TPAIRTYPE[T];static;
public
class procedure MyCase(const Value:T;const Pairs:array of TPAIRTYPE[T];ElseProc:TProc=nil);static;
class property CaseOption[Value:T;Action:TProc]:TPAIRTYPE[T] read GetCaseOption;default;
end;
uses generics.defaults;
class procedure CaseAnyTypeClassSupport[T].MyCase(const Value:T;const Pairs:array of TPAIRTYPE[T];ElseProc:TProc=nil);
var Pair:TPAIRTYPE[T];
Comparer:IComparer[T];
begin
Comparer:=TComparer[T].Default;
for Pair in Pairs do
if Comparer.Compare(Value,Pair.Value)=0 then
begin
Pair.Proc();
exit;
end;
if Assigned(ElseProc) then ElseProc();
end;
class function CaseAnyTypeClassSupport
begin
Result.Value:=Value;
Result.Proc:=action;
end;
and a usage
procedure TForm2.FormCreate(Sender: TObject);
var Stuff:CaseAnyTypeClassSupport[string];
begin
Stuff.MyCase('2',
[
Stuff['4',procedure
begin
showmessage('Option 1');
end],
Stuff['2',procedure
begin
showmessage('Option 2');
end]
],procedure
begin
showmessage('Else option');
end);
end;
Because of restriction of Delphi compiler you could not write usage like this
with stuff do
MyCase('2',
[
['4',procedure
begin
showmessage('Option 1');
end],
['2',procedure
begin
showmessage('Option 2');
end]
],procedure
begin
showmessage('Else option');
end);
end;
You can enhance MyCase for supports variance types(type with subtypes) the same way.
пятница, 5 февраля 2010 г.
среда, 3 февраля 2010 г.
Fixed point combinators and Existential type
After adapting mind to functional programming style I have comprehended that stuff.
четверг, 21 января 2010 г.
вторник, 19 января 2010 г.
пятница, 15 января 2010 г.
вторник, 12 января 2010 г.
Polymorphism.
There was my last day of annual leave yesterday.
bounded polymorphism,
F-bounded polymorphism,
match-bounded polymorphism
It is bedtime.
bounded polymorphism,
F-bounded polymorphism,
match-bounded polymorphism
It is bedtime.
Подписаться на:
Сообщения (Atom)