- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我使用的语言是c#
。
假设我们要遍历名为 customers
的序列的元素,该序列是名为 Customer
的虚构类型的对象序列。在代码方面,我们有以下内容:
IEnumerable<Customer> customers = module.GetCustomers();
其中 module
是服务层的类,通过其中一种方法,我们可以检索所有客户。也就是说,customers
元素的迭代将是:
foreach(var customer in customers)
{
}
现在让我们在遍历 customers
的元素后获取客户数量。这可以像下面这样完成:
int numberOfCustomers = customers.Count();
我现在的顾虑/问题如下:
我们使用 Count()
方法再次遍历 customers
的元素。但是,如果我们已经创建了该对象的内存集合,例如调用方法 ToList()
:
List<Customer> customers = module.GetCustomers()
.ToList();
我们将使用 customers
列表的 Count
属性在 O(1)
中获得客户数量。
为了找出这两个选项中最好的一个,我编写了一个简单的控制台应用程序,并使用 StopWatch
类来分析它们。但是,我没有得出明确的结果。
这两个选项中哪个是最好的?
更新
我运行了以下控制台应用程序:
class Program
{
static void Main(string[] args)
{
IEnumerable<int> numbers = Enumerable.Range(0, 1000);
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();
foreach (var number in numbers)
Console.WriteLine(number);
Console.WriteLine(numbers.Count());
stopwatch.Stop();
// I got 175ms
Console.WriteLine(stopwatch.ElapsedMilliseconds);
Console.ReadKey();
stopwatch.Restart();
List<int> numbers2 = numbers.ToList();
foreach (var number in numbers2)
Console.WriteLine(number);
Console.WriteLine(numbers2.Count);
stopwatch.Stop();
// I got 86ms
Console.WriteLine(stopwatch.ElapsedMilliseconds);
Console.ReadKey();
}
}
然后我运行了这个:
class Program
{
static void Main(string[] args)
{
IEnumerable<int> numbers = Enumerable.Range(0, 1000);
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();
List<int> numbers2 = numbers.ToList();
foreach (var number in numbers2)
Console.WriteLine(number);
Console.WriteLine(numbers2.Count);
stopwatch.Stop();
// I got 167ms
Console.WriteLine(stopwatch.ElapsedMilliseconds);
Console.ReadKey();
stopwatch.Restart();
foreach (var number in numbers)
Console.WriteLine(number);
Console.WriteLine(numbers.Count());
stopwatch.Stop();
// I got 104ms
Console.WriteLine(stopwatch.ElapsedMilliseconds);
Console.ReadKey();
}
}
最佳答案
我通常更喜欢让我的存储库方法返回一个 IReadOnlyCollection<>
,这有助于调用者知道他们可以安全地对其进行多次迭代:
IReadOnlyCollection<Customer> customers = module.GetCustomers();
如果我做不到,并且我知道我将多次迭代我得到的内容,我通常会使用 .ToList() 来确保我正在处理一个 in-内存收集:
var customers = module.GetCustomers().ToList();
在 customers 已经是内存集合的情况下,这会通过创建列表增加一些开销,但它有助于避免通过执行诸如从数据库中多次检索数据之类的操作而产生大量开销的风险次。
您的基准测试存在缺陷有几个原因,但最大的原因之一是它使用了 Console.WriteLine()
, 它执行 I/O 操作。该操作将花费远远超过迭代集合和计算结果的总和。事实上,在 Console.WriteLine()
中花费的时间量方差将超过您正在测试的代码中的差异。
但这实际上很好地说明了我的观点——I/O 操作比 CPU 和内存操作花费的时间长得多,因此通常值得添加 .ToList()
,这可能会增加运行时间微秒,以避免添加可能增加毫秒的 I/O 操作的可能性最小。
关于c# - 遍历序列然后调用 Count() 或在开始时创建一个列表然后调用 Count,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25455766/
SQLite、Content provider 和 Shared Preference 之间的所有已知区别。 但我想知道什么时候需要根据情况使用 SQLite 或 Content Provider 或
警告:我正在使用一个我无法完全控制的后端,所以我正在努力解决 Backbone 中的一些注意事项,这些注意事项可能在其他地方更好地解决......不幸的是,我别无选择,只能在这里处理它们! 所以,我的
我一整天都在挣扎。我的预输入搜索表达式与远程 json 数据完美配合。但是当我尝试使用相同的 json 数据作为预取数据时,建议为空。点击第一个标志后,我收到预定义消息“无法找到任何内容...”,结果
我正在制作一个模拟 NHL 选秀彩票的程序,其中屏幕右侧应该有一个 JTextField,并且在左侧绘制弹跳的选秀球。我创建了一个名为 Ball 的类,它实现了 Runnable,并在我的主 Draf
这个问题已经有答案了: How can I calculate a time span in Java and format the output? (18 个回答) 已关闭 9 年前。 这是我的代码
我有一个 ASP.NET Web API 应用程序在我的本地 IIS 实例上运行。 Web 应用程序配置有 CORS。我调用的 Web API 方法类似于: [POST("/API/{foo}/{ba
我将用户输入的时间和日期作为: DatePicker dp = (DatePicker) findViewById(R.id.datePicker); TimePicker tp = (TimePic
放宽“邻居”的标准是否足够,或者是否有其他标准行动可以采取? 最佳答案 如果所有相邻解决方案都是 Tabu,则听起来您的 Tabu 列表的大小太长或您的释放策略太严格。一个好的 Tabu 列表长度是
我正在阅读来自 cppreference 的代码示例: #include #include #include #include template void print_queue(T& q)
我快疯了,我试图理解工具提示的行为,但没有成功。 1. 第一个问题是当我尝试通过插件(按钮 1)在点击事件中使用它时 -> 如果您转到 Fiddle,您会在“内容”内看到该函数' 每次点击都会调用该属
我在功能组件中有以下代码: const [ folder, setFolder ] = useState([]); const folderData = useContext(FolderContex
我在使用预签名网址和 AFNetworking 3.0 从 S3 获取图像时遇到问题。我可以使用 NSMutableURLRequest 和 NSURLSession 获取图像,但是当我使用 AFHT
我正在使用 Oracle ojdbc 12 和 Java 8 处理 Oracle UCP 管理器的问题。当 UCP 池启动失败时,我希望关闭它创建的连接。 当池初始化期间遇到 ORA-02391:超过
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve
引用这个plunker: https://plnkr.co/edit/GWsbdDWVvBYNMqyxzlLY?p=preview 我在 styles.css 文件和 src/app.ts 文件中指定
为什么我的条形这么细?我尝试将宽度设置为 1,它们变得非常厚。我不知道还能尝试什么。默认厚度为 0.8,这是应该的样子吗? import matplotlib.pyplot as plt import
当我编写时,查询按预期执行: SELECT id, day2.count - day1.count AS diff FROM day1 NATURAL JOIN day2; 但我真正想要的是右连接。当
我有以下时间数据: 0 08/01/16 13:07:46,335437 1 18/02/16 08:40:40,565575 2 14/01/16 22:2
一些背景知识 -我的 NodeJS 服务器在端口 3001 上运行,我的 React 应用程序在端口 3000 上运行。我在 React 应用程序 package.json 中设置了一个代理来代理对端
我面临着一个愚蠢的问题。我试图在我的 Angular 应用程序中延迟加载我的图像,我已经尝试过这个2: 但是他们都设置了 src attr 而不是 data-src,我在这里遗漏了什么吗?保留 d
我是一名优秀的程序员,十分优秀!