using System; using System.Collections.Concurrent; using System.Linq.Expressions; using System.Reflection; namespace Marr.Data.Reflection { public class SimpleReflectionStrategy : IReflectionStrategy { private static readonly ConcurrentDictionary MemberCache = new ConcurrentDictionary(); private static readonly ConcurrentDictionary GetterCache = new ConcurrentDictionary(); private static readonly ConcurrentDictionary SetterCache = new ConcurrentDictionary(); private static MemberInfo GetMember(Type entityType, string name) { MemberInfo member; var key = entityType.FullName + name; if (!MemberCache.TryGetValue(key, out member)) { member = entityType.GetMember(name, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)[0]; MemberCache[key] = member; } return member; } /// /// Gets an entity field value by name. /// public object GetFieldValue(object entity, string fieldName) { var member = GetMember(entity.GetType(), fieldName); if (member.MemberType == MemberTypes.Field) { return (member as FieldInfo).GetValue(entity); } if (member.MemberType == MemberTypes.Property) { return BuildGetter(entity.GetType(), fieldName)(entity); } throw new DataMappingException(string.Format("The DataMapper could not get the value for {0}.{1}.", entity.GetType().Name, fieldName)); } /// /// Instantiates a type using the FastReflector library for increased speed. /// /// /// public object CreateInstance(Type type) { return Activator.CreateInstance(type); } public GetterDelegate BuildGetter(Type type, string memberName) { GetterDelegate getter; var key = type.FullName + memberName; if (!GetterCache.TryGetValue(key, out getter)) { getter = GetPropertyGetter((PropertyInfo)GetMember(type, memberName)); } return getter; } public SetterDelegate BuildSetter(Type type, string memberName) { SetterDelegate setter; var key = type.FullName + memberName; if (!SetterCache.TryGetValue(key, out setter)) { setter = GetPropertySetter((PropertyInfo)GetMember(type, memberName)); } return setter; } private static SetterDelegate GetPropertySetter(PropertyInfo propertyInfo) { var propertySetMethod = propertyInfo.GetSetMethod(); if (propertySetMethod == null) return null; #if NO_EXPRESSIONS return (o, convertedValue) => { propertySetMethod.Invoke(o, new[] { convertedValue }); return; }; #else var instance = Expression.Parameter(typeof(object), "i"); var argument = Expression.Parameter(typeof(object), "a"); var instanceParam = Expression.Convert(instance, propertyInfo.DeclaringType); var valueParam = Expression.Convert(argument, propertyInfo.PropertyType); var setterCall = Expression.Call(instanceParam, propertyInfo.GetSetMethod(), valueParam); return Expression.Lambda(setterCall, instance, argument).Compile(); #endif } private static GetterDelegate GetPropertyGetter(PropertyInfo propertyInfo) { var getMethodInfo = propertyInfo.GetGetMethod(); if (getMethodInfo == null) return null; #if NO_EXPRESSIONS return o => propertyInfo.GetGetMethod().Invoke(o, new object[] { }); #else try { var oInstanceParam = Expression.Parameter(typeof(object), "oInstanceParam"); var instanceParam = Expression.Convert(oInstanceParam, propertyInfo.DeclaringType); var exprCallPropertyGetFn = Expression.Call(instanceParam, getMethodInfo); var oExprCallPropertyGetFn = Expression.Convert(exprCallPropertyGetFn, typeof(object)); var propertyGetFn = Expression.Lambda ( oExprCallPropertyGetFn, oInstanceParam ).Compile(); return propertyGetFn; } catch (Exception ex) { Console.Write(ex.Message); throw; } #endif } } }