dofi

dofi一个人的日子......

 

使用BackgroundWorker做多线程,SoEasy

  最近我们在学校实验室学习,老师要求用uc/os-II实现一个多线程,恰好在暑期实习的公司也做过多线程的一个东西,于是想着分享出来,同时也想把这个实例记录下来,以后用到的时候再看看。

  首先开始写这个实例之前,我想先说说为什么要用这个BackgroundWorker类。我们经常会遇到这样的情况,就是后台有大量的计算,而前台要不停的根据后台的进度调整前台的界面显示,于是微软针对这个情况实现了BackgroundWorker这个类来帮助程序员方便的解决这个问题。

  对于这个问题和这个类,微软的MSDN是这样描述的:BackgroundWorker 类允许您在单独的专用线程上运行操作。耗时的操作(如下载和数据库事务)在长时间运行时可能会导致用户界面 (UI) 似乎处于停止响应状态。如果您需要能进行响应的用户界面,而且面临与这类操作相关的长时间延迟,则可以使用 BackgroundWorker 类方便地解决问题。

  我在公司做的这个东西的目的是这样的:公司以前做了一个教务管理系统,但以前使用的是Access数据库,但现在要升级成SQLserver数据库,将以前的acess数据库中的数据导SQLserver中,于是就需要这么一个界面显示导入数据的进度。  

  以前没做过这东西,于是瞎子摸象,用VS2010用VB拖控件做,做着做着,问题不断涌现,我的想法是用表格做,再在里面添加想要的控件,这样看起来比较整齐一些,但是这样做麻烦,让同事解决这些问题时,他说你自己做一个用户控件就行了,不需要这么麻烦。哎,有时候,人的想法可以,但是实施起来麻烦,经过他这么一指点,唰唰唰,几下就做了一个用户控件,然后我开始布局这些控件的位置,他给我的建议是:进度条的多少根据后台数据库的数据的多少动态生成,这个好办,好在我学过java,做过类似的东西,不过那时候界面是定死的,控件不能随外界因素变化。好了,我感觉我的废话有点多了,开始这个实例。

  首先在这个界面中最重要的就是这个进度条,首先我们设计这个进度条组合控件,在此以截图说明:

  以上这个就是我们要的进度条组合控件,简单说明一下,Lable:显示是哪类数据;后面那个图标是动态的,说明数据正在处理中,logFile:打开日志文件;下面是进度条,后面是百分比显示。

  这个做好之后,你可以将这个控件加到其他windows窗口上,布局你自己定,但是要经过一定的计算,确定他们的相对位置。

  经过做这个东西,我领悟到一个原则,就是一个控件你的所有功能只对你自身进行操作,不要牵扯其他的东西,我做这个就是只对我做的控件进行操作,从做这个学到这个也算一个收获,以下是对这个控件的处理代码,用vb写的:

  ProgressControl.vb

  

Imports System.Drawing
Imports System.Windows.Forms
Imports System.ComponentModel
Imports System.Diagnostics
Imports System.IO


Public Class ProgressControl

    'click the logButton, then open the logFile 
    Private logFile As String

    'when the num come to maxValue, the progress is 100%
    Private maxValue As Integer = 1

    'record the finishing count
    Private finishCount As Integer = 1

    'handle of set the statePic' image
    Private Delegate Sub SetPicImageHandle(ByVal img As Image)

    'set the statePic image
    Private Sub SetPicImage(ByVal img As Image)
        If Me.statePic.InvokeRequired Then
            Dim handle As SetPicImageHandle = New SetPicImageHandle(AddressOf SetPicImage)
            Me.statePic.Invoke(handle, New Object() {img})
        Else
            Me.statePic.Image = img
        End If
    End Sub

    'handle of set the statePic' visible
    Private Delegate Sub SetPicVisibleHandle(ByVal v As Boolean)

    'set the statePic' visible
    Private Sub SetPicVisible(ByVal visible As Boolean)
        If Me.statePic.InvokeRequired Then
            Dim handle As SetPicVisibleHandle = New SetPicVisibleHandle(AddressOf SetPicVisible)
            Me.statePic.Invoke(handle, New Object() {visible})
        Else
            Me.statePic.Visible = visible
        End If
    End Sub

    'handle of set the logButton' visible
    Private Delegate Sub SetLogButtonVisibleHandle(ByVal v As Boolean)

    'set the logButton' visible
    Private Sub setLogButtonVisible(ByVal visible As Boolean)

        If Me.logButton.InvokeRequired Then
            Dim handle As SetLogButtonVisibleHandle = New SetLogButtonVisibleHandle(AddressOf setLogButtonVisible)
            Me.logButton.Invoke(handle, New Object() {visible})
        Else
            Me.logButton.Visible = visible
        End If

    End Sub

    'open the log file
    Private Sub logButton_Click(sender As System.Object, e As System.EventArgs) Handles logButton.Click

        Try
            Process.Start(logFile)
        Catch ex As FileNotFoundException
            MessageBox.Show("FileNotFoundException!")
        Catch ex As Win32Exception
            MessageBox.Show("the file isn'n exist or the file name is wrong!", "Win32Exception")
        End Try

    End Sub

    'set the statePic' image and whether show it or not
    Public Sub SetStatePic(ByVal img As Image, ByVal visible As Boolean)

        'set statePic' image
        SetPicImage(img)

        'set the statePic' visible
        SetPicVisible(visible)

    End Sub

    'set the progressbar' value and percentLabel' text
    Public Sub SetProgressValue(ByVal value As Integer)

        If value < 0 Then
            Throw New ArgumentException("the Sub SetProgressValue()' parameter is wrong!")
        Else
            Try

                Dim finishPercent As Integer = value / maxValue * 100
                Dim progressBarPercnet As Integer = progressBar.Step * finishCount

                If finishPercent = 100 Then

                    progressBar.Value = 100
                    percentLabel.Text = "100%"

                ElseIf finishPercent > 0 AndAlso finishPercent = progressBarPercnet Then

                    'change progressBar' length
                    progressBar.PerformStep()

                    'change the percentLable' text
                    percentLabel.Text = (finishCount * progressBar.Step).ToString & "%"
                    finishCount = finishCount + 1

                End If

            Catch ex As OverflowException

                MessageBox.Show(ex.Message)

            Catch e As Exception

                MessageBox.Show(e.Message)

            End Try
            
        End If

    End Sub

    'whether show the logButton or not
    Public Sub SetLogButton(ByVal visible As Boolean)

        logButton.Visible = visible

    End Sub

    'set the logFileName
    Public Sub SetLogFileName(ByVal name As String)

        logFile = name

    End Sub

    'set the itemLabel' text
    Public Sub SetItemName(ByVal name As String)

        itemLabel.Text = name

    End Sub

    'handle of set the progressBar' step
    Private Delegate Sub AppendProgressBarStepHandle(ByVal value As Integer)

    'set the progressBar' step
    Private Sub AppendProgressBarStep(ByVal value As Integer)
        If Me.progressBar.InvokeRequired Then
            Dim handle As AppendProgressBarStepHandle = New AppendProgressBarStepHandle(AddressOf AppendProgressBarStep)
            Me.progressBar.Invoke(handle, New Object() {value})
        ElseIf value <= 0 Then
            Throw New ArgumentException("the Sub AppendProgressBarStep()' parameter is wrong!")
        Else
            Me.progressBar.Step = value
        End If
    End Sub

    'handle of set the maxValue
    Private Delegate Sub AppendMaxValueHandle(ByVal value As Integer)

    'set the maxValue
    Private Sub AppendMaxValue(ByVal value As Integer)
        If Me.progressBar.InvokeRequired Then
            Dim handle As AppendMaxValueHandle = New AppendMaxValueHandle(AddressOf AppendMaxValue)
            Me.progressBar.Invoke(handle, New Object() {value})
        ElseIf value <= 0 Then
            Throw New ArgumentException("the Sub AppendMaxValue()' parameter is wrong!")
        Else
            Me.maxValue = value
        End If
    End Sub

    'set the maxValue
    Public Sub SetMaxValue(ByVal value As Integer)
        AppendMaxValue(value)
    End Sub

    Public Sub SetProgressBarStep(ByVal value As Integer)
        AppendProgressBarStep(value)
    End Sub

End Class

 

  

 

  上面的注释是用英语写的,但都是经常见的英语单词,不难理解,代码是VB.NET,vb很好理解,难的就是其中的委托,有兴趣的朋友可以看看这部分内容,感觉很有用,在这不多讲了。因为公司的这个系统是为日本做的,所以他们用的是日系windows操作系统,为了不因汉字引起问题,所以同事让我以英语加注释,很简单,没什么复杂的,我本人也是希望东西越简单越好,整那么复杂干嘛,想显示自己的高端,不,我可不这样想的。乔布斯设计苹果的原则就是简单,说这话感觉有点自恋了,呵呵...

  为了不占用太大的空间,我将图片缩小了,但是还能看到其中的内容,可以说明问题就行了。再看看怎样布局这些控件的,这部分就要认真计算一下:

  ProgressContainer.vb

  

Imports System.Drawing
Imports System.Windows.Forms
Imports System.Threading

Public Class ProgressContainer
    Implements IProgress

    'operationControl' size
    Private Const OPERATIONCONTROL_HEIGHT As Integer = 70

    'controls' space value
    Private Const SPACELENGTH As Integer = 30

    'totalStateLabel' max length
    Private Const TOTALSTATE_MAXLENGTH = 230

    'the panel' width of progressContainer
    Private Const PANEL_WIDTH = 500

    'the progressControl' count can be show in the panel
    Private Const PROGRESSCONTROL_MAXCOUNT = 5

    'the closeFlag is used to control whether the closeBox is available
    Private closeFlag As Boolean = False


    Sub New()

        ' This call is required by the designer.
        InitializeComponent()

        ' Add any initialization after the InitializeComponent() call.

        'set bottom controls' properties
        totalStateLabel.Text = "处理中..."
        totalStatePic.Image = Global.ProgressClass.My.Resources.loading
        OKButton.Enabled = False

        'set the MinimizeBox and MaximizeBox' Initial state
        Me.MinimizeBox = True
        Me.MaximizeBox = False

        'the TotalProgress form can't be closed
        closeFlag = False

    End Sub


    Private Sub TotalProgress_Unload(ByVal sender As System.Object, ByVal e As System.ComponentModel.CancelEventArgs) Handles MyBase.FormClosing

        If closeFlag = True Then

            'MessageBox.Show("make sure to close the dialog box", "CloseDialog", MessageBoxButtons.YesNo)

            If MessageBox.Show("make sure to close the dialog box", "CloseDialog", MessageBoxButtons.YesNo) = Windows.Forms.DialogResult.Yes Then
                Return
            Else
                e.Cancel = True
            End If

        Else
            e.Cancel = True
        End If

    End Sub

    Private Sub OKButton_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles OKButton.Click
        Me.Close()
    End Sub

    'the first string of Dictionary is progressControl' Name,the second is logFileName
    Public Sub CreateNewProgress(allProgress As System.Collections.Generic.Dictionary(Of String, String)) Implements IProgress.CreateNewProgress

        Dim progressControl As ProgressControl
        Dim index As Integer = 0
        Dim progressControlMaxCount As Integer = 0

        If allProgress.Count <= 0 Then
            MessageBox.Show("the Sub sendOperatonItems()' parameter has no values, please check it!")
        End If

        Try
            For Each key As String In allProgress.Keys

                progressControl = New ProgressControl()

                'set the progressControl' name
                progressControl.SetItemName(key)

                'set the logFile name
                progressControl.SetLogFileName(allProgress(key))

                'set the progressBar' value and percentLabel' text
                progressControl.percentLabel.Text = "0%"

                'set the statePic' image and show it or not
                progressControl.SetStatePic(Global.ProgressClass.My.Resources.loading, False)

                'whether show the logButton
                progressControl.SetLogButton(False)

                'set operationControl’location
                progressControl.Top = index * OPERATIONCONTROL_HEIGHT

                'add the progressControl to the progressContainer' panel
                panel.Controls.Add(progressControl)

                'set the operationControl' index to easy access it
                panel.Controls.SetChildIndex(progressControl, index)

                index = index + 1

            Next

            'set the value of most can display of progressControl
            If allProgress.Count > 5 Then
                progressControlMaxCount = PROGRESSCONTROL_MAXCOUNT
            Else
                progressControlMaxCount = allProgress.Count
            End If

            'set totalStateLabel totalStatePic and OKButton' location
            totalStateLabel.Location = New Point(2 * SPACELENGTH, progressControlMaxCount * OPERATIONCONTROL_HEIGHT + 2 * SPACELENGTH)
            totalStatePic.Location = New Point(Left + TOTALSTATE_MAXLENGTH + 4 * SPACELENGTH, progressControlMaxCount * OPERATIONCONTROL_HEIGHT + 2 * SPACELENGTH)
            OKButton.Location = New Point(Left + TOTALSTATE_MAXLENGTH + 6 * SPACELENGTH, progressControlMaxCount * OPERATIONCONTROL_HEIGHT + 2 * SPACELENGTH)

            'set the ProgressContainer' size
            Me.Size = New Size(PANEL_WIDTH + 2 * SPACELENGTH, progressControlMaxCount * OPERATIONCONTROL_HEIGHT + 5 * SPACELENGTH)

        Catch ex As InvalidOperationException
            MessageBox.Show("InvalidOperationException!")
        End Try

    End Sub

    Public Sub SetStartImg(progressNum As Integer) Implements IProgress.SetStartImg

        Dim progressControl As ProgressControl

        Try
            progressControl = panel.Controls.Item(progressNum)

            'change the statePic' image and show it
            progressControl.SetStatePic(Global.ProgressClass.My.Resources.loading, True)
        Catch ex As Exception
            MessageBox.Show("the value or type of Sub start()' parameter is wrong,please check it!")
        End Try

    End Sub

    Public Sub SetProgressValue(progressNum As Integer, progressValue As Integer) Implements IProgress.SetProgressValue

        Dim progressControl As ProgressControl

        Try
            progressControl = panel.Controls.Item(progressNum)

            'change the progressBar' value and percnetLabel' text
            progressControl.SetProgressValue(progressValue)
        Catch ex As Exception
            MessageBox.Show("the value or type of Sub sendProgressValue()' first parameter is wrong,please check it!")
        End Try

    End Sub


    Public Sub SetResultImg(progressNum As Integer, result As Boolean) Implements IProgress.SetResultImg

        Dim progressControl As ProgressControl

        Try
            progressControl = panel.Controls.Item(progressNum)

            'change the statePic to show the statePic' result
            If result = True Then
                progressControl.SetStatePic(Global.ProgressClass.My.Resources.yes, True)
            Else
                progressControl.SetStatePic(Global.ProgressClass.My.Resources.no, True)
            End If

        Catch ex As Exception
            MessageBox.Show("the value or type of Sub sendOperationResult()' first parameter is wrong,please check it!")
        End Try

    End Sub

    Public Sub FinishAllProgress() Implements IProgress.FinishAllProgress

        Dim progressControl As ProgressControl

        For i As Integer = 0 To panel.Controls.Count - 1

            progressControl = panel.Controls.Item(i)

            'to show the logButton
            progressControl.SetLogButton(True)

        Next

        'set the properties of totalStateLabel, totalStatePic and OkButton
        totalStateLabel.Text = "处理完了"
        totalStatePic.Image = Global.ProgressClass.My.Resources.yes
        OKButton.Enabled = True

        'the TotalProgress form can be closed
        closeFlag = True

    End Sub


    Public Sub SetMaxValue(progressNum As Integer, value As Integer) Implements IProgress.SetMaxValue

        Dim progressControl As ProgressControl

        progressControl = panel.Controls.Item(progressNum)

        'set the maxValue
        progressControl.SetMaxValue(value)

    End Sub

    Public Sub SetProgressBarStep(progressNum As Integer, value As Integer) Implements IProgress.SetProgressBarStep

        Dim progerssControl As ProgressControl

        progerssControl = panel.Controls.Item(progressNum)

        'set progressBar' step
        progerssControl.SetProgressBarStep(value)

    End Sub

End Class

 

  

 

  代码注释还是英文,方法名都是见名知意,在公司学的另一个就是不管给什么起名字,都要让别人知道你这里是干什么的,不能明白全部,也要略知一二。还有就是你的代码风格,要整齐有序。这个布局是这样设计的:按照你要处理的数据类型多少,自动显示多少,超过五个进度条组合控件时,只显示五个,多出的显示一个滚动条,然后拖动滚动条你就会看到多出的那些。

  好了,这个前台显示界面就算做好了,说明一下我只负责制作这个界面,那要给别人用,所以没有必要让别人知道其中的内容,只要设置一些接口即可,别人想要对你这个界面做怎样的操作,给他个接口就行。下面是这些接口:

  IProgress.vb

  

Public Interface IProgress

    'set all ' name and logfiles' name
    Sub CreateNewProgress(ByVal allProgress As Dictionary(Of String, String))

    'set the statePic' image when start one progress
    Sub SetStartImg(ByVal progressNum As Integer)

    'set the progress value of the progress item you want 
    Sub SetProgressValue(ByVal progressNum As Integer, ByVal progressValue As Integer)

    'set the result' image of one progress when it was finish
    Sub SetResultImg(ByVal progressNum As Integer, ByVal result As Boolean)

    'when you complete all Progress, you can call this sub
    Sub FinishAllProgress()

    'set the progressBar' step
    Sub SetProgressBarStep(ByVal progressNum As Integer, ByVal value As Integer)

    'set the maxValue
    Sub SetMaxValue(ByVal progressNum As Integer, ByVal value As Integer)

End Interface

 

  

 

  其实上面这些只是为我们应用BackgroundWorker做铺垫,好了,看看我们的主界面是怎样的:

  没搞错吧,就一个button,对,就只有这个,因为这个只是对我们上面做的那个界面的测试,对这个界面测试就要用到多线程。我们先来看看主界面的代码:

  MainForm.vb

  

Imports ProgressClass
Imports ProgressClass.ProgressContainer
Imports System.ComponentModel

Public Class MainForm

    Dim check As ProgressContainer
    Dim progresses As Dictionary(Of String, String)

    Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click

        worker.WorkerReportsProgress = True
        worker.WorkerSupportsCancellation = True

        'set the total form' 
        startProgress()

        'start the disposing program of Dowork' event 
        worker.RunWorkerAsync()

    End Sub


    Private Sub worker_DoWork(sender As System.Object, e As System.ComponentModel.DoWorkEventArgs) Handles worker.DoWork

        Dim workerTmp As BackgroundWorker = CType(sender, BackgroundWorker)

        'call the dispoding program of event 
        Work(workerTmp)

    End Sub

    'receive the updating progress notice
    Private Sub worker_ProgressChanged(sender As System.Object, e As System.ComponentModel.ProgressChangedEventArgs) Handles worker.ProgressChanged
        Dim index As Integer = CType(e.UserState, Integer)
        Dim progressValue As Integer = CType(e.ProgressPercentage, Integer)

        'update the progressBar' value
        check.SetProgressValue(index, progressValue)

    End Sub

    Private Sub worker_RunWorkerCompleted(sender As System.Object, e As System.ComponentModel.RunWorkerCompletedEventArgs) Handles worker.RunWorkerCompleted
        'when the Work was completed, change the totalStateLabel' text, totalStatePic' image and make the OKButton visible
        check.FinishAllProgress()
    End Sub

    Private Sub startProgress()

        check = New ProgressContainer()
        progresses = New Dictionary(Of String, String)()

        'add the items' name and log files' position
        progresses.Add("1", "e:/aaa.txt")
        progresses.Add("3", "456")

        check.CreateNewProgress(progresses)
        check.Show()
    End Sub

    Private Sub Work(ByVal workerTmp As BackgroundWorker)

        If progresses.Count > 0 Then

            For i As Integer = 0 To progresses.Count - 1

                check.SetStartImg(i)
                check.SetMaxValue(i, 1000)
                check.SetProgressBarStep(i, 9)

                For j As Integer = 0 To 1000
                    System.Threading.Thread.Sleep(10)

                    Try
                        workerTmp.ReportProgress(j, i)
                    Catch ex As InvalidOperationException
                        MessageBox.Show("InvalidOperationException")
                    End Try

                    workerTmp.ReportProgress(j, i)
                Next

                check.SetResultImg(i, True)

            Next

        End If

    End Sub

End Class

 

  

 

  下面重点介绍一下上面这部分代码,不能本末倒置嘛,要不然就文不对题了,呵呵...

  因为这个界面只有一个按钮,所以程序应该是从单击这个按钮开始的:

  Button1_Click()这个方法响应Check按钮,单击后,先做一些初始化工作,

  worker.WorkerReportsProgress :获取或设置一个值,该值指示 BackgroundWorker 能否报告进度更新。

      worker.WorkerSupportsCancellation :获取或设置一个值,该值指示 BackgroundWorker是否支持异步取消。

  startProgress()是对前台界面的初始化,基本上初始化的工作都是在ProgressContainer.vb完成的,你可以沿着调用关系去看看这些。

  worker.RunWorkerAsync()就是执行后台操作,直接运行work()方法,其中就是一些循环,我只是模拟,目的是为了证明我这个程序没错,也不能说没错,只能说是能完成基本的操作,因为我这个程序还没有经过专业人员的测试,所以不敢打这个包票。

  workerTmp.ReportProgress(j, i)就是传给前台的一些数据,这个方法引发ProgressChanged事件,第一个参数是进度条值,第二个是操作哪个进度条。

  Dim index As Integer = CType(e.UserState, Integer)

     Dim progressValue As Integer = CType(e.ProgressPercentage, Integer)

  以上这两行代码取到这个进度条的标志值和进度值。

  check.SetProgressValue(index, progressValue) :就是对这个进度条组合控件进行操作。

    RunWorkerCompleted() : 当后台操作已完成、被取消或引发异常时发生,我们这部分只是对结束后台运算时,前台所要最后做的处理,我们只调用了check.FinishAllProgress()。

  这部分算是介绍完了,看看效果:

  如果您有什么意见或者建议,请给我留言,相互学习,对我也是一个提高。

  

posted on 2011-10-20 11:25  dofi  阅读(1996)  评论(5编辑  收藏  举报

导航