![]() |
VOOZH | about |
dotnet add package FastExpressionCompiler.LightExpression --version 5.4.1
NuGet\Install-Package FastExpressionCompiler.LightExpression -Version 5.4.1
<PackageReference Include="FastExpressionCompiler.LightExpression" Version="5.4.1" />
<PackageVersion Include="FastExpressionCompiler.LightExpression" Version="5.4.1" />Directory.Packages.props
<PackageReference Include="FastExpressionCompiler.LightExpression" />Project file
paket add FastExpressionCompiler.LightExpression --version 5.4.1
#r "nuget: FastExpressionCompiler.LightExpression, 5.4.1"
#:package FastExpressionCompiler.LightExpression@5.4.1
#addin nuget:?package=FastExpressionCompiler.LightExpression&version=5.4.1Install as a Cake Addin
#tool nuget:?package=FastExpressionCompiler.LightExpression&version=5.4.1Install as a Cake Tool
<img src="./logo.png" alt="logo"/>
👁 Latest Release Notes
👁 License
👁 Build Windows,Ubuntu
Targets .NET 6+, .NET 4.7.2+, .NET Standard 2.0+
NuGet packages:
The project was originally a part of the DryIoc, so check it out 😉
ExpressionTree compilation is used by the wide variety of tools, e.g. IoC/DI containers, Serializers, ORMs and OOMs.
But Expression.Compile() is just slow.
Moreover the compiled delegate may be slower than the manually created delegate because of the reasons:
TL;DR;
Expression.Compile creates a DynamicMethod and associates it with an anonymous assembly to run it in a sand-boxed environment. This makes it safe for a dynamic method to be emitted and executed by partially trusted code but adds some run-time overhead.
See also a deep dive to Delegate internals.
The FastExpressionCompiler .CompileFast() extension method is 10-40x times faster than .Compile().
The compiled delegate may be in some cases a lot faster than the one produced by .Compile().
Note: The actual performance may vary depending on the multiple factors: platform, how complex is expression, does it have a closure, does it contain nested lambdas, etc.
In addition, the memory consumption taken by the compilation will be much smaller (check the Allocated column in the benchmarks below).
Updated to .NET 9.0
BenchmarkDotNet v0.15.0, Windows 11 (10.0.26100.4061/24H2/2024Update/HudsonValley)
Intel Core i9-8950HK CPU 2.90GHz (Coffee Lake), 1 CPU, 12 logical and 6 physical cores
.NET SDK 9.0.203
[Host] : .NET 9.0.4 (9.0.425.16305), X64 RyuJIT AVX2
DefaultJob : .NET 9.0.4 (9.0.425.16305), X64 RyuJIT AVX2
var a = new A();
var b = new B();
Expression<Func<X>> e = () => new X(a, b);
Compiling expression:
| Method | Mean | Error | StdDev | Ratio | RatioSD | Rank | Gen0 | Gen1 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|---|
| CompileFast | 3.183 us | 0.0459 us | 0.0407 us | 1.00 | 0.02 | 1 | 0.1984 | 0.1945 | 1.23 KB | 1.00 |
| Compile | 147.312 us | 1.9291 us | 1.8946 us | 46.28 | 0.81 | 2 | 0.4883 | 0.2441 | 4.48 KB | 3.65 |
Invoking the compiled delegate (comparing to the direct constructor call):
| Method | Mean | Error | StdDev | Ratio | RatioSD | Rank | Gen0 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|
| DirectConstructorCall | 6.055 ns | 0.0632 ns | 0.0560 ns | 1.00 | 0.01 | 1 | 0.0051 | 32 B | 1.00 |
| CompiledLambda | 7.853 ns | 0.2013 ns | 0.1681 ns | 1.30 | 0.03 | 2 | 0.0051 | 32 B | 1.00 |
| FastCompiledLambda | 7.962 ns | 0.2186 ns | 0.4052 ns | 1.31 | 0.07 | 2 | 0.0051 | 32 B | 1.00 |
var a = new A();
var b = new B();
Expression<Func<X>> getXExpr = () => CreateX((aa, bb) => new X(aa, bb), new Lazy<A>(() => a), b);
Compiling expression:
| Method | Mean | Error | StdDev | Ratio | RatioSD | Rank | Gen0 | Gen1 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|---|
| CompileFast | 11.12 us | 0.189 us | 0.158 us | 1.00 | 0.02 | 1 | 0.6104 | 0.5798 | 3.77 KB | 1.00 |
| Compile | 415.09 us | 4.277 us | 3.571 us | 37.34 | 0.60 | 2 | 1.9531 | 1.4648 | 12.04 KB | 3.19 |
Invoking compiled delegate comparing to direct method call:
| Method | Mean | Error | StdDev | Ratio | RatioSD | Rank | Gen0 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|
| DirectMethodCall | 40.29 ns | 0.549 ns | 0.487 ns | 1.00 | 0.02 | 1 | 0.0268 | 168 B | 1.00 |
| Invoke_CompiledFast | 40.59 ns | 0.157 ns | 0.123 ns | 1.01 | 0.01 | 1 | 0.0166 | 104 B | 0.62 |
| Invoke_Compiled | 1,142.12 ns | 11.877 ns | 14.586 ns | 28.35 | 0.48 | 2 | 0.0420 | 264 B | 1.57 |
var a = new A();
var bParamExpr = Expression.Parameter(typeof(B), "b");
var expr = Expression.Lambda(
Expression.New(_ctorX,
Expression.Constant(a, typeof(A)), bParamExpr),
bParamExpr);
Compiling expression:
| Method | Mean | Error | StdDev | Ratio | RatioSD | Rank | Gen0 | Gen1 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|---|
| CompileFast_LightExpression | 3.107 us | 0.0562 us | 0.0498 us | 0.99 | 0.02 | 1 | 0.1755 | 0.1678 | 1.08 KB | 1.00 |
| CompileFast_SystemExpression | 3.126 us | 0.0288 us | 0.0256 us | 1.00 | 0.01 | 1 | 0.1755 | 0.1678 | 1.08 KB | 1.00 |
| Compile_SystemExpression | 103.948 us | 1.9593 us | 2.5477 us | 33.26 | 0.84 | 2 | 0.7324 | 0.4883 | 4.74 KB | 4.40 |
Invoking the compiled delegate compared to the normal delegate and the direct call:
| Method | Mean | Error | StdDev | Ratio | Rank | Gen0 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|
| DirectCall | 10.19 ns | 0.108 ns | 0.085 ns | 1.00 | 1 | 0.0051 | 32 B | 1.00 |
| CompiledFast_LightExpression | 10.70 ns | 0.089 ns | 0.070 ns | 1.05 | 2 | 0.0051 | 32 B | 1.00 |
| CompiledFast_SystemExpression | 10.91 ns | 0.071 ns | 0.066 ns | 1.07 | 2 | 0.0051 | 32 B | 1.00 |
| Compiled_SystemExpression | 11.59 ns | 0.098 ns | 0.081 ns | 1.14 | 3 | 0.0051 | 32 B | 1.00 |
FastExpressionCompiler.LightExpression.Expression is the lightweight version of System.Linq.Expressions.Expression.
It is designed to be a drop-in replacement for the System Expression - just install the FastExpressionCompiler.LightExpression package instead of FastExpressionCompiler and replace the usings
using System.Linq.Expressions;
using static System.Linq.Expressions.Expression;
with
using static FastExpressionCompiler.LightExpression.Expression;
namespace FastExpressionCompiler.LightExpression.UnitTests
You may look at it as a bare-bone wrapper for the computation operation node which helps you to compose the computation tree (without messing with the IL emit directly).
It won't validate operations compatibility for the tree the way System.Linq.Expression does it, and partially why it is so slow.
Hopefully you are checking the expression arguments yourself and not waiting for the Expression exceptions to blow-up.
Creating the expression:
| Method | Mean | Error | StdDev | Median | Ratio | RatioSD | Rank | Gen0 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|---|
| Create_LightExpression | 156.6 ns | 3.19 ns | 8.18 ns | 151.9 ns | 1.00 | 0.07 | 1 | 0.0827 | 520 B | 1.00 |
| Create_SystemExpression | 1,065.0 ns | 14.24 ns | 11.89 ns | 1,069.3 ns | 6.82 | 0.34 | 2 | 0.2060 | 1304 B | 2.51 |
Creating and compiling:
| Method | Mean | Error | StdDev | Median | Ratio | RatioSD | Rank | Gen0 | Gen1 | Allocated | Alloc Ratio |
|---|---|---|---|---|---|---|---|---|---|---|---|
| Create_LightExpression_and_CompileFast | 4.957 us | 0.0986 us | 0.2362 us | 4.913 us | 1.00 | 0.07 | 1 | 0.3510 | 0.3052 | 2.15 KB | 1.00 |
| Create_SystemExpression_and_CompileFast | 6.518 us | 0.1889 us | 0.5541 us | 6.300 us | 1.32 | 0.13 | 2 | 0.4578 | 0.4272 | 2.97 KB | 1.38 |
| Create_SystemExpression_and_Compile | 205.000 us | 4.0938 us | 7.3819 us | 206.353 us | 41.44 | 2.45 | 3 | 0.9766 | 0.4883 | 7.15 KB | 3.33 |
FastExpressionCompiler
CompileFast extension methods for the System.Linq.Expressions.LambdaExpression.FastExpressionCompiler.LightExpression
CompileFast extension methods for FastExpressionCompiler.LightExpression.LambdaExpression.ExpressionVisitor.ToExpression method to convert back to the System.Linq.Expressions.Expression.ToLightExpression conversion method to convert from the System.Linq.Expressions.Expression to FastExpressionCompiler.LightExpression.Expression.Both FastExpressionCompiler and FastExpressionCompiler.LightExpression
ToCSharpString() method to output the compilable C# code represented by the expression.ToExpressionString() method to output the expression construction C# code, so given the expression object you'll get e.g. Expression.Lambda(Expression.New(...)).Marten, Rebus, StructureMap, Lamar, ExpressionToCodeLib, NServiceBus, LINQ2DB, MapsterMapper
Considering: Moq, Apex.Serialization
Install from the NuGet and add the using FastExpressionCompiler; and replace the call to the .Compile() with the .CompileFast() extension method.
Note: CompileFast has an optional parameter bool ifFastFailedReturnNull = false to disable fallback to Compile.
Hoisted lambda expression (created by the C# Compiler):
var a = new A(); var b = new B();
Expression<Func<X>> expr = () => new X(a, b);
var getX = expr.CompileFast();
var x = getX();
Manually composed lambda expression:
var a = new A();
var bParamExpr = Expression.Parameter(typeof(B), "b");
var expr = Expression.Lambda(
Expression.New(_ctorX,
Expression.Constant(a, typeof(A)), bParamExpr),
bParamExpr);
var f = expr.CompileFast();
var x = f(new B());
Note: You may simplify Expression usage and enable faster refactoring with the C# using static statement:
using static System.Linq.Expressions.Expression;
// or
// using static FastExpressionCompiler.LightExpression.Expression;
var a = new A();
var bParamExpr = Parameter(typeof(B), "b");
var expr = Lambda(
New(_ctorX, Constant(a, typeof(A)), bParamExpr),
bParamExpr);
var f = expr.CompileFast();
var x = f(new B());
The idea is to provide the fast compilation for the supported expression types
and fallback to the system Expression.Compile() for the not supported types:
FEC does not support yet:
QuoteDynamicRuntimeVariablesDebugInfoMemberInit with the MemberMemberBinding and the ListMemberBinding binding typesNewArrayInit multi-dimensional array initializer is not supported yetTo find what nodes are not supported in your expression you may use the technic described below in the Diagnostics section.
The compilation is done by traversing the expression nodes and emitting the IL. The code is tuned for the performance and the minimal memory consumption.
The expression is traversed twice:
DynamicMethod.If visitor finds the not supported expression node or the error condition,
the compilation is aborted, and null is returned enabling the fallback to System .Compile().
FEC V3 has added powerful diagnostics and code generation tools.
You may pass the optional CompilerFlags.EnableDelegateDebugInfo into the CompileFast methods.
EnableDelegateDebugInfo adds the diagnostic info into the compiled delegate including its source Expression and compiled IL code.
It can be used as following:
System.Linq.Expressions.Expression<Func<int, Func<int>>> e =
n => () => n + 1;
var f = e.CompileFast(flags: CompilerFlags.EnableDelegateDebugInfo);
var d = f.TryGetDebugInfo();
d.PrintExpression();
d.PrintCSharp();
d.PrintIL(); // available in NET8+
<details><summary>Expand to see the output of the above code...</summary>
Output of d.PrintExpression() is the valid C#:
var p = new ParameterExpression[1]; // the parameter expressions
var e = new Expression[3]; // the unique expressions
var expr = Lambda<Func<int, Func<int>>>(
e[0]=Lambda<Func<int>>(
e[1]=MakeBinary(ExpressionType.Add,
p[0]=Parameter(typeof(int), "n"),
e[2]=Constant(1)), new ParameterExpression[0]),
p[0 // (int n)
]);
Output of d.PrintCSharp() is the valid C#:
var @cs = (Func<int, Func<int>>)((int n) => //Func<int>
(Func<int>)(() => //int
n + 1));
Output of d.PrintIL() (includes the IL of the nested lambda):
<Caller>
0 ldarg.0
1 ldfld object[] ExpressionCompiler.ArrayClosure.ConstantsAndNestedLambdas
6 stloc.0
7 ldloc.0
8 ldc.i4.0
9 ldelem.ref
10 stloc.1
11 ldloc.1
12 ldc.i4.1
13 newarr object
18 stloc.2
19 ldloc.2
20 stfld object[] ExpressionCompiler.NestedLambdaForNonPassedParams.NonPassedParams
25 ldloc.2
26 ldc.i4.0
27 ldarg.1
28 box int
33 stelem.ref
34 ldloc.1
35 ldfld object ExpressionCompiler.NestedLambdaForNonPassedParams.NestedLambda
40 ldloc.2
41 ldloc.1
42 ldfld object[] ExpressionCompiler.NestedLambdaForNonPassedParamsWithConstants.ConstantsAndNestedLambdas
47 newobj ExpressionCompiler.ArrayClosureWithNonPassedParams(System.Object[], System.Object[])
52 call Func<int> ExpressionCompiler.CurryClosureFuncs.Curry(System.Func`2[FastExpressionCompiler.LightExpression.ExpressionCompiler+ArrayClosure,System.Int32], ArrayClosure)
57 ret
</Caller>
<0_nested_in_Caller>
0 ldarg.0
1 ldfld object[] ExpressionCompiler.ArrayClosureWithNonPassedParams.NonPassedParams
6 ldc.i4.0
7 ldelem.ref
8 unbox.any int
13 ldc.i4.1
14 add
15 ret
</0_nested_in_Caller>
</details>
FEC V3.1 has added the compiler flag CompilerFlags.ThrowOnNotSupportedExpression.
When passed to CompileFast(flags: CompilerFlags.ThrowOnNotSupportedExpression) and the expression contains not (yet) supported Expression node the compilation will throw the exception instead of returning null.
To get the whole list of the not yet supported cases you may check in Result.NotSupported_ enum values.
The Code Generation capabilities are available via the ToCSharpString and ToExpressionString extension methods.
Note: When converting the source expression to either C# code or to the Expression construction code you may find
the // NOT_SUPPORTED_EXPRESSION comments marking the not supported yet expressions by FEC. So you may test the presence or absence of this comment.
FastExpressionCompiler.LightExpression.Expression instead of System.Linq.Expressions.Expression for the faster expression creation..TryCompileWithPreCreatedClosure and .TryCompileWithoutClosure methods when you know the expression at hand and may skip the first traversing round, e.g. for the "static" expression which does not contain the bound constants. Note: You cannot skip the 1st round if the expression contains the Block, Try, or Goto expressions.<a target="_blank" href="https://icons8.com/icons/set/bitten-ice-pop">Bitten Ice Pop icon</a> icon by <a target="_blank" href="https://icons8.com">Icons8</a>
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net5.0 net5.0 was computed. net5.0-windows net5.0-windows was computed. net6.0 net6.0 is compatible. net6.0-android net6.0-android was computed. net6.0-ios net6.0-ios was computed. net6.0-maccatalyst net6.0-maccatalyst was computed. net6.0-macos net6.0-macos was computed. net6.0-tvos net6.0-tvos was computed. net6.0-windows net6.0-windows was computed. net7.0 net7.0 was computed. net7.0-android net7.0-android was computed. net7.0-ios net7.0-ios was computed. net7.0-maccatalyst net7.0-maccatalyst was computed. net7.0-macos net7.0-macos was computed. net7.0-tvos net7.0-tvos was computed. net7.0-windows net7.0-windows was computed. net8.0 net8.0 is compatible. net8.0-android net8.0-android was computed. net8.0-browser net8.0-browser was computed. net8.0-ios net8.0-ios was computed. net8.0-maccatalyst net8.0-maccatalyst was computed. net8.0-macos net8.0-macos was computed. net8.0-tvos net8.0-tvos was computed. net8.0-windows net8.0-windows was computed. net9.0 net9.0 is compatible. net9.0-android net9.0-android was computed. net9.0-browser net9.0-browser was computed. net9.0-ios net9.0-ios was computed. net9.0-maccatalyst net9.0-maccatalyst was computed. net9.0-macos net9.0-macos was computed. net9.0-tvos net9.0-tvos was computed. net9.0-windows net9.0-windows was computed. net10.0 net10.0 was computed. net10.0-android net10.0-android was computed. net10.0-browser net10.0-browser was computed. net10.0-ios net10.0-ios was computed. net10.0-maccatalyst net10.0-maccatalyst was computed. net10.0-macos net10.0-macos was computed. net10.0-tvos net10.0-tvos was computed. net10.0-windows net10.0-windows was computed. |
| .NET Core | netcoreapp2.0 netcoreapp2.0 was computed. netcoreapp2.1 netcoreapp2.1 was computed. netcoreapp2.2 netcoreapp2.2 was computed. netcoreapp3.0 netcoreapp3.0 was computed. netcoreapp3.1 netcoreapp3.1 was computed. |
| .NET Standard | netstandard2.0 netstandard2.0 is compatible. netstandard2.1 netstandard2.1 is compatible. |
| .NET Framework | net461 net461 was computed. net462 net462 was computed. net463 net463 was computed. net47 net47 was computed. net471 net471 was computed. net472 net472 is compatible. net48 net48 was computed. net481 net481 was computed. |
| MonoAndroid | monoandroid monoandroid was computed. |
| MonoMac | monomac monomac was computed. |
| MonoTouch | monotouch monotouch was computed. |
| Tizen | tizen40 tizen40 was computed. tizen60 tizen60 was computed. |
| Xamarin.iOS | xamarinios xamarinios was computed. |
| Xamarin.Mac | xamarinmac xamarinmac was computed. |
| Xamarin.TVOS | xamarintvos xamarintvos was computed. |
| Xamarin.WatchOS | xamarinwatchos xamarinwatchos was computed. |
Showing the top 5 NuGet packages that depend on FastExpressionCompiler.LightExpression:
| Package | Downloads |
|---|---|
|
SecurityLogin.Redis
An security method to login |
|
|
StirlingLabs.StringToExpression
StringToExpression allows you to create methods that take strings and outputs .NET expressions. It is highly configurable allowing you to define your own language with your own syntax. |
|
|
Orsak.AspNetCore
Package Description |
|
|
Faster.Ioc
Package Description |
|
|
SecurityLogin.Cache
An security method to login |
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 5.4.1 | 700 | 4/13/2026 |
| 5.4.0 | 201 | 3/30/2026 |
| 5.3.0 | 11,990 | 6/10/2025 |
| 5.2.0 | 766 | 5/12/2025 |
| 5.2.0-preview-02 | 282 | 4/27/2025 |
| 5.2.0-preview-01 | 184 | 4/26/2025 |
| 5.1.1 | 2,640 | 4/2/2025 |
| 5.1.0 | 335 | 3/31/2025 |
| 5.0.3 | 634 | 3/26/2025 |
| 5.0.2 | 2,988 | 1/29/2025 |
| 5.0.1 | 800 | 12/22/2024 |
| 5.0.0 | 699 | 11/22/2024 |
| 4.2.2 | 7,987 | 10/13/2024 |
| 4.2.1 | 3,314 | 7/2/2024 |
| 4.2.0 | 12,453 | 4/29/2024 |
| 4.1.0 | 3,384 | 1/20/2024 |
| 4.0.1 | 4,522 | 11/23/2023 |
| 4.0.0 | 318 | 11/12/2023 |
| 3.4.0-preview-01 | 360 | 8/19/2023 |
| 3.3.4 | 5,398 | 1/17/2023 |
## v5.4.1 Bug-fix release
- #498 System.InvalidProgramException when using loop
- #499 System.InvalidProgramException for Sorting and comparison function
**Contributions**: @bluewingHuang
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.4.0...v5.4.1
## v5.4.0 Feature and bug-fix release
### Features
- #398 Optimize Switch with OpCodes.Switch
- #484 PrintIL options to output op-codes in the format directly copyable to AssertOpCodes method
- #482 [impl] Update to the modern optimized and fixed versions of ImTools SmallList and SmallMap
### Fixes
- Multiple fixes in `ToCSharpString`, `PrintCSharp`
- #480 "CLR detected an invalid program" exception
- #483 fix recursive PrintIL output for Label Offset and Argument Ordinal
- #485 PrintCSharp for Switch does not prints "break" when required
- #487 Fix ToCSharpString output for boolean equality expressions
- #490 Regression in compiling lambdas with ref struct parameters
- by [exyi](https://github.com/exyi) #491 Fix unsigned int to floating point number conversion
- by [exyi](https://github.com/exyi) #493 Fix ParentFlags propagating in TryEmitNewExpression
- #495 Incomplete pattern detection for error 1007: Return goto from TryCatch with Assign generates invalid IL instead of failing gracefully
### Full Changelog
https://github.com/dadhi/FastExpressionCompiler/compare/v5.3.3...v5.4.0
## v5.3.0 Feature release
- feat: [performance] Reuse the ILGenerator used by the nested lambda compilation
- feat: [performance] Reuse DynamicMethod if possible
- feat: Add debug diagnostics to the nested lambdas
- fix: System.ExecutionEngineException with nullables on repeated calls to ConcurrentDictionary
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.2.0...v5.3.0
## v5.2.0 Feature release
- feat: #466 Simplify creation of the updateable value that can be put into Closure with the Expression.ConstantRef
- feat: #472 Interpreter to reduce the IL emitted by the Logical, Comparison, Arithmetics expressions, by calculating them during compile time and using the calculation result in IL
- feat: Expose TestTools and FastExpressionCompiler.ILDecoder to enable testing and diagnostic in the consumer code
## v5.1.1. Bug-fix release
- fix: #461 InvalidProgramException when null checking type by ref
- fix: #462 Optimize Equality comparison with null
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.1.0...v5.1.1
## v5.1.0 Small feature release
- add: #458 Add Support for TryFault
- fix: #460 ArgumentException when converting from object to type with explicit operator
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.0.3...v5.1.0
## v5.0.3 Bug-fix release
- fix: #451 Operator implicit/explicit produces InvalidProgram
- fix: #455 TypeAs should return null when it cannot cast to a nullable #455
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.0.2...v5.0.3
## v5.0.2 Bug-fix release
-fix: #449 MemberInit produces InvalidProgram
**Full Changelog**: https://github.com/dadhi/FastExpressionCompiler/compare/v5.0.1...v5.0.2
## v5.0.1 Bug-fix release
- fix: #437 Shared variables with nested lambdas returning incorrect values
- fix: #439 Support unused Field access in Block
- fix: #440 Errors with simplified Switch cases
- fix: #441 Fails to pass Constant as call parameter by-reference
- fix: #442 TryCatch and the Goto outside problems
- fix: #443 Nested Calls with Lambda Parameters Throwing NotSupportedExpressionException
## v5.0.0 Major feature release
- feat: #271 Add conversion from the System Expression to the Light Expression to allow hoisted system expression embedding in the larger light expression
- feat: #432 Add explicit target for .NET 9
- feat: #434 ToCSharpString should generate stable names for the unnamed variables, parameters and labels
- fix: #262 Optimize OpCode.Ceq plus Branch into a single compare-and-branch
- fix: #424 TryCatch expression as argument in Call expression produces invalid program
- fix: #426 Directly passing a method result to another method by ref, fails with InvalidProgramException
- fix: #427 CSharp output with more consistency, robustness and more understandable code
- fix: #428 Expression.Switch without a default case incorrectly calls first case for unmatched values
- fix: #430 TryCatch - Bad label content in ILGenerator