C #에서 리플렉션을 사용하여 문자열에서 속성 값 가져 오기
내 코드에서 Reflection 1 예제를 사용하여 데이터 변환을 구현하려고 합니다.
GetSourceValue
기능은 여러 유형을 비교하는 스위치가 있습니다, 그러나 나는 이러한 유형 및 속성을 제거해야 할 GetSourceValue
매개 변수로 단 하나의 문자열을 사용하여 속성 값을 얻는다. 문자열에 클래스와 속성을 전달하고 속성 값을 확인하고 싶습니다.
이것이 가능한가?
public static object GetPropValue(object src, string propName)
{
return src.GetType().GetProperty(propName).GetValue(src, null);
}
물론, 유효성 검사와 기타 사항을 추가하고 싶겠지 만 그게 요점입니다.
다음과 같은 것은 어떻습니까?
public static Object GetPropValue(this Object obj, String name) {
foreach (String part in name.Split('.')) {
if (obj == null) { return null; }
Type type = obj.GetType();
PropertyInfo info = type.GetProperty(part);
if (info == null) { return null; }
obj = info.GetValue(obj, null);
}
return obj;
}
public static T GetPropValue<T>(this Object obj, String name) {
Object retval = GetPropValue(obj, name);
if (retval == null) { return default(T); }
// throws InvalidCastException if types are incompatible
return (T) retval;
}
이렇게하면 다음과 같이 단일 문자열을 사용하여 속성으로 내려갈 수 있습니다.
DateTime now = DateTime.Now;
int min = GetPropValue<int>(now, "TimeOfDay.Minutes");
int hrs = now.GetPropValue<int>("TimeOfDay.Hours");
이러한 메서드를 정적 메서드 또는 확장으로 사용할 수 있습니다.
추가 Class
:
public class Foo
{
public object this[string propertyName]
{
get { return this.GetType().GetProperty(propertyName).GetValue(this, null); }
set { this.GetType().GetProperty(propertyName).SetValue(this, value, null); }
}
public string Bar { get; set; }
}
그런 다음 다음과 같이 사용할 수 있습니다.
Foo f = new Foo();
// Set
f["Bar"] = "asdf";
// Get
string s = (string)f["Bar"];
네임 스페이스 ( )를 사용하는 CallByName
것은 어떻습니까? 리플렉션을 사용하여 일반 개체, COM 개체 및 동적 개체의 속성, 필드 및 메서드를 가져옵니다.Microsoft.VisualBasic
Microsoft.VisualBasic.dll
using Microsoft.VisualBasic;
using Microsoft.VisualBasic.CompilerServices;
그리고
Versioned.CallByName(this, "method/function/prop name", CallType.Get).ToString();
jheddings의 훌륭한 대답. 속성 이름이 property1.property2 [X] .property3이 될 수 있도록 집계 된 배열 또는 개체 컬렉션의 참조를 허용하도록 개선하고 싶습니다.
public static object GetPropertyValue(object srcobj, string propertyName)
{
if (srcobj == null)
return null;
object obj = srcobj;
// Split property name to parts (propertyName could be hierarchical, like obj.subobj.subobj.property
string[] propertyNameParts = propertyName.Split('.');
foreach (string propertyNamePart in propertyNameParts)
{
if (obj == null) return null;
// propertyNamePart could contain reference to specific
// element (by index) inside a collection
if (!propertyNamePart.Contains("["))
{
PropertyInfo pi = obj.GetType().GetProperty(propertyNamePart);
if (pi == null) return null;
obj = pi.GetValue(obj, null);
}
else
{ // propertyNamePart is areference to specific element
// (by index) inside a collection
// like AggregatedCollection[123]
// get collection name and element index
int indexStart = propertyNamePart.IndexOf("[")+1;
string collectionPropertyName = propertyNamePart.Substring(0, indexStart-1);
int collectionElementIndex = Int32.Parse(propertyNamePart.Substring(indexStart, propertyNamePart.Length-indexStart-1));
// get collection object
PropertyInfo pi = obj.GetType().GetProperty(collectionPropertyName);
if (pi == null) return null;
object unknownCollection = pi.GetValue(obj, null);
// try to process the collection as array
if (unknownCollection.GetType().IsArray)
{
object[] collectionAsArray = unknownCollection as object[];
obj = collectionAsArray[collectionElementIndex];
}
else
{
// try to process the collection as IList
System.Collections.IList collectionAsList = unknownCollection as System.Collections.IList;
if (collectionAsList != null)
{
obj = collectionAsList[collectionElementIndex];
}
else
{
// ??? Unsupported collection type
}
}
}
}
return obj;
}
Ed S 의 코드를 사용 하면
보호 수준으로 인해 'ReflectionExtensions.GetProperty (Type, string)'에 액세스 할 수 없습니다.
GetProperty()
Xamarin.Forms에서 사용할 수없는 것 같습니다 . TargetFrameworkProfile
입니다 Profile7
내 휴대용 클래스 라이브러리 (.NET 프레임 워크 4.5, 윈도우 8, ASP.NET 코어 1.0, Xamarin.Android, Xamarin.iOS, Xamarin.iOS 클래식)에서.
이제 작동하는 해결책을 찾았습니다.
using System.Linq;
using System.Reflection;
public static object GetPropValue(object source, string propertyName)
{
var property = source.GetType().GetRuntimeProperties().FirstOrDefault(p => string.Equals(p.Name, propertyName, StringComparison.OrdinalIgnoreCase));
return property?.GetValue(source);
}
중첩 속성 토론에 대해 DataBinder.Eval Method (Object, String)
아래와 같이 사용하면 모든 반사 요소를 피할 수 있습니다.
var value = DataBinder.Eval(DateTime.Now, "TimeOfDay.Hours");
물론 System.Web
어셈블리에 대한 참조를 추가해야 하지만 이는 큰 문제가 아닐 것입니다.
호출 방법이 .NET Standard (1.6 기준)에서 변경되었습니다. 또한 C # 6의 null 조건부 연산자를 사용할 수 있습니다.
using System.Reflection;
public static object GetPropValue(object src, string propName)
{
return src.GetType().GetRuntimeProperty(propName)?.GetValue(src);
}
System.Reflection 네임 스페이스의 PropertyInfo를 사용 합니다. 리플렉션은 우리가 접근하려는 속성에 상관없이 잘 컴파일됩니다. 런타임 중에 오류가 발생합니다.
public static object GetObjProperty(object obj, string property)
{
Type t = obj.GetType();
PropertyInfo p = t.GetProperty("Location");
Point location = (Point)p.GetValue(obj, null);
return location;
}
객체의 Location 속성을 가져 오는 것이 잘 작동합니다.
Label1.Text = GetObjProperty(button1, "Location").ToString();
Location : {X = 71, Y = 27}를 얻을 수 있습니다. 또한 location.X 또는 location.Y를 같은 방식으로 반환 할 수 있습니다.
public static List<KeyValuePair<string, string>> GetProperties(object item) //where T : class
{
var result = new List<KeyValuePair<string, string>>();
if (item != null)
{
var type = item.GetType();
var properties = type.GetProperties(BindingFlags.Public | BindingFlags.Instance);
foreach (var pi in properties)
{
var selfValue = type.GetProperty(pi.Name).GetValue(item, null);
if (selfValue != null)
{
result.Add(new KeyValuePair<string, string>(pi.Name, selfValue.ToString()));
}
else
{
result.Add(new KeyValuePair<string, string>(pi.Name, null));
}
}
}
return result;
}
이것은 목록에있는 값으로 모든 속성을 가져 오는 방법입니다.
다음 코드는 개체의 인스턴스에 포함 된 모든 속성 이름 및 값의 전체 계층 구조를 표시하기위한 재귀 메서드입니다. 이 방법은 GetPropertyValue()
이 스레드에서 위 의 AlexD 답변 의 단순화 된 버전을 사용합니다 . 이 토론 스레드 덕분에이 작업을 수행하는 방법을 알아낼 수있었습니다!
예를 들어,이 메서드를 사용 WebService
하여 다음과 같이 메서드를 호출하여 응답 에있는 모든 속성의 폭발 또는 덤프를 표시 합니다.
PropertyValues_byRecursion("Response", response, false);
public static object GetPropertyValue(object srcObj, string propertyName)
{
if (srcObj == null)
{
return null;
}
PropertyInfo pi = srcObj.GetType().GetProperty(propertyName.Replace("[]", ""));
if (pi == null)
{
return null;
}
return pi.GetValue(srcObj);
}
public static void PropertyValues_byRecursion(string parentPath, object parentObj, bool showNullValues)
{
/// Processes all of the objects contained in the parent object.
/// If an object has a Property Value, then the value is written to the Console
/// Else if the object is a container, then this method is called recursively
/// using the current path and current object as parameters
// Note: If you do not want to see null values, set showNullValues = false
foreach (PropertyInfo pi in parentObj.GetType().GetTypeInfo().GetProperties())
{
// Build the current object property's namespace path.
// Recursion extends this to be the property's full namespace path.
string currentPath = parentPath + "." + pi.Name;
// Get the selected property's value as an object
object myPropertyValue = GetPropertyValue(parentObj, pi.Name);
if (myPropertyValue == null)
{
// Instance of Property does not exist
if (showNullValues)
{
Console.WriteLine(currentPath + " = null");
// Note: If you are replacing these Console.Write... methods callback methods,
// consider passing DBNull.Value instead of null in any method object parameters.
}
}
else if (myPropertyValue.GetType().IsArray)
{
// myPropertyValue is an object instance of an Array of business objects.
// Initialize an array index variable so we can show NamespacePath[idx] in the results.
int idx = 0;
foreach (object business in (Array)myPropertyValue)
{
if (business == null)
{
// Instance of Property does not exist
// Not sure if this is possible in this context.
if (showNullValues)
{
Console.WriteLine(currentPath + "[" + idx.ToString() + "]" + " = null");
}
}
else if (business.GetType().IsArray)
{
// myPropertyValue[idx] is another Array!
// Let recursion process it.
PropertyValues_byRecursion(currentPath + "[" + idx.ToString() + "]", business, showNullValues);
}
else if (business.GetType().IsSealed)
{
// Display the Full Property Path and its Value
Console.WriteLine(currentPath + "[" + idx.ToString() + "] = " + business.ToString());
}
else
{
// Unsealed Type Properties can contain child objects.
// Recurse into my property value object to process its properties and child objects.
PropertyValues_byRecursion(currentPath + "[" + idx.ToString() + "]", business, showNullValues);
}
idx++;
}
}
else if (myPropertyValue.GetType().IsSealed)
{
// myPropertyValue is a simple value
Console.WriteLine(currentPath + " = " + myPropertyValue.ToString());
}
else
{
// Unsealed Type Properties can contain child objects.
// Recurse into my property value object to process its properties and child objects.
PropertyValues_byRecursion(currentPath, myPropertyValue, showNullValues);
}
}
}
public static TValue GetFieldValue<TValue>(this object instance, string name)
{
var type = instance.GetType();
var field = type.GetFields(BindingFlags.NonPublic | BindingFlags.Static | BindingFlags.Instance).FirstOrDefault(e => typeof(TValue).IsAssignableFrom(e.FieldType) && e.Name == name);
return (TValue)field?.GetValue(instance);
}
public static TValue GetPropertyValue<TValue>(this object instance, string name)
{
var type = instance.GetType();
var field = type.GetProperties(BindingFlags.NonPublic | BindingFlags.Static | BindingFlags.Instance).FirstOrDefault(e => typeof(TValue).IsAssignableFrom(e.PropertyType) && e.Name == name);
return (TValue)field?.GetValue(instance);
}
public class YourClass
{
//Add below line in your class
public object this[string propertyName] => GetType().GetProperty(propertyName)?.GetValue(this, null);
public string SampleProperty { get; set; }
}
//And you can get value of any property like this.
var value = YourClass["SampleProperty"];
다음은 중첩 경로를 알려주는 문자열이 필요하지 않은 중첩 속성을 찾는 또 다른 방법입니다. 단일 속성 방법에 대해 Ed S.에게 감사드립니다.
public static T FindNestedPropertyValue<T, N>(N model, string propName) {
T retVal = default(T);
bool found = false;
PropertyInfo[] properties = typeof(N).GetProperties();
foreach (PropertyInfo property in properties) {
var currentProperty = property.GetValue(model, null);
if (!found) {
try {
retVal = GetPropValue<T>(currentProperty, propName);
found = true;
} catch { }
}
}
if (!found) {
throw new Exception("Unable to find property: " + propName);
}
return retVal;
}
public static T GetPropValue<T>(object srcObject, string propName) {
return (T)srcObject.GetType().GetProperty(propName).GetValue(srcObject, null);
}
어떤 객체를 검사하는지 절대 언급하지 않으며, 주어진 객체를 참조하는 객체를 거부하므로 정적 객체를 의미한다고 가정합니다.
using System.Reflection;
public object GetPropValue(string prop)
{
int splitPoint = prop.LastIndexOf('.');
Type type = Assembly.GetEntryAssembly().GetType(prop.Substring(0, splitPoint));
object obj = null;
return type.GetProperty(prop.Substring(splitPoint + 1)).GetValue(obj, null);
}
검사중인 객체를 지역 변수로 표시했습니다 obj
. null
정적을 의미하고 그렇지 않으면 원하는대로 설정합니다. 또한는 GetEntryAssembly()
"실행중인"어셈블리를 가져 오는 데 사용할 수있는 몇 가지 방법 중 하나입니다. 유형을로드하는 데 어려움이있는 경우이 방법을 사용하는 것이 좋습니다.
아래 방법은 나에게 완벽하게 작동합니다.
class MyClass {
public string prop1 { set; get; }
public object this[string propertyName]
{
get { return this.GetType().GetProperty(propertyName).GetValue(this, null); }
set { this.GetType().GetProperty(propertyName).SetValue(this, value, null); }
}
}
속성 값을 얻으려면 :
MyClass t1 = new MyClass();
...
string value = t1["prop1"].ToString();
속성 값을 설정하려면 :
t1["prop1"] = value;
Dim NewHandle As YourType = CType(Microsoft.VisualBasic.CallByName(ObjectThatContainsYourVariable, "YourVariableName", CallType), YourType)
짧은 방법 ....
var a = new Test { Id = 1 , Name = "A" , date = DateTime.Now};
var b = new Test { Id = 1 , Name = "AXXX", date = DateTime.Now };
var compare = string.Join("",a.GetType().GetProperties().Select(x => x.GetValue(a)).ToArray())==
string.Join("",b.GetType().GetProperties().Select(x => x.GetValue(b)).ToArray());
jheddings 및 AlexD는 모두 속성 문자열을 해결하는 방법에 대한 훌륭한 답을 썼다. 나는 그 목적을 위해 전용 라이브러리를 작성했기 때문에 믹스에 내 것을 던지고 싶습니다.
Pather.CSharp 의 메인 클래스는Resolver
. 기본적으로 속성, 배열 및 사전 항목을 확인할 수 있습니다.
예를 들어 다음과 같은 개체가있는 경우
var o = new { Property1 = new { Property2 = "value" } };
을 얻으려면 Property2
다음과 같이 할 수 있습니다.
IResolver resolver = new Resolver();
var path = "Property1.Property2";
object result = r.Resolve(o, path);
//=> "value"
이것이 해결할 수있는 경로의 가장 기본적인 예입니다. 그 밖에 무엇을 할 수 있는지, 어떻게 확장 할 수 있는지 알고 싶다면 Github 페이지로 이동하세요 .
Heleonix.Reflection 라이브러리를 살펴보십시오 . 경로로 멤버를 가져 오거나 설정 / 호출하거나 리플렉션보다 빠른 getter / setter (대리자로 컴파일 된 람다)를 만들 수 있습니다. 예를 들면 :
var success = Reflector.Get(DateTime.Now, null, "Date.Year", out int value);
또는 getter를 한 번 만들고 재사용을 위해 캐시합니다 (더 성능이 좋지만 중간 멤버가 null 인 경우 NullReferenceException을 throw 할 수 있음).
var getter = Reflector.CreateGetter<DateTime, int>("Date.Year", typeof(DateTime));
getter(DateTime.Now);
또는 List<Action<object, object>>
다른 getter 를 생성하려면 컴파일 된 델리게이트에 대한 기본 유형을 지정하기 만하면됩니다 (유형 변환은 컴파일 된 람다에 추가됩니다).
var getter = Reflector.CreateGetter<object, object>("Date.Year", typeof(DateTime));
getter(DateTime.Now);
여기 내 해결책이 있습니다. 또한 COM 개체와 함께 작동하며 COM 개체에서 컬렉션 / 배열 항목에 액세스 할 수 있습니다.
public static object GetPropValue(this object obj, string name)
{
foreach (string part in name.Split('.'))
{
if (obj == null) { return null; }
Type type = obj.GetType();
if (type.Name == "__ComObject")
{
if (part.Contains('['))
{
string partWithoundIndex = part;
int index = ParseIndexFromPropertyName(ref partWithoundIndex);
obj = Versioned.CallByName(obj, partWithoundIndex, CallType.Get, index);
}
else
{
obj = Versioned.CallByName(obj, part, CallType.Get);
}
}
else
{
PropertyInfo info = type.GetProperty(part);
if (info == null) { return null; }
obj = info.GetValue(obj, null);
}
}
return obj;
}
private static int ParseIndexFromPropertyName(ref string name)
{
int index = -1;
int s = name.IndexOf('[') + 1;
int e = name.IndexOf(']');
if (e < s)
{
throw new ArgumentException();
}
string tmp = name.Substring(s, e - s);
index = Convert.ToInt32(tmp);
name = name.Substring(0, s - 1);
return index;
}
다른 답변을 바탕으로 얻은 것입니다. 오류 처리를 너무 구체적으로 만드는 데 약간의 과잉.
public static T GetPropertyValue<T>(object sourceInstance, string targetPropertyName, bool throwExceptionIfNotExists = false)
{
string errorMsg = null;
try
{
if (sourceInstance == null || string.IsNullOrWhiteSpace(targetPropertyName))
{
errorMsg = $"Source object is null or property name is null or whitespace. '{targetPropertyName}'";
Log.Warn(errorMsg);
if (throwExceptionIfNotExists)
throw new ArgumentException(errorMsg);
else
return default(T);
}
Type returnType = typeof(T);
Type sourceType = sourceInstance.GetType();
PropertyInfo propertyInfo = sourceType.GetProperty(targetPropertyName, returnType);
if (propertyInfo == null)
{
errorMsg = $"Property name '{targetPropertyName}' of type '{returnType}' not found for source object of type '{sourceType}'";
Log.Warn(errorMsg);
if (throwExceptionIfNotExists)
throw new ArgumentException(errorMsg);
else
return default(T);
}
return (T)propertyInfo.GetValue(sourceInstance, null);
}
catch(Exception ex)
{
errorMsg = $"Problem getting property name '{targetPropertyName}' from source instance.";
Log.Error(errorMsg, ex);
if (throwExceptionIfNotExists)
throw;
}
return default(T);
}
'Program Tip' 카테고리의 다른 글
자바 문자열을 날짜로 변환 (0) | 2020.09.28 |
---|---|
차이점은 무엇입니까 (0) | 2020.09.28 |
십진수 대 두 배! (0) | 2020.09.28 |
Gmail을 통해 .NET에서 이메일 보내기 (0) | 2020.09.28 |
MySQL Datetime 열의 기본값을 어떻게 설정합니까? (0) | 2020.09.28 |