Отражение, свойства и атрибуты - изучение архитектуры ORM - PullRequest
1 голос
/ 13 января 2011

Я просто играю с атрибутами, которые содержат некоторую информацию о конкретном поле базы данных.Я уже создал класс FieldAttribute, и он имеет информацию по умолчанию, такую ​​как имя, тип и т. Д.

Мой вопрос: возможно ли создать один метод для извлечения этой информации и встраивания в свойства?Я думаю, что я могу лучше общаться с кодом:

[AttributeUsage(AttributeTargets.Property,AllowMultiple=false,Inherited=true)]
public class FieldAttribute:Attribute
{
    protected string _FieldName;
    protected esriFieldType _FieldType;
    protected bool _NotNullable;
    protected bool _IsPrimary;
    protected bool _IsForeign;
    protected int _Index;

    public virtual string FieldName
    {
        get { return this._FieldName; }
        set { this._FieldName = value; }
    }

    public virtual esriFieldType FieldType
    {
        get { return this._FieldType; }
        set { this._FieldType = value; }
    }

    public virtual bool NotNullable
    {
        get { return this._NotNullable; }
        set { this._NotNullable = value; }
    }

    public virtual int Index
    {
        get { return this._Index; }
        set { this._Index = value; }
    }

    public FieldAttribute(string fieldName, esriFieldType fieldType,int position)
    {
        this.FieldName = fieldName;
        this.FieldType = fieldType;
        _IsPrimary = false;
        _IsForeign = false;
        Index = position;
    }
}

// my main abstract class
public abstract class ObjectWrapper:IObjectWrapper
{
    protected int _ObjectId;
    protected IObject _UnderlyingObject;

    public virtual IObject UnderlyingObject
    {
        get { return this._UnderlyingObject; }
    }

    public virtual int ObjectId
    {
        get { return this._ObjectId; }
    }

    public virtual void Store()
    {
        try
        {
            _UnderlyingObject.Store();
        }
        catch (COMException comEx)
        {
            // log com exception
        }
        catch (Exception ex)
        {
            // log
        }
    }

    // this method retrieves field information
    private FieldAttribute GetFieldAttribute(string propertyName)
    {
        FieldAttribute propertyAttribute = null;

        try
        {
            PropertyInfo propInfo = this.GetType().GetProperty(propertyName);
            object[] attributes = propInfo.GetCustomAttributes(typeof(FieldAttribute), true);
            if (attributes.Length == 1)
                propertyAttribute = (FieldAttribute)attributes[0];
        }
        catch (AmbiguousMatchException ambEx)
        {
            // log
        }
        catch (ArgumentException argEx)
        {
            // log
        }

        return propertyAttribute;
    }
}

// my concrete class
public class Foo:ObjectWrapper
{        
    [Field("FOO_NAME",esriFieldType.esriFieldTypeString,1);
    public string FooName { get; set; }

    [Field("FOO_AGE",esriFieldType.esriFieldTypeInteger,2);
    public int FooAge { get; set; }
}

Что я хочу сделать здесь, это создать общий GetMethod, который использует значения атрибутов поля для извлечения данных из базы данных / _underlyingObject и просто "заполнить"getter и setter for me.EG:

когда я вызываю Foo.FooName, получатель проверяет атрибуты, передавая имя получателя методу GetFieldAttribute.

Как мне выполнитьthis?

Идея состоит в том, что когда есть небольшая структура, она станет простым "поставщиком базы данных ESRI" для данных.

Мне очень жаль, что я не могу объяснить это должным образом.

Спасибо за помощь.

G.

1 Ответ

2 голосов
/ 13 января 2011

Если вы не планируете создание экземпляра своего Foo-объекта с помощью какого-либо метода извлечения из репозитория данных, это потребует динамической компиляции для добавления необходимого в ваш метод получения и установки.

Вместо этого я рекомендую вам взглянуть наДля такого шаблона

var foo = DataRepository.GetObject<Foo>(some_id);

сигнатура метода GetObject будет

public static T GetObject<T>(object id) where T: new()

. В методе GetObject вы будете размышлять над типом, найдя все его атрибуты Field и сопоставив их сфактический вызов базы данных.Это очень простой и распространенный подход, и его можно реализовать следующим образом:

public static IEnumerable<Tuple<PropertyInfo, FieldAttribute>> GetFieldAttributes<T>()
{
    var properties = typeof(T).GetProperties();
    foreach (var prop in properties)
    {
        var attribute = prop.GetCustomAttributes(typeof(FieldAttribute), true).FirstOrDefault();
        if (attribute != null)
        {
            yield return new Tuple<PropertyInfo, FieldAttribute>(prop, attribute);
        }
    }
}

Теперь со списком атрибутов og вы можете создать строку типа sql

var columns = GetFieldAttributes<T>().Select(t => t.Item2.FieldName);
var sql = "SELECT "+ String.Join("," columns) +" FROM table WHERE id = "+ id;

var result = ... execute the query and return ie a datareader

Получивможет создать экземпляр Foo, заполнить его свойства и вернуть экземпляр

var instance = new T();

result.Read()
var dictionary = new Dictionary<string, object>();

// get the columns returned from database
for (int i = 0; i < reader.FieldCount; i++)
{
    string fieldName = reader.GetName(i);
    object value = reader.GetValue(i);

    dictionary.Add(fieldName, value);
}

// bind the columns from database to the properties of our object
foreach (var c in columns)
{
    var attr = c.Item2;
    var value = dictionary[attr.FieldName];

    if (value is DBNull)
    {
        value = null;
    }

    typeof(T).InvokeMember(c.Item1.Name, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.SetProperty | BindingFlags.Instance, null, instance, new[] { value });
}

// return the new object with all its values filled out
return instance
...