C#多线程Task创建

本文介绍了C#中多线程的使用,重点讲解了从.Net 1.0的Thread到.Net 3.0推荐的Task,包括Task.Run、Parallel的使用,以及TaskFactory在.Net 4.0中的应用。还提到了lock、Async和Await的概念。
Llama Factory

Llama Factory

模型微调
LLama-Factory

LLaMA Factory 是一个简单易用且高效的大型语言模型(Large Language Model)训练与微调平台。通过 LLaMA Factory,可以在无需编写任何代码的前提下,在本地完成上百种预训练模型的微调

目录

多线程常用函数:

一、Thread -- .Net 1.0

二、ThreadPool -- .Net 2.0

三、Task -- .Net 3.0 -- 推荐

一、Task创建

二、Task.Run 创建

三、Parallel使用

四、Task 其他函数

四、TaskFactory -- .Net 4.0

一、TaskFactory创建

二、Task.Factory.StartNew 创建

三、TaskFactory其他函数 

五、lock 、Async、Await

一、lock

二、Async、Await


涉及知识点: Thread、ThreadPool 、Task、TaskFactory

多线程常用函数:

Thread.CurrentThread.ManagedThreadId     --- 当前线程id
Thread.CurrentThread.IsThreadPoolThread --- 当前线程是否使用线程池

一、Thread -- .Net 1.0

优点:API特别丰富
缺陷:
    1、线程资源是直接从操作系统中获取,在线程挂起时,存在延时情况,造成任务执行顺序混乱
    2、线程个数没有限制,对于新手容易造成死机状态

常用函数:

Thread thread = new Thread(threadStart);
thread.Start(); //线程开启
thread.Suspend(); //线程挂起
thread.Resume(); //挂起线程恢复
thread.Join(); //线程阻塞,直到线程完成后,执行下一步
thread.IsBackground = true; //设置为后台线程
thread.Abort();//线程摧毁

二、ThreadPool -- .Net 2.0

优点:线程池会根据限定的数量去申请和释放
          若空闲线程太少时,会自动向操作系统申请线程
          若空闲线程太多时,会自动释放线程
缺陷:
           API 特别少 -- 造成很难控制线程执行顺序
例程:

WaitCallback waitCallback = (x) =>
{
    Thread.Sleep(3000);
    Console.WriteLine("Thread End");
};
ThreadPool.QueueUserWorkItem(waitCallback);
Console.WriteLine("完成");

三、Task -- .Net 3.0 -- 推荐

优点:1、线程池会根据限定的数量去申请和释放
             若空闲线程太少时,会自动向操作系统申请线程
             若空闲线程太多时,会自动释放线程 <summary>
            2、 API丰富,便于控制

前期准备:

4个函数:1个无参、1个参数、2个参数、2参数+返回值

public static void DoSomeThing()
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End");
}

public static void DoSomeThing_1(string name)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name);
}
public static void DoSomeThing_2(string name1, string name2)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name1 + name2);
}
public static string DoSomeThing_2_1(string name1, string name2)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name1 + name2);
    return name1;
}

一、Task创建

1、无参

Task task = new Task(DoSomeThing);
task.Start();

2、一个参数

Task task = new Task((x)=>DoSomeThing_1((string)x),"XXXXXX");
task.Start();

3、一个参数

Task task = new Task(() => DoSomeThing_1("XXXXX"));
task.Start();

4、两个参数

Task task = new Task(() => DoSomeThing_2("XXXXX_1", "XXXXX_2"));
task.Start();

5、两个参数+返回值 

Task<string> task = new Task<string>(() => DoSomeThing_2_1("XXXXX_1", "XXXXX_2"));
task.Start();
Console.WriteLine(task.Result);

二、Task.Run 创建

1、无参

Task.Run(DoSomeThing);

2、一个参数

Task.Run(()=>DoSomeThing_1("XXXXX"));

3、两个参数+返回值 

Task<string> result = Task.Run(() => DoSomeThing_2_1("mask1", "mask2"));
Console.WriteLine(result.Result);

三、Parallel使用

Parallel 可以启动多线程,主线程也参与计算,节约一个线程 
可以通过ParallelOptions轻松控制最大并发数量 
缺点:主线程卡住

例程:

Parallel.Invoke(
    () =>
    {
        Console.WriteLine("Thread Start1");
        Thread.Sleep(3000);
        Console.WriteLine("Thread End1");
    },
    () =>
    {
        Console.WriteLine("Thread Start2");
        Thread.Sleep(3000);
        Console.WriteLine("Thread End2");
    }
    );
Console.WriteLine("完成");

四、Task 其他函数

Task.WaitAny(tasks.ToArray()); //任意一个完成,继续  
Task.WaitAll(tasks.ToArray()); // 全部完成,继续
例程:

List<Task> tasks = new List<Task>();
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task1"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task2"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task3"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task4"); }));

Task.WaitAny(tasks.ToArray()); //任一一个完成,继续  -- 
Console.WriteLine("XXX第一个完成");

Task.WaitAll(tasks.ToArray()); // 全部完成,继续
Console.WriteLine("完成");

四、TaskFactory -- .Net 4.0

优点:1、线程池会根据限定的数量去申请和释放
             若空闲线程太少时,会自动向操作系统申请线程
             若空闲线程太多时,会自动释放线程 <summary>
            2、 API丰富,便于控制

前期准备:

4个函数:1个无参、1个参数、2个参数、2参数+返回值

public static void DoSomeThing()
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End");
}

public static void DoSomeThing_1(string name)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name);
}
public static void DoSomeThing_2(string name1, string name2)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name1 + name2);
}
public static string DoSomeThing_2_1(string name1, string name2)
{
    Console.WriteLine("Start");
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method--{i}");
    }
    Console.WriteLine("End  " + name1 + name2);
    return name1;
}

一、TaskFactory创建

1、无参

TaskFactory taskFactory = new TaskFactory();
taskFactory.StartNew(DoSomeThing);

2、一个参数

TaskFactory taskFactory = new TaskFactory();
taskFactory.StartNew((x)=>DoSomeThing_1((string)x), "XXXXX");

3、两个参数

TaskFactory taskFactory = new TaskFactory();
taskFactory.StartNew(() => DoSomeThing_2("XXXXX_1", "XXXXX_2"));

4、两个参数+返回值 

TaskFactory taskFactory = new TaskFactory();
Task<string> result = taskFactory.StartNew(() => DoSomeThing_2_1("XXXXX_1", "XXXXX_2"));
Console.WriteLine(result.Result);

二、Task.Factory.StartNew 创建

1、无参

Task.Factory.StartNew(DoSomeThing);

2、一个参数

Task.Factory.StartNew((x) => DoSomeThing_1((string)x), "XXXXX");

3、两个参数

Task.Factory.StartNew(() => DoSomeThing_2("XXXXX_1", "XXXXX_2"));

4、两个参数+返回值  

Task<string> result = Task.Factory.StartNew(() => DoSomeThing_2_1("XXXX_1", "XXXX_2"));
Console.WriteLine(result.Result);

三、TaskFactory其他函数 

taskFactory.ContinueWhenAny  //任意一个完成,继续
taskFactory.ContinueWhenAll    // 全部完成,继续

List<Task> tasks = new List<Task>();
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task1"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task2"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task3"); }));
tasks.Add(Task.Run(() => { Thread.Sleep(2000); Console.WriteLine("task4"); }));

TaskFactory taskFactory = new TaskFactory();
taskFactory.ContinueWhenAny(tasks.ToArray(),
                            (x) =>
                            { Console.WriteLine("初步完成"); });

taskFactory.ContinueWhenAll(tasks.ToArray(),
                            (x) =>
                            { Console.WriteLine("全部完成"); });

五、lock 、Async、Await

一、lock

锁定的内容,多个线程同时访问时,需要排队进行
 lock的标志,最好不用混用,一个标志配一把锁

class Program
{
    public static void Main()
    { 
        List<Task> tasks = new List<Task>();
        tasks.Add(Task.Run(DoSomeThing));
        tasks.Add(Task.Run(DoSomeThing));
        Task.Run(tasks.ToArray);
        Console.WriteLine("----");
    }
    //锁标志位创建
    private static readonly Object LOCK = new Object();
    public static void DoSomeThing()
    {
        //锁内容设置
        lock (LOCK)
        {
            Console.WriteLine("Start");
            for (int i = 0; i < 10; i++)
            {
                Console.WriteLine($"--Method--{i}");
            }
            Console.WriteLine("End");
        }
    }       
}

二、Async、Await

Async 只能修饰 void、Task、Task<T> 方法 此类方法一般用Async结尾
Await 只能修饰 Task,用于等待任务完成

private void button2_Click(object sender, EventArgs e)
{
    Task.Run(DoSomeThing);
    Task.Run(DoSomeThing_1);
    Console.WriteLine("------");
}

public async void DoSomeThing()
{
    Console.WriteLine("---Start---");
    await Task.Delay(1000);
    for(int i = 0;i< 10;i++)
    {
        Console.WriteLine($"--{i}--");
    }
    Console.WriteLine("---End---");
}
public async void DoSomeThing_1()
{
    Console.WriteLine("---Start---");
    await Task.Delay(1000);
    for (int i = 0; i < 10; i++)
    {
        Console.WriteLine($"--Method_2--{i}--");
    }
    Console.WriteLine("---End---");
}

如有错误,烦请批评指正

您可能感兴趣的与本文相关的镜像

Llama Factory

Llama Factory

模型微调
LLama-Factory

LLaMA Factory 是一个简单易用且高效的大型语言模型(Large Language Model)训练与微调平台。通过 LLaMA Factory,可以在无需编写任何代码的前提下,在本地完成上百种预训练模型的微调

<think>我们正在讨论C#中的多线程Task使用。根据引用,Task是更高级、更易于使用的多线程编程模型。我们将提供使用指南和示例。首先,Task是System.Threading.Tasks命名空间中的类,用于表示异步操作。它比直接使用Thread更高效且易于管理。主要使用方式:1.使用Task.Run来启动后台任务。2.使用Task.WaitAll或Task.WaitAny来等待任务完成。3.使用Task.Delay来模拟耗时操作(非阻塞的等待)。示例1:启动两个任务并等待它们完成(如引用[1]所示)示例2:使用async/await进行异步编程(引用[3]提到状态机,这是更现代的方式)我们将提供两个示例:示例一:使用Task.Run和WaitAll(同步等待)示例二:使用async和await(异步等待,不阻塞主线程)注意:在示例中,我们使用Console.WriteLine输出信息,以便观察线程执行情况。另外,我们还需要注意异常处理、取消操作等,但用户问题主要是使用指南,我们先提供基础示例。根据引用[2],Task.Run用于在后台线程执行操作,主线程可以继续执行其他代码。现在,我们编写两个示例代码。</think>### C# 多线程 Task 使用指南 TaskC# 多线程编程的核心组件,它基于 TPL(Task Parallel Library)提供高效的线程管理。以下是关键使用方法和示例: #### 一、基础用法 1. **启动任务** ```csharp // 方式1:Task.Run Task task1 = Task.Run(() => { Console.WriteLine("任务1执行中"); Thread.Sleep(1000); }); // 方式2:Task.Factory.StartNew Task task2 = Task.Factory.StartNew(() => { Console.WriteLine("任务2执行中"); }); ``` 2. **等待任务完成** ```csharp task1.Wait(); // 阻塞当前线程直到完成 Task.WaitAll(task1, task2); // 等待所有任务 Task.WaitAny(task1, task2); // 等待任意任务完成 ``` #### 二、返回值处理 ```csharp Task<int> calcTask = Task.Run(() => { Thread.Sleep(500); return 42; // 返回计算结果 }); // 获取结果(会阻塞线程直到完成) int result = calcTask.Result; ``` #### 三、异步编程 (async/await) ```csharp static async Task Main() { Task<int> dataTask = FetchDataAsync(); Console.WriteLine("等待数据时主线程可继续工作"); int data = await dataTask; // 非阻塞等待 Console.WriteLine($"获取数据: {data}"); } static async Task<int> FetchDataAsync() { await Task.Delay(1000); // 模拟IO操作 return new Random().Next(100); } ``` #### 四、任务链式调用 ```csharp Task.Run(() => Step1()) .ContinueWith(t => Step2(t.Result)) .ContinueWith(t => Step3(t.Result)); ``` #### 五、异常处理 ```csharp try { Task faultedTask = Task.Run(() => throw new Exception("测试异常")); faultedTask.Wait(); } catch (AggregateException ae) { foreach (var e in ae.InnerExceptions) { Console.WriteLine($"捕获异常: {e.Message}"); } } ``` #### 六、取消任务 ```csharp var cts = new CancellationTokenSource(); Task longTask = Task.Run(() => { while (true) { cts.Token.ThrowIfCancellationRequested(); Thread.Sleep(100); } }, cts.Token); cts.CancelAfter(1000); // 1秒后取消 ``` > **最佳实践** > 1. CPU密集型操作使用`Task.Run` > 2. IO操作使用`async/await` > 3. 避免`.Result`阻塞,优先用`await` > 4. 使用`CancellationToken`支持任务取消[^1][^3]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值