ReflectionUtils.cs 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. #region License
  2. // Copyright (c) 2007 James Newton-King
  3. //
  4. // Permission is hereby granted, free of charge, to any person
  5. // obtaining a copy of this software and associated documentation
  6. // files (the "Software"), to deal in the Software without
  7. // restriction, including without limitation the rights to use,
  8. // copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. // copies of the Software, and to permit persons to whom the
  10. // Software is furnished to do so, subject to the following
  11. // conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be
  14. // included in all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  17. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  18. // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  19. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  20. // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  21. // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  22. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  23. // OTHER DEALINGS IN THE SOFTWARE.
  24. #endregion
  25. using System;
  26. using System.Collections.Generic;
  27. using System.Text;
  28. using System.Reflection;
  29. using System.Collections;
  30. using System.ComponentModel;
  31. namespace Newtonsoft.Json.Utilities
  32. {
  33. internal static class ReflectionUtils
  34. {
  35. public static bool IsInstantiatableType(Type t)
  36. {
  37. if (t == null)
  38. throw new ArgumentNullException("t");
  39. if (t.IsAbstract || t.IsInterface || t.IsArray)
  40. return false;
  41. if (!HasDefaultConstructor(t))
  42. return false;
  43. return true;
  44. }
  45. public static bool HasDefaultConstructor(Type t)
  46. {
  47. if (t == null)
  48. throw new ArgumentNullException("t");
  49. return (t.GetConstructor(BindingFlags.Instance, null, Type.EmptyTypes, null) != null);
  50. }
  51. public static bool IsAssignable (Type to, Type from) {
  52. if (to == null)
  53. throw new ArgumentNullException("to");
  54. if (to.IsAssignableFrom (from))
  55. return true;
  56. if (to.IsGenericType && from.IsGenericTypeDefinition)
  57. return to.IsAssignableFrom (from.MakeGenericType (to.GetGenericArguments ()));
  58. return false;
  59. }
  60. public static bool IsSubClass(Type type, Type check)
  61. {
  62. if (type == null || check == null)
  63. return false;
  64. if (type == check)
  65. return true;
  66. if (check.IsInterface)
  67. {
  68. foreach (Type t in type.GetInterfaces())
  69. {
  70. if (IsSubClass(t, check)) return true;
  71. }
  72. }
  73. if (type.IsGenericType && !type.IsGenericTypeDefinition)
  74. {
  75. if (IsSubClass(type.GetGenericTypeDefinition(), check))
  76. return true;
  77. }
  78. return IsSubClass(type.BaseType, check);
  79. }
  80. /// <summary>
  81. /// Gets the type of the typed list's items.
  82. /// </summary>
  83. /// <param name="type">The type.</param>
  84. /// <returns>The type of the typed list's items.</returns>
  85. public static Type GetTypedListItemType(Type type)
  86. {
  87. if (type == null)
  88. throw new ArgumentNullException("type");
  89. if (type.IsArray)
  90. return type.GetElementType ();
  91. else if (type.IsGenericType && typeof (List<>).IsAssignableFrom (type.GetGenericTypeDefinition ()))
  92. return type.GetGenericArguments () [0];
  93. else
  94. throw new Exception ("Bad type");
  95. }
  96. public static Type GetTypedDictionaryValueType(Type type)
  97. {
  98. if (type == null)
  99. throw new ArgumentNullException("type");
  100. Type genDictType = GetGenericDictionary(type);
  101. if (genDictType != null)
  102. return genDictType.GetGenericArguments () [1];
  103. else if (typeof(IDictionary).IsAssignableFrom(type))
  104. return null;
  105. else
  106. throw new Exception("Bad type");
  107. }
  108. static readonly Type GenericDictionaryType = typeof (IDictionary<,>);
  109. public static Type GetGenericDictionary (Type type) {
  110. if (type.IsGenericType && GenericDictionaryType.IsAssignableFrom (type.GetGenericTypeDefinition ()))
  111. return type;
  112. Type[] ifaces = type.GetInterfaces();
  113. if (ifaces != null)
  114. for (int i = 0; i < ifaces.Length; i++) {
  115. Type current = GetGenericDictionary (ifaces [i]);
  116. if (current != null)
  117. return current;
  118. }
  119. return null;
  120. }
  121. public static Type GetMemberUnderlyingType(MemberInfo member)
  122. {
  123. switch (member.MemberType)
  124. {
  125. case MemberTypes.Field:
  126. return ((FieldInfo)member).FieldType;
  127. case MemberTypes.Property:
  128. return ((PropertyInfo)member).PropertyType;
  129. case MemberTypes.Event:
  130. return ((EventInfo)member).EventHandlerType;
  131. default:
  132. throw new ArgumentException("MemberInfo must be if type FieldInfo, PropertyInfo or EventInfo", "member");
  133. }
  134. }
  135. /// <summary>
  136. /// Determines whether the member is an indexed property.
  137. /// </summary>
  138. /// <param name="member">The member.</param>
  139. /// <returns>
  140. /// <c>true</c> if the member is an indexed property; otherwise, <c>false</c>.
  141. /// </returns>
  142. public static bool IsIndexedProperty(MemberInfo member)
  143. {
  144. if (member == null)
  145. throw new ArgumentNullException("member");
  146. PropertyInfo propertyInfo = member as PropertyInfo;
  147. if (propertyInfo != null)
  148. return IsIndexedProperty(propertyInfo);
  149. else
  150. return false;
  151. }
  152. /// <summary>
  153. /// Determines whether the property is an indexed property.
  154. /// </summary>
  155. /// <param name="property">The property.</param>
  156. /// <returns>
  157. /// <c>true</c> if the property is an indexed property; otherwise, <c>false</c>.
  158. /// </returns>
  159. public static bool IsIndexedProperty(PropertyInfo property)
  160. {
  161. if (property == null)
  162. throw new ArgumentNullException("property");
  163. return (property.GetIndexParameters().Length > 0);
  164. }
  165. /// <summary>
  166. /// Gets the member's value on the object.
  167. /// </summary>
  168. /// <param name="member">The member.</param>
  169. /// <param name="target">The target object.</param>
  170. /// <returns>The member's value on the object.</returns>
  171. public static object GetMemberValue(MemberInfo member, object target)
  172. {
  173. switch (member.MemberType)
  174. {
  175. case MemberTypes.Field:
  176. return ((FieldInfo)member).GetValue(target);
  177. case MemberTypes.Property:
  178. try
  179. {
  180. return ((PropertyInfo)member).GetValue(target, null);
  181. }
  182. catch (TargetParameterCountException e)
  183. {
  184. throw new ArgumentException("MemberInfo has index parameters", "member", e);
  185. }
  186. default:
  187. throw new ArgumentException("MemberInfo is not of type FieldInfo or PropertyInfo", "member");
  188. }
  189. }
  190. /// <summary>
  191. /// Sets the member's value on the target object.
  192. /// </summary>
  193. /// <param name="member">The member.</param>
  194. /// <param name="target">The target.</param>
  195. /// <param name="value">The value.</param>
  196. public static void SetMemberValue(MemberInfo member, object target, object value)
  197. {
  198. switch (member.MemberType)
  199. {
  200. case MemberTypes.Field:
  201. ((FieldInfo)member).SetValue(target, value);
  202. break;
  203. case MemberTypes.Property:
  204. ((PropertyInfo)member).SetValue(target, value, null);
  205. break;
  206. default:
  207. throw new ArgumentException("MemberInfo must be if type FieldInfo or PropertyInfo", "member");
  208. }
  209. }
  210. /// <summary>
  211. /// Determines whether the specified MemberInfo can be read.
  212. /// </summary>
  213. /// <param name="member">The MemberInfo to determine whether can be read.</param>
  214. /// <returns>
  215. /// <c>true</c> if the specified MemberInfo can be read; otherwise, <c>false</c>.
  216. /// </returns>
  217. public static bool CanReadMemberValue(MemberInfo member)
  218. {
  219. switch (member.MemberType)
  220. {
  221. case MemberTypes.Field:
  222. return true;
  223. case MemberTypes.Property:
  224. return ((PropertyInfo) member).CanRead;
  225. default:
  226. return false;
  227. }
  228. }
  229. /// <summary>
  230. /// Determines whether the specified MemberInfo can be set.
  231. /// </summary>
  232. /// <param name="member">The MemberInfo to determine whether can be set.</param>
  233. /// <returns>
  234. /// <c>true</c> if the specified MemberInfo can be set; otherwise, <c>false</c>.
  235. /// </returns>
  236. public static bool CanSetMemberValue(MemberInfo member)
  237. {
  238. switch (member.MemberType)
  239. {
  240. case MemberTypes.Field:
  241. return true;
  242. case MemberTypes.Property:
  243. return ((PropertyInfo)member).CanWrite;
  244. default:
  245. return false;
  246. }
  247. }
  248. public static IEnumerable<MemberInfo> GetFieldsAndProperties (Type type, BindingFlags bindingAttr) {
  249. MemberInfo [] members = type.GetFields (bindingAttr);
  250. for (int i = 0; i < members.Length; i++)
  251. yield return members [i];
  252. members = type.GetProperties (bindingAttr);
  253. for (int i = 0; i < members.Length; i++)
  254. yield return members [i];
  255. }
  256. }
  257. }