编辑
这个问题的简单版本是,如果我有一些object o
,我将如何检查是否某种类型实现IEnumerable<string>
带反射?最初的问题更具体,但对上述问题的答案也同样好。很抱歉,如果我在这个问题上提供太多的详细信息
结束编辑
以下是一个人为的ValueInjecter POC。除了最底层的isCollectionMapping
方法之外,一切都很顺利。当且仅当源属性和目标属性都是实现IEnumerable<respectiveTypes>
的任何对象时,我试图让它返回true。
我已尝试IsAssignableFrom
和IsInstanceOfType
,但似乎都没有效果。
其他一切都有效,因为当我取消注释方法的第二行以明确检查名称“Children”的属性时,它可以正常工作。
注意 - 我知道此示例存在问题。也就是说,我正在尝试检查任何旧IEnumerable<>
,但始终知道返回List<>
;在这一点上,它只是一个愚蠢的概念证明。
[TestClass]
public class UnitTest1 {
[TestMethod]
public void TestMethod1() {
List<string> strings = new List<string>();
Subject S = new Subject() {
id = 1,
SubjectName = "S1",
Children = { new Subject() { id = 2, SubjectName = "S1a" },
new Subject() { id = 3, SubjectName = "S1b", Children = { new Subject() { id = 4} } } }
};
SubjectViewModel VM = (SubjectViewModel)new SubjectViewModel().InjectFrom<CollectionToCollection>(S); ;
Assert.AreEqual(2, VM.Children.Count);
Assert.AreEqual(1, VM.Children.Single(s => s.id == 3).Children.Count);
}
}
public class Subject {
public Subject() {
Children = new List<Subject>();
}
public string SubjectName { get; set; }
public int id { get; set; }
public List<Subject> Children { get; set; }
}
public class SubjectViewModel {
public SubjectViewModel() {
Children = new List<SubjectViewModel>();
}
public string SubjectName { get; set; }
public int id { get; set; }
public List<SubjectViewModel> Children { get; set; }
}
public class CollectionToCollection : Omu.ValueInjecter.ConventionInjection {
protected override bool Match(ConventionInfo c) {
return c.TargetProp.Name == c.SourceProp.Name;
}
protected override object SetValue(ConventionInfo c) {
if (isCollectionMapping(c))
return (c.SourceProp.Value as IEnumerable<Subject>).Select(s => (SubjectViewModel)(new SubjectViewModel().InjectFrom<CollectionToCollection>(s))).ToList();
else
return c.SourceProp.Value;
}
private bool isCollectionMapping(ConventionInfo c) {
return c.SourceProp.Value.GetType().IsInstanceOfType(typeof(IEnumerable<Subject>)) && c.TargetProp.Value.GetType().IsAssignableFrom(typeof(IEnumerable<SubjectViewModel>));
//return c.SourceProp.Name == "Children" && c.TargetProp.Name == "Children";
}
}
答案 0 :(得分:29)
如果我有一些对象
o
,我该怎么做? 检查o
是否属于某种类型 实现IEnumerable<string>
?
简单如下:
o is IEnumerable<string>
顺便说一句,你当前的代码没有工作,因为它正在逆转对可分配性关系的测试(就好像这个方法被称为IsAssignableTo
),即假设:
Bar bar = ...
Foo foo = bar
暗示:
typeof(Bar).IsAssignableFrom(typeof(Foo)) // wrong
实际上,实际意义是:
typeof(Foo).IsAssignableFrom(typeof(Bar))
即,我正在尝试检查任何内容 旧
IEnumerable<>
:
在这种情况下,您需要测试类型是否实现了通用接口的构造版本:
o.GetType()
.GetInterfaces()
.Any(t => t.IsGenericType
&& t.GetGenericTypeDefinition() == typeof(IEnumerable<>))
答案 1 :(得分:10)
这个问题的简单版本是,如果我有一些对象o,我将如何检查o是否是某种类型实现
IEnumerable<string>
?
像这样:
object o = whatever;
bool isSequenceOfStrings = o is IEnumerable<string>;
答案 2 :(得分:5)
我本可以错过一些东西(没有读过你的整个代码示例),但这似乎不需要在这里反思。
如何使用:
if (c.SourceProp.Value is IEnumerable<Subject>)
return true;
如果您不知道具体类型,请使用泛型:
public bool MyFunction<T>(...)
{
if (c.SourceProp.Value is IEnumerable<T>)
return true;
}
或者如果您需要使用界面,请这样做(保存演员表):
var enumerable = c.SourceProp.Value as IEnumerable<string>;
if (enumerable != null)
{
// Use IEnumerable<string>
return true;
}
return false;