Подтвердить что ты не робот

Сначала отобразить каркас структуры частной собственности

Я использую EF 4.1 и искал хороший обходной путь из-за отсутствия поддержки enum. Свойство back для int кажется логичным.

    [Required]
    public VenueType Type
    {
        get { return (VenueType) TypeId; }
        set { TypeId = (int) value; }
    }

    private int TypeId { get; set; }

Но как я могу сделать это свойство частным и по-прежнему его отображать. Другими словами:

Как я могу сначала скопировать частную собственность с использованием кода EF 4.1?

4b9b3361

Ответ 1

вы не можете перенести частные объекты в EF-код. Вы можете попробовать изменить его на protected и настроить его в классе, унаследованном от EntityConfiguration.
Edit
Теперь он изменен, см. fooobar.com/questions/124968/...

Ответ 2

Здесь соглашение, которое вы можете использовать в EF 6+ для сопоставления выбранных непубличных свойств (просто добавьте атрибут [Column] к свойству).

В вашем случае вы измените TypeId на:

    [Column]
    private int TypeId { get; set; }

В вашем DbContext.OnModelCreating вам необходимо зарегистрировать соглашение:

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Conventions.Add(new NonPublicColumnAttributeConvention());
    }

Наконец, здесь соглашение:

/// <summary>
/// Convention to support binding private or protected properties to EF columns.
/// </summary>
public sealed class NonPublicColumnAttributeConvention : Convention
{

    public NonPublicColumnAttributeConvention()
    {
        Types().Having(NonPublicProperties)
               .Configure((config, properties) =>
                          {
                              foreach (PropertyInfo prop in properties)
                              {
                                  config.Property(prop);
                              }
                          });
    }

    private IEnumerable<PropertyInfo> NonPublicProperties(Type type)
    {
        var matchingProperties = type.GetProperties(BindingFlags.SetProperty | BindingFlags.GetProperty | BindingFlags.NonPublic | BindingFlags.Instance)
                                     .Where(propInfo => propInfo.GetCustomAttributes(typeof(ColumnAttribute), true).Length > 0)
                                     .ToArray();
        return matchingProperties.Length == 0 ? null : matchingProperties;
    }
}

Ответ 3

Другим обходным решением может быть установка вашего поля как внутреннего:

    [NotMapped]
    public dynamic FacebookMetadata {
        get
        {
            return JObject.Parse(this.FacebookMetadataDb);
        }
        set
        {
            this.FacebookMetadataDb = JsonConvert.SerializeObject(value);
        }
    }

    ///this one
    internal string FacebookMetadataDb { get; set; }

и добавьте его в модель тура:

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Conventions.Remove<System.Data.Entity.ModelConfiguration.Conventions.ManyToManyCascadeDeleteConvention>();

        ///here
        modelBuilder.Entity<FacebookPage>().Property(p => p.FacebookMetadataDb);

        base.OnModelCreating(modelBuilder);
    }

Ответ 4

Расширение ответа @crimbo выше (fooobar.com/questions/124960/...), здесь мое изменение включает публичные свойства с частными получателями:

private IEnumerable<PropertyInfo> NonPublicProperties(Type type)
{
    var matchingProperties = type.GetProperties(BindingFlags.SetProperty | BindingFlags.GetProperty | BindingFlags.NonPublic | BindingFlags.Instance)
                                 .Where(propInfo => propInfo.GetCustomAttributes(typeof(ColumnAttribute), true).Length > 0)
                                 .Union(
                                        type.GetProperties(BindingFlags.SetProperty | BindingFlags.GetProperty | BindingFlags.Public | BindingFlags.Instance)
                                            .Where(propInfo => propInfo.GetCustomAttributes(typeof(ColumnAttribute), true).Length > 0
                                                               && propInfo.GetGetMethod().IsNull())
                                  )
                                 .ToArray();
    return matchingProperties.Length == 0 ? null : matchingProperties;
}

Ответ 5

Еще один способ справиться с этим - определить конфигурацию настраиваемого объекта и добавить для этого привязку.

В вашем классе добавьте класс, который наследуется от EntityTypeConfiguration (это можно найти в System.Data.Entity.ModelConfiguration)

public partial class Report : Entity<int>
    {
        //Has to be a property
        private string _Tags {get; set;}

        [NotMapped]
        public string[] Tags
        {
            get => _Tags == null ? null : JsonConvert.DeserializeObject<string[]>(_Tags);
            set => _Tags = JsonConvert.SerializeObject(value);
        }

        [MaxLength(100)]
        public string Name { get; set; }

        [MaxLength(250)]
        public string Summary { get; set; }

        public string JsonData { get; set; }

        public class ReportConfiguration: EntityTypeConfiguration<Report>
        {
            public ReportConfiguration()
            {
                Property(p => p._tags).HasColumnName("Tags");
            }
        }
    }

В вашем контексте добавьте следующее:

using Models.ReportBuilder;
public partial class ReportBuilderContext:DbContext
{
    public DbSet<Report> Reports { get; set; }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Configurations.Add(new Report.ReportConfiguration());
        base.OnModelCreating(modelBuilder);
    }
}

Желаю сказать, что я нашел это самостоятельно, но наткнулся на него здесь: https://romiller.com/2012/10/01/mapping-to-private-properties-with-code-first/