C#網絡編程系列十:實現簡單的郵件收發器

知道電子郵件的應用程序的原理也是非常有必要的,在這一個專題中將介紹電子郵件應用程序的原理、電子郵件應用程序中涉及的協議和實現一個簡答的電子郵件收發器程序。

引言:在我們的平常工作中,郵件的發送和接收應該是我們經常要使用到的功能的。因此知道電子郵件的應用程序的原理也是非常有必要的,在這一個專題中將介紹電子郵件應用程序的原理、電子郵件應用程序中涉及的協議和實現一個簡答的電子郵件收發器程序。

一、郵件應用程序基本知識

1.1 電子郵件原理及相關協議

說到電子郵件的原理,其實和我們現實生活中寄郵件和寄包裹是一樣的原理的。就讓我們先回顧下現實生活中寄郵件的流程吧——首先,我們先寫好信,信封上面寫好收信人的地址,寫信人的地址,然後把信放到寄信箱中,然後郵局的人會某個時候去這個信箱中的信取出來,然後郵局的人根據信封上寫的收信人地址進行轉發到當地的郵局,當地郵局然後把信寄到收信人的信箱中(寄包裹的話可能會電話聯繫,像我們在淘寶,京東買的東西的,收貨人就是通過電話聯繫一樣),最後收信人會到自己的信箱中收取信件。上面大致是我們平時生活中寄信的一個流程的。前面已經講過電子郵件的原理和這個差不多的,下面就介紹了本專題中電子郵件的原理,大家可以和現實生活中的寄信過程進行對比下的,這樣可以更加容易理解和掌握:

我們通過電子郵件應用(例如 基於客戶端的Outlook電子郵件軟件 和一些基於Web的電子郵件系統——新浪郵箱、谷歌郵箱、QQ郵箱等都屬於電子郵件應用)將一封寫好的郵件(相當於現實生活中的信,當然郵件也要寫明收件人地址,郵件內容等信息的)通過電子郵件協議(SMTP,在後面的電子郵件相關協議中會介紹)發送到SMTP服務器(就是存儲郵件的地方,相當於生活中的郵局一樣),然後SMTP服務器根據收件人的地址通過SMTP協議轉發到相應SMTP接收服務器上,(SMTP服務器進行轉發相當於現實生活中郵局的人配送信的過程,配送到收件人當地的郵局,然而現實生活中郵局都是一家,所以可以相互識別——意思就是發送到當地郵局,當地郵局會接收,並且幫助你發送到指定人的信箱中,在網上上就是通過SMTP協議來規定這樣的一個過程的,發送到別人的SMTP服務器上別人的服務器必須要認識發送來的郵件並接收)結束,接收端郵件服務器(POP3服務器)把郵件存放到接受者的電子信箱內(相當於當地郵局的人把信放到收信人的郵箱中),最後收件人可以登錄自己的電子信箱,再與POP3服務器進行連接,從POP3服務器上下載發送來的郵件,這樣在收件人的電子信箱中就可以看到發送來的電子郵件了(這就是現實生活中收信人從自己的信箱中取信的一個過程)。

注:括號中都是個人的理解,如果有什麼不對的地方還望大家指出來,我好及時更正。

上面已經把電子郵件的原理和現實生活中寄信的過程進行對比,相信大家可以更加清楚電子郵件的原理和發送接收過程的,其實網絡上的很多應用都可以以現實生活的例子去理解,這樣的話我認爲可以加深對知識的理解。下面就介紹下電子郵件中的相關協議的內容:

網絡上的應用的核心就是協議,因爲協議讓網絡上的客戶端相互認識發生來的數據,所以電子郵件應用也不例外,也有相關的電子郵件協議來完成發送電子郵件和接收電子郵件的過程,這些協議主要是:SMTP(簡單郵件傳輸協議,Simple Mail Transfer Protocol)、POP3(郵局協議,Post Office Protocol)和IMAP(網絡郵件訪問協議,Internet Message Access Protocol)。

SMTP——SMTP 主要負責將郵件從一臺機器轉發至另一臺機器(可以對照上面電子郵件的過程來理解SMTP的作用)

POP3——3表示POP協議的版本,主要負責將郵件從郵箱中(POP3服務器)傳輸到本地計算機。

IMAP——現在常用的版本爲第四版本,即IMAP4,主要負責郵件的檢索和處理功能,客戶端不需要下載郵件到本地計算機,可直接從郵件客戶端軟件對服務器上的信件和文件目錄進行操作,它是POP3的替代協議的。

1.2 郵件系統的分類

郵件系統主要分爲兩類的——基於客戶端的郵件系統和基於Web瀏覽器的郵件系統。Office OutLook就是基於客戶端的郵件客戶端系統,而像我們經常使用的QQ郵箱、新浪、網易郵箱等都是屬於基於Web瀏覽器的郵件系統,基於客戶端的郵件系統的收發過程,通過下面的圖片來描述(圖片從網上摘下的):

圖 1.1 基於客戶端的郵件收發過程

發送方通過郵件客戶端,將編輯好的郵件向郵件服務(SMTP服務器,在發送過程中也叫發送端郵件服務器)發送,發送端郵件服務器根據收件人的地址來識別接收端郵件服務器(POP3服務器),然後向POP3服務器發送郵件信息,接收端郵件服務器將郵件存放在接收者的電子信箱中,並告知接收者有新郵件,接收者通過郵件客戶端與POP3服務器連接後,就可以查看新郵件。

然而,基於Web瀏覽器的郵件系統與基於客戶端的郵件系統不同的地方有:

基於Web瀏覽器郵件系統用戶代理(代理的概念也就是用戶不是直接與服務器進行通信,而是通過代理的方式,讓代理去與服務器通信,然後用戶在從代理中獲的服務器的信息,代理也就是中間人的作用,相當於生活中中介,在.net中很多技術都用到了代理,例如委託的概念其實也就是代理的一個概念的)是Web瀏覽器,基於客戶端的郵件系統而是郵件客戶端應用程序,一般是Windows Form程序。

瀏覽器發送郵件到SMTP服務器和從POP3服務器中獲得郵件的方式都是通過HTTP協議來實現,與基於客戶端的郵件系統不同(基於客戶端的郵件系統發送通過SMTP協議或ESMTP(Extended SMTP),獲得通過POP3或IMAP協議)。

1.3 目前主要的電子郵件服務系統

電子郵件服務系統——就是向大家提供郵箱服務的服務系統,這樣的系統當然是由專門的公司進行研發的,我們一般叫這樣的公司爲郵件服務商,我們平常使用的網易郵箱,新、Gmail郵箱等都是建立在電子郵件服務系統(這裏我的理解是——我們使用的新浪,網易等郵箱相當於現實生活中每個人的信箱,通過信箱可以獲得郵局來的信,同樣道理通過郵箱可以獲得郵件服務系統的郵件,這樣電子郵件系統相當於郵局) 。現在主要電子郵件服務系統主要有下面幾種:

基於Postfix/Qmail的郵件系統。例如,雅虎郵箱基於Qmail系統

微軟Exchange 郵件系統

IBM Lotus Domino郵件系統

Scalix郵件系統

Zimbra郵件系統

MDeamon郵件系統

二、.Net 平臺對郵件發送功能的支持

在.NET類庫中,在System.Net.Mail命名空間下定義了對郵件處理的類,這樣使郵件的發送更加方便(這些類也就是對SMTP協議的封裝,使我們更好地區編程,只需要使用類中的方法和屬性等去完成郵件的發送,避免寫複雜的SMTP協議的命令),下面是一張在System.Net.Mail命名空間下對郵件發送的支持的類截圖:

從圖片中類的名字中也可以看出每個類的作用的,在這裏我就不一個介紹的, 大家可以參考MSDN去看每個類的使用,並且我在後面程序的實現部分也會有詳細的註釋去介紹程序中使用到類的使用。從圖中還可以i看出一點——就是隻有SMTP的字樣,卻沒有POP3這樣的字樣的,這說明.Net類庫本身中並沒有提供對POP3協議的封裝類,但是我們可以使用Jmail組件來完成從POP3服務器中收取郵件的功能,具體的使用將在後面的郵件收發器程序中郵件的接收部分介紹的。

三、郵件收發器程序的實現

3.1 郵件發送功能的實現

3.1.1 SMTP協議

SMTP 協議是用於電子郵件的傳輸的協議,電子郵件是通過SMTP服務器進行發送的,SMTP服務器的默認端口爲25,通常發送郵件有兩種方式——一種是不使用客戶端認證,即客戶端可以使用匿名發送郵件(這種方式叫做SMTP);另一種是客戶端必須提供用戶名和密碼認證(這種方式叫做ESMTP,Extended SMTP)目前大部分郵件服務器採用用戶名和密碼認證的方式。

客戶端發送郵件過程爲——先通過客戶端軟件(本程序中的郵件收發器)將郵件發送到SMTP服務器,然後再由SMTP服務器發送到目標SMTP服務器。下面介紹SMTP協議的內容:

SMTP協議總共定義了14個命令,命令由命令碼和氣候的參數域組成, 不區別大小寫的(通過前面專題的講述可以得出各個協議的命令組成都差不多的),下面就簡單介紹下5個常用的命令碼

電子郵件由信封、首部、正文和結束符號4部分組成,下面就具體介紹下這4個部分的內容:

1. 信封

信封包括髮信人的郵件地址和接收人的郵件地址,具體對應兩條SMTP命令——Mail from:[email protected](發信人的地址)和Rcpt to:  [email protected]

2. 首部

首部中常用的命令有:

Subject:<郵件主題>——表示郵件的主題

Date:<時間>——表示發郵件的時間

reply-to:<郵件地址>——表示郵件的回覆地址

Content-Type:<郵件類型>——表示郵件包含文本、HTML超文本和附件的類型。

X-Priority:<郵件優先級>——表示郵件發送的優先級,優先級爲3表示爲普通郵件;如 X-Priority:3

3. 正文

正文當然指的就是郵件的內容了, 用Data命令指定,首部以一個空行結束,下面就是正文部分

4. 結束符號

郵件以“."結束,

接收方收到SMTP命令之後,會給出一個響應碼,每個命令都只有一個響應碼,SMTP響應碼也是由3位數字組成,後面附加一些文本信息,響應信息的格式爲:

響應碼<空格>文本信息<回車換行>

客戶端發出一條命令後,服務器端返回一個響應,發送者在發送下一條命令前必須等待服務器的響應,成功接收到響應碼後才繼續發送命令。

附:SMTP常用的響應碼:

 

3.1.2 郵件的發送過程

第一步:客戶端與服務器建立連接(該步中客戶端首先發送EHLO local 連接命令,服務器如果返回“220”響應碼錶示服務器準備就緒了,客戶端再繼續發送“Auto login”命令,請求登錄,服務器收到命令後返回“334”響應碼,表示要輸入用戶名,之後客戶端發送用戶名命令,等到響應後再發送密碼命令,具體在程序的實現中也會有註釋。)

第二步:客戶端發送郵件的信封

第三步:開始發送郵件數據,(包括郵件首部,正文和結束符號,注:結束符號要單獨佔一行,表示郵件發送結束)

第四步: 客戶端與服務器斷開連接。

3.1.3 發送功能的實現代碼

相信有了上面的理論解釋郵件發送的過程後,實現郵件發送的功能並不難的,並且.net類庫中SMTPClient類幫我們封裝了SMTP協議,使得我們實現郵件發送功能就不要記住那些具體的命令了, 只需要使用該類中提供的方法來完成郵件的發送(當然你也可以通過發送命令的方式實現,SMTPClient類的方法也是幫我們完成發送命令功能而已的),下面是郵件發送功能的核心代碼:

  1. View Code   
  2.  #region 郵件發送功能代碼  
  3.          // 添加附件  
  4.          private void btnAddFile_Click(object sender, EventArgs e)  
  5.          {  
  6.              OpenFileDialog openFileDialog = new OpenFileDialog();  
  7.              openFileDialog.CheckFileExists = true;  
  8.              // 只接受有效的文件名  
  9.              openFileDialog.ValidateNames = true;  
  10.              // 允許一次選擇多個文件作爲附件  
  11.              openFileDialog.Multiselect = true;  
  12.              openFileDialog.Filter = "所有文件(*.*)|*.*";  
  13.              if (openFileDialog.ShowDialog() != DialogResult.OK)  
  14.              {  
  15.                  return;  
  16.              }  
  17.              if (openFileDialog.FileNames.Length > 0)  
  18.              {  
  19.                  // 因爲這裏允許選擇多個文件,所以這裏用AddRange而沒有用Add方法  
  20.                  cmbAttachment.Items.AddRange(openFileDialog.FileNames);  
  21.              }  
  22.          }  
  23.    
  24.          // 刪除附件  
  25.          private void btnDeleteFile_Click(object sender, EventArgs e)  
  26.          {  
  27.              int index = cmbAttachment.SelectedIndex;  
  28.              if (index == -1)  
  29.              {  
  30.                  MessageBox.Show("請選擇要刪除的附件!""提示", MessageBoxButtons.OK, MessageBoxIcon.Information);  
  31.                  return;  
  32.              }  
  33.              else 
  34.              {  
  35.                  cmbAttachment.Items.RemoveAt(index);  
  36.              }  
  37.          }  
  38.    
  39.          // 發送郵件  
  40.          private void btnSend_Click(object sender, EventArgs e)  
  41.          {  
  42.              this.Cursor = Cursors.WaitCursor;  
  43.              // 實例化一個發送的郵件  
  44.              // 相當於與現實生活中先寫信,程序中把信(郵件)抽象爲郵件類了  
  45.              MailMessage mailMessage = new MailMessage();  
  46.              // 指明郵件發送的地址,主題,內容等信息  
  47.              // 發信人的地址爲登錄收發器的地址,這個收發器相當於我們平時Web版的郵箱或者是OutLook中配置的郵箱  
  48.              mailMessage.From = new MailAddress(tbxUserMail.Text);  
  49.              mailMessage.To.Add(txbSendTo.Text);  
  50.              mailMessage.Subject = txbSubject.Text;  
  51.              mailMessage.SubjectEncoding = Encoding.Default;  
  52.              mailMessage.Body = richtbxBody.Text;  
  53.              mailMessage.BodyEncoding = Encoding.Default;  
  54.              // 設置郵件正文不是Html格式的內容  
  55.              mailMessage.IsBodyHtml = false;  
  56.              // 設置郵件的優先級爲普通優先級  
  57.              mailMessage.Priority = MailPriority.Normal;  
  58.              //mailMessage.ReplyTo = new MailAddress(tbxUserMail.Text);  
  59.    
  60.              // 封裝發送的附件  
  61.              System.Net.Mail.Attachment attachment = null;  
  62.              if (cmbAttachment.Items.Count > 0)  
  63.              {  
  64.                  for (int i = 0; i < cmbAttachment.Items.Count; i++)  
  65.                  {  
  66.                      string fileNamePath = cmbAttachment.Items[i].ToString();  
  67.                      string extName = Path.GetExtension(fileNamePath).ToLower();  
  68.                      if (extName == ".rar" || extName == ".zip")  
  69.                      {  
  70.                          attachment = new System.Net.Mail.Attachment(fileNamePath, MediaTypeNames.Application.Zip);  
  71.                      }  
  72.                      else 
  73.                      {  
  74.                          attachment = new System.Net.Mail.Attachment(fileNamePath,MediaTypeNames.Application.Octet);  
  75.                      }  
  76.    
  77.                      // 表示MIMEContent-Disposition標頭信息  
  78.                      // 對於ContentDisposition具體類的解釋大家可以參考MSDN  
  79.                      // 這裏我就不重複貼出來了,給個地址: http://msdn.microsoft.com/zh-cn/library/System.Net.Mime.ContentDisposition.aspx (着重看備註部分)  
  80.                      ContentDisposition cd = attachment.ContentDisposition;  
  81.                      cd.CreationDate = File.GetCreationTime(fileNamePath);  
  82.                      cd.ModificationDate = File.GetLastWriteTime(fileNamePath);  
  83.                      cd.ReadDate = File.GetLastAccessTime(fileNamePath);  
  84.                      // 把附件對象加入到郵件附件集合中  
  85.                      mailMessage.Attachments.Add(attachment);  
  86.                  }  
  87.              }  
  88.    
  89.              // 發送寫好的郵件  
  90.              try 
  91.              {  
  92.                  // SmtpClient類用於將郵件發送到SMTP服務器  
  93.                  // 該類封裝了SMTP協議的實現,  
  94.                  // 通過該類可以簡化發送郵件的過程,只需要調用該類的Send方法就可以發送郵件到SMTP服務器了。  
  95.                  smtpClient.Send(mailMessage);  
  96.                  MessageBox.Show("郵件發送成功!""提示", MessageBoxButtons.OK, MessageBoxIcon.Information);  
  97.    
  98.              }  
  99.              catch(SmtpException smtpError)  
  100.              {  
  101.                  MessageBox.Show("郵件發送失敗:[" + smtpError.StatusCode + "];["   
  102.                      + smtpError.Message+"];\r\n["+smtpError.StackTrace+"]." 
  103.                      ,"錯誤",MessageBoxButtons.RetryCancel,MessageBoxIcon.Error);  
  104.              }  
  105.              finally 
  106.              {  
  107.                  mailMessage.Dispose();  
  108.                  this.Cursor = Cursors.Default;  
  109.              }  
  110.          }  
  111.    
  112.          #endregion 

3.2 郵件接收功能的實現

3.2.1 POP3協議

前面介紹了郵件的發送,當然接收者需要登錄郵箱來查看收到的郵件了,此時就必有有一個協議去讀取服務器上郵件,POP3就是這樣的一個協議。還有兩外一種協議也是用來接收郵件的——IMAP協議,它與POP3協議區別有:

1. IMAP使用的端口號是143而POP3郵件服務器通過監聽110端口來提供POP3服務;

2 . IMAP 允許客戶端在郵件服務器上建立文件夾來保持郵件,而不用把郵件下載到本地。而POP3需要把郵件下載到本地。

和SMTP協議一樣,客戶端要通過POP3協議從POP3服務器上獲取郵件,也需要先與POP3服務器建立TCP連接,等待服務器向客戶端發送確認信息表明連接成功時,客戶端纔可以繼續發送命令給服務器來獲取郵件,在POP3協議中,規定的命令也是幾十條的,每條命令由命令和參數兩部分組成,都是以回車換行結束,並且命令和參數之間由空格分隔,命令通常也是由3-4個字母組成,參數最多可以爲40個字符的長度,而服務器的響應信息是由一個狀態碼和可能附加信息的字符組成,所有的響應信息也是以回車換行結束的。狀態碼和其他協議定義的狀態碼有點不一樣,POP3服務器響應的狀態碼有兩種——“+OK”(確定)和"-ERR"(失敗)。這樣客戶端可以通過檢查響應的狀態碼所包含的字符來判斷服務器是否響應客戶端發送的命令,即響應信息中包含“+OK”表示成功響應,包含“-ERR”表示服務器未響應。同時在程序的實現中大家可以通過Debug來查看響應消息的組成,這樣可以加深理解。

3.2.2 郵件接收的過程

客戶端從服務器接收郵件的過程主要經歷3個狀態:授權狀態、操作狀態和更新狀態

(1)授權狀態——客戶端發送與POP3服務器的TCP連接請求,服務器接收後發送一個響應確認信息之後,此時客戶端需要發送正確的用戶名和密碼進行確認,因爲在郵件服務器上有很多用戶郵箱,只有提供正確的用戶名和密碼纔有權限訪問自己的郵箱,就像現實生活中我們郵箱的鑰匙一樣的。

發送用戶名命令: USER [email protected]

發送密碼命令: PASS ******(這兩個命令都在代碼中有給出的,大家可以參考代碼來理解郵件的接收過程)

(2) 操作狀態——如果客戶端提供了正確的用戶名和密碼,則授權狀態也就通過了,就相當於打開了在服務器上自己的郵箱,現在用戶就有權限進去下載,查看和刪除郵件等操作的,然後在現實生活中的取郵件和刪除郵件都很簡單(只要打開了郵箱門,用手去拿就可以了),然後在網絡應用上,這些操作都需要發送POP3命令給服務器,服務器接收到命令後再給出響應。操作中常用的命令有:

STAT 命令——該命令從服務器中獲取郵件總數和總字節數,服務器響應命令返回郵件總數和總字節數。如:

  1. 客戶端發送POP3命令: STAT   
  2. 服務器響應命令: +OK 2 1340<BR>服務器響應命令: 

LIST 命令——該命令從服務器中獲得郵件列表和大小,服務器響應命令返回列出郵件列表和大小。如: 

  1. 客戶端發送POP3命令:LIST   
  2. 服務器響應命令: +OK 2 message(1430 octect)   
  3. 服務器響應命令:1    700   
  4. 服務器響應命令:2    730   
  5. 服務器響應命令:<一個空行> 

RETR 命令—— 該命令從服務器中獲得一個郵件,格式爲 RETR <郵件編號>如:

  1. 客戶端發送POP3命令:RETR 1   
  2. 服務器響應命令: 700 octets   
  3. 服務器響應命令:<郵件頭和內容>   
  4. 服務器響應命令: <空行> 

DELETE 命令——該命令告訴服務器將郵件標記爲刪除。(此時只是邏輯刪除) 

(3)更新狀態——客戶端發送QUIT命令後,此時就進入更新狀態,POP3服務器釋放在操作狀態中取得的資源,並將邏輯刪除的郵件進行物理刪除,然後關閉與客戶端的TCP連接。這樣整個郵件處理的過程就結束了。

3.2.3 接收功能的實現代碼

有了前面接收郵件過程的介紹,再參考代碼的實現,相信大家可以更好的理解客戶端從POP3服務器中獲取郵件的過程的,由於.net類庫並沒有幫我們封裝POP3協議的實現類,要實現郵件的獲取可以採用發送命令的方式,也可以使用Jmail組件,這個組件其實就是POP3協議的封裝類,既然微軟沒有幫我們做,其他公司幫我們做好後來幫助我們簡單的實現郵件的接收的一個類庫罷了。然後在使用這個組件的過程中出現了好幾個問題的,在源碼中我都解釋,大家可以下載源代碼後查看的。

實現郵件接收的核心代碼如下:

  1. // 登錄郵箱(這裏是本程序——郵件收發器)  
  2.        private void btnLogin_Click_1(object sender, EventArgs e)  
  3.        {  
  4.            // 與POP3服務器建立TCP連接  
  5.            // 建立連接後把服務器上的郵件下載到本地  
  6.            // 設置當前界面的光標爲等待光標(就是我們看到的一個動的圓形)  
  7.            Cursor.Current = Cursors.WaitCursor;  
  8.  
  9.            lsttbxStatus.Items.Clear();  
  10.            try 
  11.            {  
  12.                // POP3服務器通過監聽TCP110端口來提供POP3服務的  
  13.                // 向POP3服務器發出tcp請求  
  14.                tcpClient = new TcpClient(tbxPOP3Server.Text, 110);  
  15.                lsttbxStatus.Items.Add("正在連接...");  
  16.            }  
  17.            catch 
  18.            {  
  19.                MessageBox.Show("連接失敗""錯誤", MessageBoxButtons.RetryCancel, MessageBoxIcon.Error);  
  20.                lsttbxStatus.Items.Add("連接失敗!");  
  21.                return;  
  22.            }  
  23.  
  24.            // 連接成功的情況  
  25.            networkStream = tcpClient.GetStream();  
  26.            streamReader = new StreamReader(networkStream, Encoding.Default);  
  27.            streamWriter = new StreamWriter(networkStream, Encoding.Default);  
  28.            streamWriter.AutoFlush = true;  
  29.            string str;  
  30.            // 讀取服務器返回的響應連接信息  
  31.            str = GetResponse();  
  32.            if (CheckResponse(str) == false)  
  33.            {  
  34.                lsttbxStatus.Items.Add("服務器拒接了連接請求");  
  35.                return;  
  36.            }  
  37.            // 如果服務器接收請求  
  38.            // 向服務器發送憑證——用戶名和密碼  
  39.  
  40.            // 向服務器發送用戶名,請求確認  
  41.            lsttbxStatus.Items.Add("覈實用戶名階段...");  
  42.            SendToServer("USER " + tbxUserMail.Text);  
  43.            str = GetResponse();  
  44.            if (CheckResponse(str) == false)  
  45.            {  
  46.                lsttbxStatus.Items.Add("用戶名錯誤.");  
  47.                return;  
  48.            }  
  49.  
  50.            // 用戶名審覈通過後再發送密碼等待確認  
  51.            // 向服務器發送密碼,請求確認  
  52.            SendToServer("PASS "+txbPassword.Text);  
  53.            str = GetResponse();  
  54.            if (CheckResponse(str) == false)  
  55.            {  
  56.                lsttbxStatus.Items.Add("密碼錯誤!");  
  57.                return;  
  58.            }  
  59.  
  60.            lsttbxStatus.Items.Add("身份驗證成功,可以開始會話");  
  61.            // 向服務器發送LIST 命令,請求獲得郵件列表和大小  
  62.            lsttbxStatus.Items.Add("獲取郵件....");  
  63.            SendToServer("LIST");  
  64.            str = GetResponse();  
  65.            if (CheckResponse(str) == false)  
  66.            {  
  67.                lsttbxStatus.Items.Add("獲取郵件列表失敗");  
  68.                return;  
  69.            }  
  70.  
  71.            lsttbxStatus.Items.Add("郵件獲取成功");  
  72.  
  73.            // 窗口控件控制  
  74.            tabControlMyMailbox.Enabled = true;  
  75.            btnReadMail.Enabled = false;  
  76.            btnDownLoad.Enabled = false;  
  77.            btnDeleteMail.Enabled = false;  
  78.  
  79.            // 登陸成功後實例化郵件發送對象,以便後面完成發送郵件的操作  
  80.            // 實例化郵件發送類(SmtpClient)對象  
  81.            if (smtpClient == null)  
  82.            {  
  83.                smtpClient = new SmtpClient();  
  84.                smtpClient.Host = tbxSmtpServer.Text;  
  85.                smtpClient.Port = 25;  
  86.                      
  87.                // 不使用默認憑證,即需要認證登陸  
  88.                smtpClient.UseDefaultCredentials = false;  
  89.                smtpClient.Credentials = new NetworkCredential(tbxUserMail.Text, txbPassword.Text);  
  90.                smtpClient.DeliveryMethod = SmtpDeliveryMethod.Network;  
  91.            }  
  92.  
  93.            // 登陸成功後,自動接收新郵件  
  94.            // 開始接收郵件  
  95.            try 
  96.            {  
  97.                btnRefreshMailList.PerformClick();  
  98.            }  
  99.            catch 
  100.            {  
  101.                MessageBox.Show("讀取郵件列表失敗!""錯誤", MessageBoxButtons.RetryCancel, MessageBoxIcon.Error);  
  102.            }  
  103.  
  104.            lsttbxStatus.Items.Add("登陸成功!");  
  105.            lsttbxStatus.TopIndex = lsttbxStatus.Items.Count - 1;  
  106.            Cursor.Current = Cursors.Default;  
  107.  
  108.            // 窗口控件控制  
  109.            richtbxMailContentReview.Enabled = true;  
  110.            tbxUserMail.Enabled = false;  
  111.            txbPassword.Enabled = false;  
  112.            btnLogin.Enabled = false;  
  113.            btnLogout.Enabled = true;  
  114.            tbxSmtpServer.Enabled = false;  
  115.            tbxPOP3Server.Enabled = false;  
  116.            btnReadMail.Enabled = true;  
  117.            btnDownLoad.Enabled = true;  
  118.            btnDeleteMail.Enabled = true;  
  119.            tabControlMyMailbox.Focus();  
  120.        }  
  121.  
  122.        #region 處理與POP3服務器交互事件  
  123.        // 獲取服務器響應的信息  
  124.        private string GetResponse()  
  125.        {  
  126.            string str = null;  
  127.            try 
  128.            {  
  129.                str = streamReader.ReadLine();  
  130.                if (str == null)  
  131.                {  
  132.                    lsttbxStatus.Items.Add("連接失敗——服務器沒有響應");  
  133.                }  
  134.                else 
  135.                {  
  136.                    lsttbxStatus.Items.Add("收到:[" + str + "]");  
  137.                    if (str.StartsWith("-ERR"))  
  138.                    {  
  139.                        str = null;  
  140.                    }  
  141.                }  
  142.            }  
  143.            catch(Exception err)  
  144.            {  
  145.                lsttbxStatus.Items.Add("連接失敗:[" + err.Message + "]");  
  146.            }  
  147.  
  148.            return str;  
  149.        }  
  150.  
  151.        // 檢查響應信息  
  152.        private bool CheckResponse(string responseString)  
  153.        {  
  154.            if (responseString == null)  
  155.            {  
  156.                return false;  
  157.            }  
  158.            else 
  159.            {  
  160.                if (responseString.StartsWith("+OK"))  
  161.                {  
  162.                    return true;  
  163.                }  
  164.                else 
  165.                {  
  166.                    return false;  
  167.                }  
  168.            }  
  169.        }  
  170.  
  171.        // 向服務器發送命令  
  172.        private bool SendToServer(string str)  
  173.        {  
  174.            try 
  175.            {  
  176.                // 這裏必須使用WriteLine方法的,因爲POP3協議中定義的命令是以回車換行結束的  
  177.                // 如果客戶端發送的命令沒有以回車換行結束,POP3服務器就不能識別,也就不能響應客戶端的請求了  
  178.                // 如果想用Write方法,則str輸入的參數字符中必須包含“\r\n”,也就是回車換行字符串。  
  179.                streamWriter.WriteLine(str);  
  180.                streamWriter.Flush();  
  181.                lsttbxStatus.Items.Add("發送:[" + str + "]");  
  182.                return true;  
  183.            }  
  184.            catch(Exception ex)  
  185.            {  
  186.                lsttbxStatus.Items.Add("發送失敗:[" + ex.Message + "]");  
  187.                return false;  
  188.            }  
  189.        }  
  190.        #endregion 

3.3 程序運行結果演示

首先輸入郵箱名和密碼登錄到POP3服務器來獲取郵件列表的演示:

然後在郵件列表中選中一個郵件進行閱讀,然後進行回覆郵件的操作演示(郵件的發送都可以附加附件發送出去):

閱讀郵件的界面:

回覆郵件的界面:

同時點擊發送按鈕後,就可以把郵件發送到sina的SMTP服務器上,再由新浪的SMTP服務器轉發到QQ的SMTP服務器,QQ的POP3服務器中QQ的SMTP服務器獲取收到的郵件,當QQ用戶輸入正確的郵箱名和密碼後就可以從QQ的POP3服務器上獲取收到的郵件。

點擊發送按鈕後成功發送郵件的圖片:

四、總結

介紹到這裏,本專題的內容就已經介紹完了,希望通過本專題可以讓大家明白郵件發送和接收的原理,並且可以自定義一個簡單郵件收發器的功能的,在後面一專題將介紹FTP協議(文件傳輸協議),並實現一個簡單的文件上傳和下載的程序。

源代碼下載地址: http://files.cnblogs.com/zhili/MailSendAndReceive.zip 

原文鏈接:http://www.cnblogs.com/zhili/archive/2012/09/24/MailSend_POP3_SMTP.html


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章