我有一个C#方法返回非常多的对象.这将在Matlab中使用.
namespace MyNameSpace{
public static class MyClass{
public static IEnumerable<MyDataObject> GetVeryLargeResponse(){
while(CheckForSomeFunkyConditionThatsRarelyTrue()){
yield return GetMyNextDataObject();
}
yield break;
}
}
}
Run Code Online (Sandbox Code Playgroud)
在Matlab中我打电话的时候
result = MyClass.GetVeryLargeResponse();
Run Code Online (Sandbox Code Playgroud)
我希望结果是类型的IEnumerable<MyDataObject>,以便能够Enumerator<MyDataObject>通过调用获得result.GetEnumerator().
我在哪里获得result哪种类型MyNameSpace.<GetVeryLargeResponse>d_3没有GetEnumerator()方法可用.我确实看到了一个result超级课程System.Collections.Generic.IEnumerable<MyClass>.
有没有办法可以在Matlab中迭代这个,甚至result可以IEnumerable<MyClass>在Matlab中"转换".
PS
Array/ IListetc是不可行的IQueryable专门处理的.结果确实有一个GetEnumerator()方法 - 它可以用显式接口实现来实现.
如果Matlab不愿意处理它,你总是可以编写自己的映射类型和扩展方法来简化:
public static class Extensions
{
public static EnumerableWrapper<T> Wrap<T>(this IEnumerable<T> source)
{
return new EnumerableWrapper<T>(source);
}
}
public class EnumerableWrapper<T> : IEnumerable<T>
{
private readonly IEnumerable<T> source;
public EnumerableWrapper(IEnumerable<T> source)
{
this.source = source;
}
public IEnumerator<T> GetEnumerator()
{
return new EnumeratorWrapper<T>(source.GetEnumerator());
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
public class EnumeratorWrapper<T> : IEnumerator<T>
{
private readonly IEnumerator<T> source;
public EnumeratorWrapper(IEnumerator<T> source)
{
this.source = source;
}
public T Current { get { return source.Current; } }
object IEnumerator.Current { get { return Current; } }
public bool MoveNext()
{
return source.MoveNext();
}
public void Reset()
{
source.Reset();
}
public void Dispose()
{
source.Dispose();
}
}
Run Code Online (Sandbox Code Playgroud)
然后尝试:
result = MyClass.GetVeryLargeResponse().Wrap();
Run Code Online (Sandbox Code Playgroud)
对Matlab来说似乎很奇怪,不是开箱即用支持这个......
请考虑以下示例:
namespace MyNameSpace {
public class Person {
public string name { get; set; }
public int age { get; set; }
}
public class MyClass {
private static List<Person> people = new List<Person> {
new Person {name = "name1", age=10},
new Person {name = "name2", age=20},
new Person {name = "name3", age=30},
new Person {name = "name4", age=40},
new Person {name = "name5", age=50}
};
public static IEnumerable<Person> GetPeople() {
foreach (var p in people) {
yield return p;
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
%# load my assembly
str = 'C:\path\to\MyNameSpace.dll';
NET.addAssembly(str);
obj = MyNameSpace.MyClass.GetPeople(); %# IEnumerable<Person>
%# call ToArray() extension method: this forces immediate query evaluation
NET.addAssembly('System.Core'); %# contains 'System.Linq' namespace
arr = NET.invokeGenericMethod('System.Linq.Enumerable', 'ToArray', ...
{'MyNameSpace.Person'}, obj);
%# loop through results
for i=1:arr.Length
fprintf('name=%s, age=%d\n', char(arr(i).name), int32(arr(i).age));
end
Run Code Online (Sandbox Code Playgroud)
代码生成输出:
name=name1, age=10
name=name2, age=20
name=name3, age=30
name=name4, age=40
name=name5, age=50
Run Code Online (Sandbox Code Playgroud)
正如您所看到的,我确实将返回的对象转换为数组Person[](我意识到您试图避免).奇怪的是,如果我们看一下类的层次结构:
>> superclasses(obj)
Superclasses for class MyNameSpace.<GetPeople>d__0:
System.Object
System.Collections.Generic.IEnumerable<MyNameSpace*Person>
System.Collections.IEnumerable
System.Collections.Generic.IEnumerator<MyNameSpace*Person>
System.IDisposable
System.Collections.IEnumerator
handle
Run Code Online (Sandbox Code Playgroud)
你可以看到System.Collections.Generic.IEnumerator<Person>,但不知何故,对象似乎没有暴露继承的GetEnumerator方法:
>> methods(obj,'-full')
Methods for class MyNameSpace.<GetPeople>d__0:
MyNameSpace.<GetPeople>d__0 obj <GetPeople>d__0(int32 scalar <>1__state)
logical scalar RetVal Equals(MyNameSpace.<GetPeople>d__0 this, System.Object obj)
int32 scalar RetVal GetHashCode(MyNameSpace.<GetPeople>d__0 this)
System.Type RetVal GetType(MyNameSpace.<GetPeople>d__0 this)
System.String RetVal ToString(MyNameSpace.<GetPeople>d__0 this)
event.listener L addlistener(handle sources, char vector eventname, function_handle scalar callback) % Inherited from handle
event.proplistener L addlistener(handle sources, meta.property properties, char vector eventname, function_handle scalar callback) % Inherited from handle
event.proplistener L addlistener(handle sources, string propertyname, char vector eventname, function_handle scalar callback) % Inherited from handle
event.proplistener L addlistener(handle sources, cell propertynames, char vector eventname, function_handle scalar callback) % Inherited from handle
delete(handle obj) % Inherited from handle
logical TF eq(A, B) % Inherited from handle
handle HM findobj(handle H, varargin) % Inherited from handle
meta.property prop findprop(handle scalar object, string propname) % Inherited from handle
logical TF ge(A, B) % Inherited from handle
logical TF gt(A, B) % Inherited from handle
logical validity isvalid(handle obj) % Inherited from handle
logical TF le(A, B) % Inherited from handle
logical TF lt(A, B) % Inherited from handle
logical TF ne(A, B) % Inherited from handle
notify(handle sources, string eventname) % Inherited from handle
notify(handle sources, string eventname, event.EventData scalar eventdata) % Inherited from handle
Run Code Online (Sandbox Code Playgroud)