BlazarTech.QueryableValues.SqlServer
3.2.0
See the version list below for details.
dotnet add package BlazarTech.QueryableValues.SqlServer --version 3.2.0
NuGet\Install-Package BlazarTech.QueryableValues.SqlServer -Version 3.2.0
<PackageReference Include="BlazarTech.QueryableValues.SqlServer" Version="3.2.0" />
paket add BlazarTech.QueryableValues.SqlServer --version 3.2.0
#r "nuget: BlazarTech.QueryableValues.SqlServer, 3.2.0"
// Install BlazarTech.QueryableValues.SqlServer as a Cake Addin #addin nuget:?package=BlazarTech.QueryableValues.SqlServer&version=3.2.0 // Install BlazarTech.QueryableValues.SqlServer as a Cake Tool #tool nuget:?package=BlazarTech.QueryableValues.SqlServer&version=3.2.0
QueryableValues
This library allows us to efficiently compose an IEnumerable<T> in our Entity Framework Core queries when using the SQL Server Database Provider. This is done by using the AsQueryableValues
extension method that is made available on the DbContext class. Everything is evaluated on the server with a single roundtrip, in a way that preserves the query's execution plan, even when the values behind the IEnumerable<T> are changed on subsequent executions.
The supported types for T
are:
- Simple Type: Int32, Int64, Decimal, Double, DateTime, DateTimeOffset, Guid, and String.
- Complex Type:
- Can be an anonymous type.
- Can be a user defined class or struct, with read/write properties and a public constructor.
- Must have one or more simple type properties.
For a detailed explanation, please continue reading here.
When Should I Use It?
The AsQueryableValues
extension method is intended for queries that are dependent on a non-constant sequence of external values. In this case, the underline SQL query will be efficient on subsequent executions.
It provides a solution to the following long standing EF Core issue and enables other currently unsupported scenarios; like the ability to efficiently create joins with in-memory data.
Getting Started
Installation
QueryableValues is distributed as a NuGet Package. The major version number of this library is aligned with the version of Entity Framework Core that's supported by it; for example, if you are using EF Core 5, then you must use version 5 of QueryableValues.
Please choose the appropriate command below to install it using the NuGet Package Manager Console window in Visual Studio:
EF Core | Command |
---|---|
3.x | Install-Package BlazarTech.QueryableValues.SqlServer -Version 3.2.0 |
5.x | Install-Package BlazarTech.QueryableValues.SqlServer -Version 5.2.0 |
6.x | Install-Package BlazarTech.QueryableValues.SqlServer -Version 6.2.0 |
Configuration
Look for the place in your code where you are setting up your DbContext and calling the UseSqlServer extension method, then use a lambda expression to access the SqlServerDbContextOptionsBuilder
provided by it. It is on this builder that you must call the UseQueryableValues
extension method, as shown in the following simplified examples:
When using the OnConfiguring
method inside your DbContext:
using BlazarTech.QueryableValues;
public class MyDbContext : DbContext
{
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer(
"MyConnectionString",
sqlServerOptionsBuilder =>
{
sqlServerOptionsBuilder.UseQueryableValues();
}
);
}
}
When setting up the DbContext at registration time using dependency injection:
using BlazarTech.QueryableValues;
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services.AddDbContext<MyDbContext>(optionsBuilder => {
optionsBuilder.UseSqlServer(
"MyConnectionString",
sqlServerOptionsBuilder =>
{
sqlServerOptionsBuilder.UseQueryableValues();
}
);
});
}
}
How Do I Use It?
The AsQueryableValues
extension method is provided by the BlazarTech.QueryableValues
namespace, therefore, you must add the following using
directive to your source code file in order for it to appear as a method of your DbContext instance:
using BlazarTech.QueryableValues;
Below you can find a few examples composing a query using the values provided by an IEnumerable<T>.
Simple Type Examples
Using the Contains LINQ method:
// Sample values.
IEnumerable<int> values = Enumerable.Range(1, 10);
// Example #1 (LINQ method syntax)
var myQuery1 = dbContext.MyEntities
.Where(i => dbContext
.AsQueryableValues(values)
.Contains(i.MyEntityID)
)
.Select(i => new
{
i.MyEntityID,
i.PropA
});
// Example #2 (LINQ query syntax)
var myQuery2 =
from i in dbContext.MyEntities
where dbContext
.AsQueryableValues(values)
.Contains(i.MyEntityID)
select new
{
i.MyEntityID,
i.PropA
});
Using the Join LINQ method:
// Sample values.
IEnumerable<int> values = Enumerable.Range(1, 10);
// Example #1 (LINQ method syntax)
var myQuery1 = dbContext.MyEntities
.Join(
dbContext.AsQueryableValues(values),
i => i.MyEntityID,
v => v,
(i, v) => new
{
i.MyEntityID,
i.PropA
}
);
// Example #2 (LINQ query syntax)
var myQuery2 =
from i in dbContext.MyEntities
join v in dbContext.AsQueryableValues(values) on i.MyEntityID equals v
select new
{
i.MyEntityID,
i.PropA
});
Complex Type Examples
// If your IEnumerable<T> variable's item type is a complex type with many properties,
// project only what you need to a new variable and use it in your query.
var projectedItems = items.Select(i => new { i.CategoryId, i.ColorName });
var myQuery =
from p in dbContext.Product
join pi in dbContext.AsQueryableValues(projectedItems) on new { p.CategoryId, p.ColorName } equals new { pi.CategoryId, pi.ColorName }
select new
{
p.ProductId,
p.Description
};
About Complex Types
⚠️ All the data provided by this type is transmitted to the server, therefore, ensure that it only contains the properties that you need for your query. Not following this recommendation will degrade the query's performance.
⚠️ There is a limit of up to ten properties for any given simple type (e.g., cannot have more than ten Int32 properties). Exceeding that limit will cause an exception and may also be a sign that you should rethink your strategy.
Do You Want to Know More? 📚
Please take a look at the repository here.
Product | Versions 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. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.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. |
-
.NETStandard 2.0
- Microsoft.EntityFrameworkCore.SqlServer (>= 3.1.14 && < 5.0.0)
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 |
---|---|---|
8.1.1 | 63,058 | 2/8/2024 |
8.1.0 | 10,031 | 12/7/2023 |
8.0.0 | 1,830 | 11/26/2023 |
7.4.3 | 107,024 | 2/8/2024 |
7.4.2 | 2,290 | 12/7/2023 |
7.4.1 | 6,220 | 11/26/2023 |
7.4.0 | 60,619 | 6/25/2023 |
7.3.0 | 4,097 | 5/20/2023 |
7.2.0 | 3,744 | 3/27/2023 |
7.1.0 | 2,314 | 3/14/2023 |
7.0.0 | 16,148 | 11/13/2022 |
7.0.0-preview.2 | 183 | 9/3/2022 |
7.0.0-preview.1 | 148 | 8/7/2022 |
6.9.3 | 2,714 | 2/8/2024 |
6.9.2 | 15,824 | 12/7/2023 |
6.9.1 | 163 | 11/26/2023 |
6.9.0 | 13,286 | 6/25/2023 |
6.8.0 | 101,248 | 5/20/2023 |
6.7.0 | 3,614 | 3/27/2023 |
6.6.0 | 596 | 3/14/2023 |
6.5.0 | 96,816 | 9/3/2022 |
6.4.0 | 7,349 | 7/14/2022 |
6.3.0 | 11,580 | 1/4/2022 |
6.2.0 | 379 | 12/27/2021 |
6.1.0 | 357 | 12/9/2021 |
6.0.0 | 1,059 | 12/6/2021 |
5.9.3 | 166 | 2/8/2024 |
5.9.2 | 164 | 12/7/2023 |
5.9.1 | 146 | 11/26/2023 |
5.9.0 | 190 | 6/25/2023 |
5.8.0 | 187 | 5/20/2023 |
5.7.0 | 253 | 3/27/2023 |
5.6.0 | 329 | 3/14/2023 |
5.5.0 | 1,892 | 9/3/2022 |
5.4.0 | 1,109 | 7/14/2022 |
5.3.0 | 31,826 | 1/4/2022 |
5.2.0 | 357 | 12/27/2021 |
5.1.0 | 362 | 12/9/2021 |
5.0.0 | 355 | 12/6/2021 |
3.9.3 | 132 | 2/8/2024 |
3.9.2 | 150 | 12/7/2023 |
3.9.1 | 156 | 11/26/2023 |
3.9.0 | 274 | 6/25/2023 |
3.8.0 | 192 | 5/20/2023 |
3.7.0 | 268 | 3/27/2023 |
3.6.0 | 272 | 3/14/2023 |
3.5.0 | 469 | 9/3/2022 |
3.4.0 | 499 | 7/14/2022 |
3.3.0 | 1,458 | 1/4/2022 |
3.2.0 | 307 | 12/27/2021 |
3.1.0 | 344 | 12/9/2021 |
3.0.0 | 440 | 12/6/2021 |