Thread.Sleep vs. Task.Delay when using timeBeginPeriod / Task scheduling

Given the attached fragment LINQ-Pad.

It creates 8 tasks, performs in 500 ms, and draws a graph when the threads are actually running.

On a quad-core processor, it might look like this: enter image description here

Now, if I add Thread.Sleep or Task.Delay in the thread loops, I can visualize the Windows system timer clock (~ 15 ms):

enter image description here

Now there is also a function timeBeginPeriodwhere I can lower the resolution of the system timer (in the example, to 1 ms). And here is the difference. With Thread.SleepI get this diagram (what I expected):

enter image description here

When using, Task.DelayI get the same graph as when the time is set to 15 ms:

enter image description here

Question : Why does TPL ignore the timer setting?

( LinqPad 5.28 beta )

void Main()
{
    const int Threads = 8;
    const int MaxTask = 20;
    const int RuntimeMillis = 500;
    const int Granularity = 10;

    ThreadPool.SetMinThreads(MaxTask, MaxTask);
    ThreadPool.SetMaxThreads(MaxTask, MaxTask);

    var series = new bool[Threads][];
    series.Initialize(i => new bool[RuntimeMillis * Granularity]);

    Watch.Start();
    var tasks = Async.Tasks(Threads, i => ThreadFunc(series[i], pool));
    tasks.Wait();

    series.ForAll((x, y) => series[y][x] ? new { X = x / (double)Granularity, Y = y + 1 } : null)
        .Chart(i => i.X, i => i.Y, LINQPad.Util.SeriesType.Point)
        .Dump();

    async Task ThreadFunc(bool[] data, Rendezvous p)
    {
        double now;
        while ((now = Watch.Millis) < RuntimeMillis)
        {
            await Task.Delay(10);

            data[(int)(now * Granularity)] = true;
        }
    }
}

[DllImport("winmm.dll")] internal static extern uint timeBeginPeriod(uint period);

[DllImport("winmm.dll")] internal static extern uint timeEndPeriod(uint period);

public class Rendezvous
{
    private readonly object lockObject = new object();
    private readonly int max;
    private int n = 0;

    private readonly ManualResetEvent waitHandle = new ManualResetEvent(false);

    public Rendezvous(int count)
    {
        this.max = count;
    }

    public void Join()
    {
        lock (this.lockObject)
        {
            if (++this.n >= max)
                waitHandle.Set();
        }
    }

    public void Wait()
    {
        waitHandle.WaitOne();
    }

    public void Reset()
    {
        lock (this.lockObject)
        {
            waitHandle.Reset();
            this.n = 0;
        }
    }
}

public static class ArrayExtensions
{
    public static void Initialize<T>(this T[] array, Func<int, T> init)
    {
        for (int n = 0; n < array.Length; n++)
            array[n] = init(n);
    }

    public static IEnumerable<TReturn> ForAll<TValue, TReturn>(this TValue[][] array, Func<int, int, TReturn> func)
    {
        for (int y = 0; y < array.Length; y++)
        {
            for (int x = 0; x < array[y].Length; x++)
            {
                var result = func(x, y);
                if (result != null)
                    yield return result;
            }
        }
    }
}

public static class Watch
{
    private static long start;
    public static void Start() => start = Stopwatch.GetTimestamp();
    public static double Millis => (Stopwatch.GetTimestamp() - start) * 1000.0 / Stopwatch.Frequency;
    public static void DumpMillis() => Millis.Dump();
}

public static class Async
{
    public static Task[] Tasks(int tasks, Func<int, Task> thread)
    {
        return Enumerable.Range(0, tasks)
            .Select(i => Task.Run(() => thread(i)))
            .ToArray();
    }

    public static void JoinAll(this Thread[] threads)
    {
        foreach (var thread in threads)
            thread.Join();
    }

    public static void Wait(this Task[] tasks)
    {
        Task.WaitAll(tasks);
    }
}
+4
1

timeBeginPeriod() - , Windows 3.1. Microsoft , . , . - "" , , , .

.NET Thread.Sleep() CLR, . , , Sleep() winapi. , MSDN:

, timeGetDevCaps, timeBeginPeriod, . timeBeginPeriod, , .

- , Microsoft . , - . , .

, , . , -, , . .NET - . Task.Delay() System.Threading.Timer . , , , , 15.6, . btw, 15.625, . , 1 , . 16 , GetTickCount() .

+8

Source: https://habr.com/ru/post/1691460/


All Articles