POOPE 发表于 2021-7-26 13:34:46

多线程基础

序:
我是自学的C#,在看到多线程一章时,郁闷了,搞不懂,很难理解吧...自认为有软件天赋,却没看懂...唉...自信心被打击了,到网上查了很多的资料,包括MSDN
上也查过资料,可惜还是没搞懂多线程...
于是,硬着头皮一遍一遍的看多线程那一章,终于在看完第31遍,我写出了第一个多线程程序,还算小有所获,鉴于对网上很多资料没办法理解(因为,很多资料一来就大篇大篇
的代码,让人头晕,我是这么觉得)所以,我自己就写了这篇心得体会,希望能给大家带来些许帮助.
匆忙之中,错误难免,欢迎指正,共同进步.
正文:
首先我要提一点,关于线程的基础知识,一个程序,即一个进程,可以有很多个线程,当然,至少要有一个线程,即主线程.相信大家都知道多线程的好处吧,举个书上的例子
吧,Windows在复制文件的时候,有个动画,是在复制文件过程中进行的,也就是一边复制文件,一边播放动画,这个就是很简单的多线程,如果没有动画,复制一个大文件的时候,我们知
道计算机是死机了,还是仍然在复制呢???多线程就很好的解决了这个问题.懂了吧,恩,很好!!那么,我们就开始吧!
首先,我们写个简单的单线程程序,也就是只有程序自己创建的那个主线程,没有使用多线程.
创建一个新工程,向窗口添加一个label命名为label1;我们要让程序运行时label1就显示一个数字,假设为100;通常我们会直接在窗口加载事件中写label1.Text = "100";这样,运行
,label1果然显示了100;
代码如下:(例1)
using System;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "100";
      }
    }
}很简单吧,看懂了吗??
什么,没有,啊~~~神啊~~~救救我吧,那请你在翻书,把最最最最最基础的书翻出来看看里面的最最最最最简单例子(以后不要说我认识你)
好了,看懂的朋友继续往下看:
我们现在要将程序稍稍改动一下,添加一个Button,命名为button1,我们要在按下button1后,将lable1的text从0显示到100,
那么,我们需要添加button1的Click事件,在click事件内写入循环显示0到100.
代码如下:(例2)
using System;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
      }

      private void button1_Click(object sender, EventArgs e)
      {
            for(int i=0;i<101;i++)
            {
                label1.Text = i.ToString();
            }
      }
    }
}运行一下看看,按一下button1,结果是我们一下就看到了100,并没有看到0~100的过程,为什么呢?
呵呵,因为你的处理器速度太快了,就只能看到最后的结果,那么,怎样才能看到中间过程呢?(等一下再讲)
我们先用函数的方式来实现上面的功能
写个名为run的函数吧:
private void run()
{
    for(int i=0;i<101;i++)
    {
      label1.Text = i.ToString();
    }
}这样就可以直接调用run函数实现功能了,而不用在事件函数内写代码。(这样做是有好处的)
整个代码如下:(例3)
using System;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
      }

      private void button1_Click(object sender, EventArgs e)
      {
            run();    //调用run函数
      }

      private void run()
      {
            for(int i=0;i<101;i++)
            {
                label1.Text = i.ToString();
            }
      }
    }
}这里就需要在循环过程中加延时了,假定我们每隔1s的延时,lable1的值增加1。
方法有很多,我们就用一个timer来实现延时。
添加一个timer, 命名为timer1,在timer1的tick事件内添加语句,改变label1的值。(Tick事件是每经过指定时间间隔后被触发)
代码如下:(例4)
using System;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      int i;          //全局变量i

      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
      }

      private void button1_Click(object sender, EventArgs e)
      {
            run();
      }

      private void run()
      {
            i = 0;
            timer1.Interval = 1000; //设置timer1的间隔时间
            timer1.Start(); //启动timer1
      }

      private void timer1_Tick(object sender, EventArgs e) //timer1的Tick事件
      {
            i++;
            if (i > 100)
            {
                timer1.Stop();
            }
            label1.Text = i.ToString();
      }
    }
}同样的,我们运行一下,看看结果,很好,我们能够看到0~100循环的过程了。
下面我们就要进入多线程了,不知道各位将上面的内容看懂了没有?
开始进入多线程之前我还是先简单的说说定义线程吧。(与多线程有关的其它内容我就不说了吧,那个太多太多了)
由于要使用多线程,我们需要引用System.Threading;所以之后的代码都会在前面加上using System.Threading;
怎么定义线程呢?通过下面的语句就定义一个名为thread1的线程
private Thread thread1;
和定义函数极为相似
定义线程之后,就要进行实例化:
thread1 = new Thread(new ThreadStart(run));
这个语句的意思就是实例化thread1并将run函数设定为thread1的入口函数(大概意思就是,让run函数在线程thread1上执行,我是这样理解的)
创建线程就算完成了,那么怎么运行线程呢?
其实和启动timer1是类似的,thread1.Start();就运行了我们创建的线程thread1。
好了,大功告成!哈哈,别着急,既然我们创建了线程,那么在关闭窗口的时候,就要撤消线程。
添加FormClosing事件,在事件内部写如撤消线程的代码:
private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
    if (thread1.IsAlive) //判断thread1是否存在,不能撤消一个不存在的线程,否则会引发异常
    {
      thread1.Abort(); //撤消thread1
    }
}这样才算大功告成嘛,整理的代码如下:(例5)(在例3的基础上加以改动)
using System;
using System.Threading;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      private Thread thread1;

      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
      }

      private void button1_Click(object sender, EventArgs e)
      {
            thread1 = new Thread(new ThreadStart(run));
            thread1.Start();
      }

      private void run()
      {
            for (int i = 0; i < 101; i++)
            {
                label1.Text = i.ToString();
            }
      }

      private void Form1_FormClosing(object sender, FormClosingEventArgs e)
      {
            if (thread1.IsAlive)
            {
                thread1.Abort();
            }
      }
    }
}运行看看,按button1,出错了,怎么回事呢????
哈哈~~看看出错原因,是在run函数内的label1.Text = i.ToString();语句上出的错,没错啊,语法正确啊
哈哈~~我来解释一下,出错的原因是为了保护数据的安全所以不能跨线程调用控件,而label1.Text = i.ToString();句则是在线程thread1上面调用主线程的控件,肯定会出错的
!!
怎么办呢?用委托啊(有关委托,请参考其它资料,我就不多说了)
我的理解就是,线程thread1不能调用主线程的lable1,所以,就委托主线程来改变lable1的值。
首先看一个例子:(从例3改写)(并不创建线程,仅有主线程)
创建一个函数,用来设置lable1的值;
private void set_lableText(string s)
{
    label1.Text = s;
}当需要改变lable1的值时,就调用它,并传递要改变的值。
整理代码如下:(例6)
using System;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
      }

      private void button1_Click(object sender, EventArgs e)
      {
            run();    //调用run函数
      }

      private void run()
      {
            for(int i=0;i<101;i++)
            {
                set_lableText( i.ToString() );
            }
      }

      private void set_lableText(string s)
      {
            label1.Text = s;
      }
    }
}实现的功能与例3是一样的,只是,增加了一个函数。
 现在再来看看委托,我们就需要委托主线程调用函数set_lableText(string s);来改变lable1的值。
 首先声明一个委托:
delegate void set_Text(string s);
 创建一个全局委托变量:(应该是变量吧)
set_Text Set_Text; //请注意大小写,set_Text是委托类型,Set_Text是创建的委托(当然,这里的命名是随意的)
 类似于创建线程,需要进行实例化:
Set_Text = new set_Text(set_lableText); //括号内的set_lableText是委托要调用的函数(也就是例6写的set_lableText(string s);函数)
 现在,就剩下调用委托了,怎么调用委托呢?很简单。
同过Invoke来调用,语句如下:
label1.Invoke(Set_Text, new object[] { i.ToString() });
//Set_Text是调用的委托,object[]则是我们要传递的参数
 整理代码如下:(例7)
using System;
using System.Threading;
using System.Windows.Forms;

namespace ThreadTest
{
    public partial class Form1 : Form
    {
      private Thread thread1; //定义线程

      delegate void set_Text(string s); //定义委托

      set_Text Set_Text; //定义委托

      public Form1()
      {
            InitializeComponent();
      }

      private void Form1_Load(object sender, EventArgs e)
      {
            label1.Text = "0";
            Set_Text = new set_Text(set_lableText); //实例化
      }

      private void button1_Click(object sender, EventArgs e)
      {
            thread1 = new Thread(new ThreadStart(run));
            thread1.Start();
      }

      private void set_lableText(string s) //主线程调用的函数
      {
            label1.Text = s;
      }

      private void run()
      {
            for (int i = 0; i < 101; i++)
            {
                label1.Invoke(Set_Text, new object[] { i.ToString() }); //通过调用委托,来改变lable1的值
                Thread.Sleep(1000); //线程休眠时间,单位是ms
            }
      }

      private void Form1_FormClosing(object sender, FormClosingEventArgs e)
      {
            if (thread1.IsAlive) //判断thread1是否存在,不能撤消一个不存在的线程,否则会引发异常
            {
                thread1.Abort(); //撤消thread1
            }
      }
    }
}这样,一个简单的多线程程序就算完成了。
结语:
    希望本文能给那些徘徊在多线程门口的朋友带来些许帮助,也希望大家能多多分享自己的心得体会。



文档来源:51CTO技术博客https://blog.51cto.com/u_14036626/3180229
页: [1]
查看完整版本: 多线程基础