DynamicPropertyAccess 1.0.2

dotnet add package DynamicPropertyAccess --version 1.0.2
NuGet\Install-Package DynamicPropertyAccess -Version 1.0.2
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="DynamicPropertyAccess" Version="1.0.2" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add DynamicPropertyAccess --version 1.0.2
#r "nuget: DynamicPropertyAccess, 1.0.2"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install DynamicPropertyAccess as a Cake Addin
#addin nuget:?package=DynamicPropertyAccess&version=1.0.2

// Install DynamicPropertyAccess as a Cake Tool
#tool nuget:?package=DynamicPropertyAccess&version=1.0.2

Nuget

This library allows you to get or set values to properties by name. It uses compiled lambdas instead that are cached of reflection and is thread safe.

Getting Property Value

You can get property value by calling:

T ObjectExtensions.GetPropertyValue<T>(this object source, string propertyName)

Alternatively, you can create Func that gets property value from an object by calling:

Func<object, object?> TypeExtensions.GetPropertyGetter(this Type type, string propertyName)

Setting Property Value

You can set property value by calling:

void ObjectExtensions.SetPropertyValue(this object source, string propertyName, object value)

Alternatively, you can create Func that sets property value on an object by calling:

Action<object, object?> TypeExtensions.GetPropertySetter(this Type type, string propertyName)

Both methods have TryGet pattern alternatives.

Benchmarks

You can see benchmark for GetPropertyValue method for this library, DynamicPropertyAccessor, FastMember and Reflection.
Methods with ...Reused suffix create Func/PropertyInfo once and then reuse it in benchmark.

BenchmarkDotNet=v0.13.5, OS=Windows 10 (10.0.19044.2604/21H2/November2021Update)
AMD Ryzen 7 3700X, 1 CPU, 16 logical and 8 physical cores
  [Host]               : .NET Framework 4.8 (4.8.4614.0), X64 RyuJIT VectorSize=256
  .NET 7.0             : .NET 7.0.3 (7.0.323.6910), X64 RyuJIT AVX2
  .NET Framework 4.7.2 : .NET Framework 4.8 (4.8.4614.0), X64 RyuJIT VectorSize=256
Method Job Runtime Mean Error StdDev
DynamicPropertyAccess .NET 7.0 .NET 7.0 26.776 ns 0.4549 ns 0.4255 ns
DynamicPropertyAccessor .NET 7.0 .NET 7.0 31.530 ns 0.3986 ns 0.3728 ns
FastMember .NET 7.0 .NET 7.0 30.972 ns 0.4472 ns 0.4183 ns
Reflection .NET 7.0 .NET 7.0 33.527 ns 0.4686 ns 0.4383 ns
DynamicPropertyAccessReused .NET 7.0 .NET 7.0 1.475 ns 0.0332 ns 0.0311 ns
FastMemberReused .NET 7.0 .NET 7.0 14.770 ns 0.1585 ns 0.1483 ns
ReflectionReused .NET 7.0 .NET 7.0 12.105 ns 0.2688 ns 0.2514 ns
DynamicPropertyAccess .NET Framework 4.7.2 .NET Framework 4.7.2 66.566 ns 1.2969 ns 1.3877 ns
DynamicPropertyAccessor .NET Framework 4.7.2 .NET Framework 4.7.2 57.735 ns 1.1492 ns 1.4113 ns
FastMember .NET Framework 4.7.2 .NET Framework 4.7.2 43.592 ns 0.1853 ns 0.1547 ns
Reflection .NET Framework 4.7.2 .NET Framework 4.7.2 169.533 ns 2.0884 ns 1.6305 ns
DynamicPropertyAccessReused .NET Framework 4.7.2 .NET Framework 4.7.2 5.731 ns 0.0291 ns 0.0227 ns
FastMemberReused .NET Framework 4.7.2 .NET Framework 4.7.2 24.150 ns 0.1821 ns 0.1703 ns
ReflectionReused .NET Framework 4.7.2 .NET Framework 4.7.2 120.565 ns 2.4210 ns 4.9455 ns

As you can see, GetPropertyValue is slightly slower on .NET Framework than available alternatives. This is because of thread safety overhead. On the other hand, reused Func from GetPropertyGetter is significantly faster. Additionally, Reflection is almost as fast on .NET 7+. You can read more about it here.

Conclusions

  • Use this library (and reuse Getter/Setter) if:
    • Read/Write operations are frequently performed.
    • Thread safety is important.
  • Use other libraries if:
    • Read/Write operations are rarely performed.
    • Thread safety is not a concern.
    • Target framework is .NET Framework, .NET 6 or below.
  • Use Reflection if:
    • Target framework is .NET 7+ and Read/Write operations are rarely performed.
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • .NETStandard 2.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.2 546 2/27/2023
1.0.1 212 2/26/2023
1.0.0 217 2/25/2023