2016-07-20 9 views
0

Hello again StackOverflow 커뮤니티!VB.Net 스레딩 및 Addhandler 문제

"SendLogfileClass"클래스에서 작업하고 있습니다. 이 수업에서는 이메일을 통해 로그 파일을 이메일 계정으로 보냅니다. 그 부분은 의도 한대로 작동합니다. 내가 겪고있는 문제는 비동기 완료 이벤트를 처리하려고하는 것입니다. 이 이벤트 동안 Addhandler는 기본 폼에 StatusBar.StatusLabel을 발생시키고 설정합니다.

#Region "Imports" 

    Imports System 
    Imports System.Net 
    Imports System.Net.Mail 
    Imports System.Net.Mime 
    Imports System.Threading 
    Imports System.ComponentModel 
    Imports System.IO 

#End Region 

Public Class Form1 

#Region "Public" 

    Private SendmailThread As Thread 
    Private MailBody As String = Nothing 

#End Region 

#Region "Private" 

    Private mailSent As Boolean = False 

#End Region 

    Public Function GetTimestamp() As String 

     Dim t As Date = Date.Now 
     Dim timestamp As String = Nothing 

     Try 

      timestamp = t.ToLongTimeString & " " & t.ToLongDateString 

     Catch ex As Exception 

      Return 1 

     End Try 

     Return timestamp 

    End Function 

    Private Sub Form1_Load(sender As Object, e As EventArgs) Handles MyBase.Load 

     Try 

      If LoggerClock.Enabled = True Then 

       OutputConsole.Text = "logger Started: " & GetTimestamp() & vbNewLine 
       OutputConsole.AppendText("Logfile Opened: " & GetTimestamp() & vbNewLine) 

       StatusLabel.Text = "Logger Status: Active" 
       StatusBar.Refresh() 

      Else 

       OutputConsole.Text = "logger Started: " & GetTimestamp() & vbNewLine 

       StatusLabel.Text = "Logger Status: Inactive" 
       StatusBar.Refresh() 
      End If 

      SendlogClock.Enabled = True 

      ToggleViewForm(1) 

     Catch ex As Exception 

      Exit Sub 

     End Try 

    End Sub 

    Public Function SetStatus(ByVal [status] As String) As Integer 

     Try 

      Thread.Sleep(1000) 
      StatusLabel.Text = [status] 
      StatusBar.Refresh() 

     Catch ex As Exception 

      Return 1 

     End Try 

     Return 0 

    End Function 

    Private Sub SendlogThreadTask() 

     Try 

      SendLogfile("[email protected]", "Logger Logfile", MailBody).ToString() 

     Catch ex As Exception 

      Exit Sub 

     End Try 

    End Sub 

    Private Sub SendlogClock_Tick(sender As Object, e As EventArgs) Handles SendlogClock.Tick 

     Try 

      OutputConsole.AppendText("Logfile Closed: " & GetTimestamp() & vbNewLine) 

      SendmailThread = New Thread(AddressOf SendlogThreadTask) 
      SendmailThread.IsBackground = True 
      SendmailThread.Start() 

      OutputConsole.ResetText() 
      OutputConsole.Text = "Logfile Opened: " & GetTimestamp() & vbNewLine 

     Catch ex As Exception 

      Exit Sub 

     End Try 

    End Sub 

    Public Sub SendCompletedCallback(ByVal sender As Object, ByVal e As AsyncCompletedEventArgs) 

     Try 

      ' Get the unique identifier for this asynchronous operation. 
      Dim token As String = CStr(e.UserState) 

      If e.Cancelled Then 

       StatusLabel.Text = "Send Canceled... " & token 
       StatusBar.Refresh() 

      End If 

      If e.Error IsNot Nothing Then 

       StatusLabel.Text = "Error: " & token & " " & e.Error.ToString() & " " 
       StatusBar.Refresh() 

      Else 

       StatusLabel.Text = "Message Sent... " 
       StatusBar.Refresh() 

      End If 

      mailSent = True 

     Catch ex As Exception 

      Exit Sub 

     End Try 

    End Sub 

    Public Function SendLogfile(ByVal mailTo As String, ByVal mailSubject As String, ByVal mailBody As String, Optional ByVal doAttach As Boolean = False, Optional ByVal messageAttach As String = Nothing) As Integer 

     Try 

      ' SMTP Server 
      Dim SmtpServer As String = "mail.domain.com" 

      ' Command line argument must the the SMTP host. 
      Dim Cli As New SmtpClient(SmtpServer) 

      ' Specify the e-mail sender. 
      ' Create a mailing address that includes a UTF8 character 
      ' in the display name. 
      Dim [from] As New MailAddress("[email protected]", "logger " & ChrW(&HD8) & " logs", System.Text.Encoding.UTF8) 

      ' Set destinations for the e-mail message. 
      Dim [to] As New MailAddress(mailTo) 

      ' Specify the message content. 
      Dim message As New MailMessage([from], [to]) 

      message.Body = mailBody 

      ' Include some non-ASCII characters in body and subject. 
      Dim someArrows As New String(New Char() {ChrW(&H2190), ChrW(&H2191), ChrW(&H2192), ChrW(&H2193)}) 

      message.Body += Environment.NewLine & someArrows 
      message.BodyEncoding = System.Text.Encoding.UTF8 
      message.Subject = mailSubject & someArrows 
      message.SubjectEncoding = System.Text.Encoding.UTF8 

      ' Put the mail attachment in a list of items 
      'Dim attachment As New Attachment(messageAttach) 

      ' Attach file. 
      'If doAttach = True Then 

      'If File.Exists(messageAttach) Then 

      'message.Attachments.Add(attachment) 

      'End If 

      'End If 

      ' Set the method that is called back when the send operation ends. 
      AddHandler Cli.SendCompleted, AddressOf SendCompletedCallback 

      ' The userState can be any object that allows your callback 
      ' method to identify this send operation. 
      ' For this example, the userToken is a string constant. 
      Dim userState As String = "OK" 

      Cli.SendAsync(message, userState) 

      'MsgBox("Sending message... press c to cancel mail. Press any other key to exit.") 

      Dim answer As String = "OK" ' or CANCEL 

      ' If the user canceled the send, and mail hasn't been sent yet, 
      ' then cancel the pending operation. 
      If answer.StartsWith("C") AndAlso mailSent = False Then 

       Cli.SendAsyncCancel() 

      End If 

      ' Clean up. 
      message.Dispose() 

     Catch ex As Exception 

      MsgBox("Encountered Error: " & vbNewLine & vbNewLine & ex.ToString()) 
      Return 1 

     End Try 

     Return 0 

    End Function 

End Class 
+0

동일한 응용 프로그램의 단순화 된 단일 클래스 버전을 추가 했으므로 원하는 작업을 수행하는 방법에 대한 예제를 제공해 주시겠습니까? –

답변

0

이벤트 핸들러는 보조 스레드에서 실행되고 해당 이벤트 처리기에서 당신이 MainForm의 기본 인스턴스를 참조하고 있습니다 :

여기에 코드의 몇 가지 관련 덩어리입니다. 기본 인스턴스는 스레드마다 다르므로 화면에서보고있는 것과 다른 양식 객체입니다.

일반적으로 SynchronizationContext 클래스를 사용하여 UI 스레드에 대한 메서드 호출을 마샬링 할 수 있지만 실제로는 보조 스레드에도 개체를 실제로 만들고 있기 때문에 가능하지 않습니다. 이 경우 기존 MainForm 개체에 대한 참조를 해당 메일 보낸 사람에게 전달해야하며이를 사용하여 InvokeRequiredInvoke/BeginInvoke 멤버를 사용하여 UI 스레드에 대한 메서드 호출을 마샬링합니다.