- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
不确定如何调用它,但假设您有一个看起来像这样的类:
class Person
{
public string Name;
public IEnumerable<Person> Friends;
}
然后你有一个人,你想递归地“展开”这个结构,所以你最终得到一个没有重复的所有人的列表。
你会怎么做?我已经做了一些似乎可行的东西,但我很好奇其他人会怎么做,尤其是如果 Linq 有一些内置的东西,你可以巧妙地使用它来解决这个小问题:)
这是我的解决方案:
public static IEnumerable<T> SelectRecursive<T>(this IEnumerable<T> subjects, Func<T, IEnumerable<T>> selector)
{
// Stop if subjects are null or empty
if(subjects == null)
yield break;
// For each subject
foreach(var subject in subjects)
{
// Yield it
yield return subject;
// Then yield all its decendants
foreach (var decendant in SelectRecursive(selector(subject), selector))
yield return decendant;
}
}
可以这样使用:
var people = somePerson.SelectRecursive(x => x.Friends);
最佳答案
我认为 LINQ 中没有内置任何功能来执行此操作。
像这样递归地做是有问题的——你最终会创建大量的迭代器。如果树很深,这可能效率很低。 Wes Dyer和 Eric Lippert都写了关于此的博客。
您可以通过删除直接递归来消除这种低效率。例如:
public static IEnumerable<T> SelectRecursive<T>(this IEnumerable<T> subjects,
Func<T, IEnumerable<T>> selector)
{
if (subjects == null)
{
yield break;
}
Queue<T> stillToProcess = new Queue<T>(subjects);
while (stillToProcess.Count > 0)
{
T item = stillToProcess.Dequeue();
yield return item;
foreach (T child in selector(item))
{
stillToProcess.Enqueue(child);
}
}
}
这也会改变迭代顺序——它变成广度优先而不是深度优先;重写它仍然是深度优先是很棘手的。我还将其更改为不使用 Any()
- 这个修订版不会多次评估任何序列,这在某些情况下会很方便。请注意,这确实有一个问题 - 由于排队,它会占用更多内存。我们可能可以通过存储迭代器队列而不是项目来缓解这种情况,但我不确定……它肯定会更复杂。
有一点需要注意(我在查找博客文章时 ChrisW 也注意到了 :) - 如果您的 friend 列表中有任何循环(即如果 A 有 B,并且 B 有 A)那么您将永远递归.
关于c# - 如何将 "unroll"构造成 "recursive",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2012274/
我阅读了有关循环展开的文档。它解释说,如果将展开因子设置为 1,则程序将像使用 #pragma nounrolling 一样工作。 但是,该文件不包括#pragma unroll(0) 案例..由于
我正在用 C++ 编写路径跟踪器,我想尝试将资源最密集的代码实现到 CUDA 或 OpenCL 中(我不确定该选择哪个)。 我听说我的显卡的 CUDA 版本不支持递归,这是我的路径追踪器大量使用的东西
在我的一个JAVA项目中,我通常必须将巨大的数组与标量相乘。因此我想通过使用所谓的循环展开来编写一种方法。到目前为止我已经想出了这个: public static float[] arrayTimes
在最近的代码审查中,出现了关于 @Unroll 注释属于类级别还是方法级别的问题。该类的大多数方法(但不是全部)都需要 @Unroll。如果在类级别声明并且并非类的所有方法都需要它,声明 @Unrol
我有同样的问题: Expression templates: improving performance in evaluating expressions? 我的目标是展开这个表达式的循环 auto
在NVIDIA的制作精良reduction optimization documentation ,他们最终得到一个看起来像这样的 warpReduce: Template __device__ v
我是 CUDA 新手,我无法理解循环展开。我编写了一段代码来理解该技术 __global__ void kernel(float *b, int size) { int tid = block
URL 缩短器在 Twitter 等空间受限的媒体中很有用。但是这样的危险已经得到充分讨论(有限的生命周期、隐藏恶意链接、可用性等)。但是有没有一种很好的方法来预先解析来自 goo.gl 或 bit.
我试图告诉我的编译器使用#pragma unroll 为我展开一个循环。但是,迭代次数由编译时变量决定,因此循环需要展开那么多次。像这样: #define ITEMS 4 #pragma unroll
在 C++11 中是否有一种简单的方法可以做到这一点?如果可能的话,我想同时保留多重继承和循环访问包中所有静态函数的能力。 #include struct A { static void foo()
给定一个 Python 元组 t = v1, v2, v3 是否有一个实用程序可以解压这些元组以便给定: def foo(v1,v2,v3): pass 取而代之的是: foo(t[0],t[1],t
假设我有一个看起来像这样的循环: for(int i = 0; i : xor ebx,ebx 0x08048406 : push ecx 0x08048407 : xor
摘自 GCC 手册: -funroll-loops Unroll loops whose number of iterations can be determined at co
不确定如何调用它,但假设您有一个看起来像这样的类: class Person { public string Name; public IEnumerable Friends; } 然
我使用 Lisp 宏的第一步...... (defconstant width 7) (defconstant height 6) ... ; board is a 2D array of width
例如假设我有以下两个函数 function a(param1) { console.log(param1); } function b(func, params) { func(par
我一直认为 foo2 下面的函数比 foo3 快,直到经过测试。 所有代码如下: #include #include #include #include struct session {
我想知道 C++ 编译器是否会像他们目前对“正常”循环所做的那样展开基于范围的循环以最大化性能,或者在某些情况下基于范围的循环会比正常循环慢? 非常感谢。 最佳答案 基于范围的循环相当于: { a
在 CUDA 中,可以使用 #pragma 展开循环。 unroll 指令通过增加指令级并行度来提高性能。 #pragma可以选择后跟一个数字,指定循环必须展开多少次。 不幸的是,文档没有给出何时应该
我有一段代码,我希望 LLVM 按特定因子展开其中的所有循环。我正在使用以下命令: opt -mem2reg -simplifycfg -loops -lcssa -loop-simplify -lo
我是一名优秀的程序员,十分优秀!