- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我试图了解 Parallel.Invoke
如何创建和重用线程。我运行了以下示例代码(来自 MSDN,https://msdn.microsoft.com/en-us/library/dd642243(v=vs.110).aspx):
using System;
using System.Threading;
using System.Threading.Tasks;
class ThreadLocalDemo
{
static void Main()
{
// Thread-Local variable that yields a name for a thread
ThreadLocal<string> ThreadName = new ThreadLocal<string>(() =>
{
return "Thread" + Thread.CurrentThread.ManagedThreadId;
});
// Action that prints out ThreadName for the current thread
Action action = () =>
{
// If ThreadName.IsValueCreated is true, it means that we are not the
// first action to run on this thread.
bool repeat = ThreadName.IsValueCreated;
Console.WriteLine("ThreadName = {0} {1}", ThreadName.Value, repeat ? "(repeat)" : "");
};
// Launch eight of them. On 4 cores or less, you should see some repeat ThreadNames
Parallel.Invoke(action, action, action, action, action, action, action, action);
// Dispose when you are done
ThreadName.Dispose();
}
}
据我了解,Parallel.Invoke
尝试在此处创建 8 个线程 - 每个操作一个。所以它创建第一个线程,运行第一个action
,然后给线程一个ThreadName
。然后它创建下一个线程(获得不同的 ThreadName
)等等。
如果它不能创建新线程,它将重用之前创建的线程之一。在这种情况下,repeat
的值将为 true
,我们可以在控制台输出中看到这一点。
到这里为止都是正确的吗?
倒数第二个注释(“启动其中的八个。在 4 个或更少的内核上,您应该看到一些重复的线程名称”)暗示 Invoke
创建的线程对应于可用的 cpu 线程处理器:在 4 个内核上,我们有 8 个 cpu 线程,至少有一个忙(运行操作系统和其他东西),所以 Invoke
只能使用 7 个不同的线程,所以我们必须至少有一个 “重复”
。
我对这条评论的解释是否正确?
我在配备 Intel® Core™ i7-2860QM 处理器(即 4 个内核,8 个 cpu 线程)的 PC 上运行了这段代码。我希望至少得到一个 "repeat"
,但我没有。当我将 Invoke
更改为执行 10 个而不是 8 个操作时,我得到了这个输出:
ThreadName = Thread6
ThreadName = Thread8
ThreadName = Thread6 (repeat)
ThreadName = Thread5
ThreadName = Thread3
ThreadName = Thread1
ThreadName = Thread10
ThreadName = Thread7
ThreadName = Thread4
ThreadName = Thread9
所以我在控制台应用程序中至少有 9 个不同的线程。这与我的处理器只有 8 个线程的事实相矛盾。
所以我想我上面的一些推理是错误的。 Parallel.Invoke
的工作方式是否与我上面描述的不同?如果是,如何?
最佳答案
如果您将少于 10 个项目传递给 Parallel.Invoke
,并且您没有在选项中指定 MaxDegreeOfParallelism
(所以 - 您的情况),它只会运行它们所有在线程池调度程序上并行使用以下代码:
var actions = new [] { action, action, action, action, action, action, action, action };
var tasks = new Task[actions.Length];
for (int index = 1; index < tasks.Length; ++index)
tasks[index] = Task.Factory.StartNew(actions[index]);
tasks[0] = new Task(actions[0]);
tasks[0].RunSynchronously();
Task.WaitAll(tasks);
所以只是一个普通的 Task.Factory.StartNew
。如果您将查看线程池中的最大线程数
int th, io;
ThreadPool.GetMaxThreads(out th, out io);
Console.WriteLine(th);
您会看到一些大数字,例如 32767。因此,执行 Parallel.Invoke
的线程数(在您的情况下)根本不限于 cpu 核心数。即使在 1 核 cpu 上,它也可能并行运行 8 个线程。
您可能会想,为什么有些线程会被重用?因为在线程池线程上完成工作时 - 该线程返回到池中并准备好接受新工作。您的示例中的操作基本上根本不起作用并且完成得非常快。因此,有时通过 Task.Factory.StartNew
启动的第一个线程已经完成了您的操作,并在所有后续线程启动之前返回到池中。因此该线程被重用。
顺便说一下,您可以在您的示例中看到 (repeat)
有 8 个 Action ,如果您足够努力,甚至可以在 8 核(16 个逻辑核)处理器上看到 7 个。
更新以回答您的评论。线程池调度程序不需要立即创建新线程。线程池中有最小和最大线程数。如何查看最大值我已经在上面展示过。查看最小数量:
int th, io;
ThreadPool.GetMinThreads(out th, out io);
这个数字通常等于核心数(例如 8)。现在,当您请求在线程池线程上执行新操作,并且线程池中的线程数小于最小值时 - 将立即创建新线程。但是,如果可用线程数大于最小值 - 在创建新线程之前会引入一定的延迟(不幸的是,我不记得确切的时间,大约 500 毫秒)。
我非常怀疑您在评论中添加的语句能否在 2-3 秒内执行。对我来说,它最多执行 0.3 秒。因此,当线程池创建前 8 个线程时,在创建第 9 个线程之前有 500 毫秒的延迟。在此延迟期间,前 8 个线程中的一些(或全部)线程已完成其工作并可用于新工作,因此无需创建新线程并且可以重复使用它们。
为了验证这一点,引入更大的延迟:
static void Main()
{
// Thread-Local variable that yields a name for a thread
ThreadLocal<string> ThreadName = new ThreadLocal<string>(() =>
{
return "Thread" + Thread.CurrentThread.ManagedThreadId;
});
// Action that prints out ThreadName for the current thread
Action action = () =>
{
// If ThreadName.IsValueCreated is true, it means that we are not the
// first action to run on this thread.
bool repeat = ThreadName.IsValueCreated;
Console.WriteLine("ThreadName = {0} {1}", ThreadName.Value, repeat ? "(repeat)" : "");
Thread.Sleep(1000000);
};
int th, io;
ThreadPool.GetMinThreads(out th, out io);
Console.WriteLine("cpu:" + Environment.ProcessorCount);
Console.WriteLine(th);
Parallel.Invoke(Enumerable.Repeat(action, 100).ToArray());
// Dispose when you are done
ThreadName.Dispose();
Console.ReadKey();
}
你会看到现在线程池每次都必须创建新线程(比内核多得多),因为它不能重用以前繁忙的线程。
您还可以增加线程池中的最小线程数,如下所示:
int th, io;
ThreadPool.GetMinThreads(out th, out io);
ThreadPool.SetMinThreads(100, io);
这将消除延迟(直到创建 100 个线程),在上面的示例中您会注意到这一点。
关于c# - 了解 Parallel.Invoke,线程的创建和重用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42296364/
我将 Bootstrap 与 css 和 java 脚本结合使用。在不影响前端代码的情况下,我真的很难在css中绘制这个背景。在许多问题中,人们将宽度和高度设置为 0%。但是由于我的导航栏,我不能使用
我正在用 c 编写一个程序来读取文件的内容。代码如下: #include void main() { char line[90]; while(scanf("%79[^\
我想使用 javascript 获取矩阵数组的所有对 Angular 线。假设输入输出如下: input = [ [1,2,3], [4,5,6], [7,8,9], ] output =
可以用pdfmake绘制lines,circles和other shapes吗?如果是,是否有documentation或样本?我想用jsPDF替换pdfmake。 最佳答案 是的,有可能。 pdfm
我有一个小svg小部件,其目的是显示角度列表(参见图片)。 现在,角度是线元素,仅具有笔触,没有填充。但是现在我想使用一种“内部填充”颜色和一种“笔触/边框”颜色。我猜想line元素不能解决这个问题,
我正在为带有三角对象的 3D 场景编写一个非常基本的光线转换器,一切都工作正常,直到我决定尝试从场景原点 (0/0/0) 以外的点转换光线。 但是,当我将光线原点更改为 (0/1/0) 时,相交测试突
这个问题已经有答案了: Why do people write "#!/usr/bin/env python" on the first line of a Python script? (22 个回
如何使用大约 50 个星号 * 并使用 for 循环绘制一条水平线?当我尝试这样做时,结果是垂直(而不是水平)列出 50 个星号。 public void drawAstline() { f
这是一个让球以对角线方式下降的 UI,但球保持静止;线程似乎无法正常工作。你能告诉我如何让球移动吗? 请下载一个球并更改目录,以便程序可以找到您的球的分配位置。没有必要下载足球场,但如果您愿意,也可以
我在我的一个项目中使用 Jmeter 和 Ant,当我们生成报告时,它会在报告中显示 URL、#Samples、失败、成功率、平均时间、最短时间、最长时间。 我也想在报告中包含 90% 的时间线。 现
我有一个不寻常的问题,希望有人能帮助我。我想用 Canvas (android) 画一条 Swing 或波浪线,但我不知道该怎么做。它将成为蝌蚪的尾部,所以理想情况下我希望它的形状更像三角形,一端更大
这个问题已经有答案了: Checking Collision of Shapes with JavaFX (1 个回答) 已关闭 8 年前。 我正在使用 JavaFx 8 库。 我的任务很简单:我想检
如何按编号的百分比拆分文件。行数? 假设我想将我的文件分成 3 个部分(60%/20%/20% 部分),我可以手动执行此操作,-_-: $ wc -l brown.txt 57339 brown.tx
我正在努力实现这样的目标: 但这就是我设法做到的。 你能帮我实现预期的结果吗? 更新: 如果我删除 bootstrap.css 依赖项,问题就会消失。我怎样才能让它与 Bootstrap 一起工作?
我目前正在构建一个网站,但遇到了 transform: scale 的问题。我有一个按钮,当用户将鼠标悬停在它上面时,会发生两件事: 背景以对 Angular 线“扫过” 按钮标签颜色改变 按钮稍微变
我需要使用直线和仿射变换绘制大量数据点的图形(缩放图形以适合 View )。 目前,我正在使用 NSBezierPath,但我认为它效率很低(因为点在绘制之前被复制到贝塞尔路径)。通过将我的数据切割成
我正在使用基于 SVM 分类的 HOG 特征检测器。我可以成功提取车牌,但提取的车牌除了车牌号外还有一些不必要的像素/线。我的图像处理流程如下: 在灰度图像上应用 HOG 检测器 裁剪检测到的区域 调
我有以下图片: 我想填充它的轮廓(即我想在这张图片中填充线条)。 我尝试了形态学闭合,但使用大小为 3x3 的矩形内核和 10 迭代并没有填满整个边界。我还尝试了一个 21x21 内核和 1 迭代,但
我必须找到一种算法,可以找到两组数组之间的交集总数,而其中一个数组已排序。 举个例子,我们有这两个数组,我们向相应的数字画直线。 这两个数组为我们提供了总共 7 个交集。 有什么样的算法可以帮助我解决
简单地说 - 我想使用透视投影从近裁剪平面绘制一条射线/线到远裁剪平面。我有我认为是使用各种 OpenGL/图形编程指南中描述的方法通过单击鼠标生成的正确标准化的世界坐标。 我遇到的问题是我的光线似乎
我是一名优秀的程序员,十分优秀!