Динамический поиск общего DbSet в DbContextC#

Место общения программистов C#
Ответить
Anonymous
 Динамический поиск общего DbSet в DbContext

Сообщение Anonymous »

Я знаю, что этот вопрос уже задавался, но я не смог найти удовлетворившего меня ответа. Я пытаюсь получить конкретный DbSet на основе имени его типа.

У меня есть следующее:
[assembly: System.Runtime.CompilerServices.InternalsVisibleTo("MyDllAssemblyName")]
[assembly: System.Runtime.CompilerServices.InternalsVisibleTo("MyCallingAssemblyName")]

class MyDbContext : DbContext {

public DbSet A { get; set; }
public DbSet B { get; set; }

public dynamic GetByName_SwitchTest(string name) {
switch (name) {
case "A": return A;
case "B": return B;
}
}

public dynamic GetByName_ReflectionTest(string fullname)
{
Type targetType = Type.GetType(fullname);
var model = GetType()
.GetRuntimeProperties()
.Where(o =>
o.PropertyType.IsGenericType &&
o.PropertyType.GetGenericTypeDefinition() == typeof(DbSet) &&
o.PropertyType.GenericTypeArguments.Contains(targetType))
.FirstOrDefault();
if (null != model)
return model.GetValue(this);
return null;
}
}


У меня нет проблем с получением самого типа, будь то простой переключатель или отражение. Однако мне нужно вернуть тип как динамический, поскольку я не знаю, какой это будет тип DbSet.
Затем где-то в той же сборке я использую это следующим образом:

// MyDbContext MyDbContextInstance..
var model = MyDbContextInstance.GetByName_SwitchTest("A");
var record1 = model.FirstOrDefault(); // It crashes here with RunTimeBinderException


На этом этапе модель содержит экземпляр типа InternalDbSet. Отсюда при любом использовании объекта модели я получаю исключение RunTimeBinderException:
«Microsoft.Data.Entity.Internal.InternalDbSet» не содержит определения для «FirstOrDefault»

Исследуя Интернет, я нашел сообщение в блоге, объясняющее это (Диксит, его блог):


Причина сбоя вызова FirstOrDefault() заключается в том, что информация о типе модели недоступна во время выполнения. Причина, по которой он
недоступен, заключается в том, что анонимные типы не являются общедоступными. Когда метод
возвращает экземпляр этого анонимного типа, он возвращает
System.Object, который ссылается на экземпляр анонимного типа — тип
, информация которого недоступна для основного программу.


И затем он указывает на решение:


Решение на самом деле довольно простое. Все, что нам нужно сделать, это открыть
AssemblyInfo.cs проекта ClassLibrary1 и добавить в него следующую
строку: [assembly:InternalsVisibleTo("assembly-name")]


Я пробовал это решение в своем коде, но оно не работает. Для информации у меня есть решение asp.net 5 с двумя сборками, работающими на dnx dotnet46. Приложение и dll, содержащие все мои модели и DbContext. Однако все соответствующие вызовы, которые я делаю, находятся в dll.

Есть ли у этого решения шанс сработать?
Я что-то упустил?
Буду очень признателен за любые подсказки?

Заранее спасибо

[EDIT]

Я пытался вернуть IQueryable, а не динамический, и я мог бы использовать базовую модель запроса.FirstOrDefault();

Я пытался вернуть IQueryable, а не динамический, и я мог бы использовать базовую модель запроса.FirstOrDefault();

Я пытался вернуть IQueryable, а не динамический, и я мог бы использовать базовую модель запроса.FirstOrDefault();

Я пытался вернуть IQueryable, а не динамический, и я мог бы использовать базовую модель запроса.FirstOrDefault();

code> но прежде всего мне бы хотелось иметь возможность фильтровать и по полю:

var record = model.FirstOrDefault(item => item.MyProperty == true);


Подробнее здесь: https://stackoverflow.com/questions/339 ... ynamically
Ответить

Быстрый ответ

Изменение регистра текста: 
Смайлики
:) :( :oops: :roll: :wink: :muza: :clever: :sorry: :angel: :read: *x)
Ещё смайлики…
   
К этому ответу прикреплено по крайней мере одно вложение.

Если вы не хотите добавлять вложения, оставьте поля пустыми.

Максимально разрешённый размер вложения: 15 МБ.

Вернуться в «C#»