Knowledge needs to be continuously accumulated, summarized and precipitated. Thinking and writing are the catalysts for growth

Table of contents

Thread Thread

Thread operations in.NET are encapsulated as Thread classes, allowing developers to intuitively operate on threads. Thread provides instance methods to manage the Thread’s life cycle and static methods to control some of the Thread’s access to storage, which are equivalent to workspace environment variables

1. Life cycle

The life cycle of a thread is created, started, may suspend, wait, resume, exception, and then terminate. The Thread class makes it easy to control the entire life cycle of a Thread

The constructor overloading of the Thread class can accept delegates with ThreadStart and ParameterizedThreadStart without arguments, and then call the Start() method of the instance to Start the Thread. The delegate of Thread’s constructor with an argument of type Object, because we can pass in any information

Thread t1 = new Thread(() => {
    Console.WriteLine($"New Thread {Thread. CurrentThread. ManagedThreadId. ToString ("00")}");
});
t1.Start();
Thread t2 = new Thread((obj) => {
    Console.WriteLine($"New Thread {Thread. CurrentThread. ManagedThreadId. ToString ("00Parameters ")}, {obj. ToString ()}");
});
t2.Start("hello kitty");
Copy the code

Once a thread is started, it can call Suspend() to Suspend the thread, which then goes to sleep (does not continue executing code in the thread), call Resume() to wake it up, and less recommended Abort() to destroy the thread by throwing an exception. The state of the thread then changes to AbortRequested

Thread waiting is also commonly used. After a worker thread is enabled on the main thread, it is sometimes necessary to wait for the completion of the worker thread before the main thread can resume work. We can call the instance method Join(), and of course we can pass in a time parameter to indicate how long the main thread can wait at most

2. Background threads

As we learned in the previous chapter, Thread is created by default as the foreground Thread. The foreground Thread will prevent the exit of the system process, that is, the task must be completed after starting. The background Thread will exit with the exit of the process. Change to background thread by setting property IsBackground=true. You can also specify the Priority of a thread by setting Priority. But this doesn’t always have to be done first if you set a high priority as you’d like. The operating system optimizes scheduling, which is one of the reasons threads are difficult to control

3. Static methods

These are the Tread instance methods, but Thread also has some common static methods. Sometimes threads are not set up properly and there are unexpected bugs

1. Thread local storage

AllocateDataSlot and AllocateNamedDataSlot are used to assign a data slot to all threads. As shown in the following example, if you don’t put data into a slot in a child thread, you can’t get the data that other threads put into it.

var slot= Thread.AllocateNamedDataSlot("testSlot");
//Thread.FreeNamedDataSlot("testSlot");
Thread.SetData(slot, "hello kitty");
Thread t1 = new Thread(() => {
    //Thread.SetData(slot, "hello kitty");
    var obj = Thread.GetData(slot);
    Console.WriteLine($Child thread: {obj}); //obj has no value}); t1.Start(); var obj2 = Thread.GetData(slot); Console.WriteLine($"Main thread: {obj2}");
Copy the code

When you declare a slot. NET reminds us to use the ThreadStaticAttribute tag field for better performance. What do you mean? Let’s look at the following example

// // Summary: // Assign unnamed data slots on all threads. In order to get better performance, please switch to System. ThreadStaticAttribute features tag field. // // Returns the result: // Named slots allocated on all threads. public static LocalDataStoreSlot AllocateDataSlot();Copy the code

In the example, ThreadStatic output would be consistent if the static fields were not marked. The ThreadStatic tag indicates whether static field values are unique for each thread

[ThreadStatic]
static string name = string.Empty;
public void Function()
{
    name = "kitty";
    Thread t1 = new Thread(() => {
        Console.WriteLine($Child thread: {name}); // output null}); t1.Start(); Console.WriteLine($"Main thread: {name}"); // output Kitty}Copy the code

There is also a ThreadLocal that provides local storage of thread data, using the same method as above, declaring data in each thread for its own use only

ThreadLocal<string> local = new ThreadLocal<string>() { };
local.Value = "hello kitty";
Thread t = new Thread(() => {
    Console.WriteLine($"Child thread: {local-value}");
});
t.Start();
Console.WriteLine($"Main thread: {local.Value}");
Copy the code

The static method above is used for Thread Local Storage TLS (Thread Local Storage). Thread.sleep is also commonly used during development and debugging to allow threads to suspend for a specified period of time to simulate time-consuming operations

2. Memory barrier

Let’s start with a common sense question, why do we Release when we Release? Release a smaller, faster, and do a lot of optimization, the optimization is transparent to us (computer transparent considered to be a black box, the internal logic details for we do not open, transparent and life means entirely understand don’t cheat the opposite), general optimization will not affect the operation of the program, we first use the Internet a case in point

bool isStop = false;
Thread t = new Thread(() => {
    bool isSuccess = false;
    while(! isStop) { isSuccess = ! isStop; }}); t.Start(); Thread.Sleep(1000); isStop =true;
t.Join();
Console.WriteLine("Main thread execution completed");
Copy the code

In the example above, if the execution completes correctly under debug until “main program completes” is printed, while under Release, it waits for the child thread to complete. Here isStop is always false in the child thread. First of all, this is a problem caused by multiple threads sharing variables, so we suggest that the best solution is not to share variables as much as possible. Second, we can use Thread.MemoryBarrier, VolatileRead/Write and other locking mechanisms to sacrifice performance for data security. (The above example test if the child thread while Console. WriteLine operation, found that the release also normal output, guess it should be memory update)

Release optimization will load the value of isStop variable in t thread into CPU Cache, while the main thread changes the value of isStop in memory, so the child thread cannot get the updated value, resulting in inconsistent data. So the solution is to get the value out of memory. Thread.memorybarrier () allows all writes prior to this method to be updated from the CPU Cache, and all subsequent reads to be fetched from memory, not the CPU Cache. Adding thread.memoryBarrier () to the while avoids data inconsistencies. VolatileRead/Write is a separate interpretation of MemoryBarrier, read from and written to the handler.

4. Return value

When we declared the thread, we could pass arguments, so what do we do to get a return value? Thread does not provide an operation that returns a value. NET provides a solution for Thead’s advanced encapsulation, which can be used directly. If you want to use the thread class, you need to implement the following thread operations with the return value. All these operations are implemented through the delegate.

private Func<T> ThreadWithReturn<T>(Func<T> func)
{
    T t = default(T);
    Thread thread = new Thread(() =>
    {
        t = func.Invoke();
    });
    thread.Start();
    return () =>
    {
        thread.Join();
        returnt; }; } // call Func<int> Func = this.threadWithReturn <int>() => {thread.sleep (2000);return DateTime.Now.Millisecond;
});
int iResult = func.Invoke();
Copy the code

ThreadPools

.net originally provided the ThreadPool class, which is very rich in functionality and has a lot of apis, so it is difficult to use. Moreover, threads do not always run like ideal. Therefore, since 2.0, we have provided the ThreadPool static class, which is full of static methods, hiding the interface of many threads, making it easier to use threads. Thread pools can be used to perform tasks, send work items, handle asynchronous I/O, wait on behalf of other threads, and handle timers.

1. Work queues

The common ThreadPool static method QueueUserWorkItem is used to queue a method into a ThreadPool queue for execution. If there are idle threads in the ThreadPool, it will execute. The argument to QueueUserWorkItem can specify a callback delegate and pass in arguments, as shown below

ThreadPool.QueueUserWorkItem((obj) => {
                Console.WriteLine($"Thread pool threads in the {Thread. CurrentThread. ManagedThreadId. ToString ("00")}, pass {obj.tostring ()}");
            },"hello kitty");
Copy the code

Worker threads and IO threads

General asynchronous task execution, do not involve network files and other IO operations, computationally intensive, developers to call. IO threads are typically used on file networks and are called by the CLR, so developers don’t have to worry about them. The worker thread initiates a file access call, and the driver notifies the IO thread after completion, which then performs the callback function for the asynchronous task

Gets and sets the minimum and maximum worker and IO threads

ThreadPool.GetMaxThreads(out int workerThreads, out int completionPortThreads);
ThreadPool.GetMinThreads(out int workerThreads, out int completionPortThreads);
ThreadPool.SetMaxThreads(16, 16);
ThreadPool.SetMinThreads(8, 8);
Copy the code

3. Different from Thread

If your computer has only eight cores, you can have eight tasks running at once. Now we have 10 tasks to run. With Thread we need to create 10 threads. With ThreadPool we might only need to use 8 threads, saving space and time. Threads in a thread pool start with the minimum number of threads by default, and then increase or decrease the number as needed. Thread pools are simple to use, but have some limitations. Threads in a thread pool are background threads that cannot be prioritized and are often used for short tasks. Threads in the thread pool can also block wait, using ManualResetEvent to notify, but this is not usually used.

4. Timer

There are many in the.net can realize the function of the timer, in ThreadPool, we can use RegisterWaitForSingleObject to sign up for a specified time entrusted to wait. Messages are output every second, as follows

ThreadPool.RegisterWaitForSingleObject(new AutoResetEvent(true), new WaitOrTimerCallback((obj, b) =>
{
    Console.WriteLine($"obj={obj},tid={Thread.CurrentThread.ManagedThreadId},datetime={DateTime.Now}");
}),"hello kitty", 1000,false);
Copy the code

The timer class can be found in several places in.net, such as System.Threading, System. timer, system.windows. Form, system.web. UI, etc. This is followed by the Timer extension from the first System.threading

System.Threading.Timer timer = new System.Threading.Timer((obj) =>
{
    Console.WriteLine($"obj={obj},tid={Thread.CurrentThread.ManagedThreadId},datetime={DateTime.Now}");
},"hello kitty", 1000100);Copy the code

Timer has a bottom TimerQueue, using the ThreadPool. UnsafeQueueUserWorkItem to complete timing function, and we use the above ThreadPool timer is a little difference

In actual development, a simple timer is sufficient, but the general business scenario is complicated, so personalized timers need to be customized, such as the date of execution of the month, the day of the week of the month, the time of execution, and the day of execution. Therefore, we use the Quarz.NET timing framework, which will be used when the framework is integrated, and it is very simple to use

Let’s talk about these two things first, the next article should be Task, Parallel and Async/Await, then summarize C# thread patterns, thread synchronization locking, exception handling, thread cancellation, thread-safe collections and common threading issues



Long water wide, see the word such as face

May you be at the age of lips and teeth

In no hurry

large

Amit ā bha

With the fate of the update, said goodbye

Follow xishaobb for more information