关于 OpenSmtp 邮件标题过长后出现乱码问题的解决
OpenSmtp 是 .Net 平台上一个不错的邮件发送组件,但是其中存在一些 bug 影响我们的使用,我在使用中遇到了邮件主题长度较大时,邮件出现乱码的情况。
// sb.Append("Subject: " + MailEncoder.ConvertHeaderToQP(cleanSubject.ToString(), charset) + "\r\n");
sb.Append("Subject: " + MailEncoder.ConvertHeaderToBase64( cleanSubject.ToString(), charset) + "\r\n");
经过检查源代码发现,原来的处理过程存在问题:首先主题是通过 ASCII 编码的,其次,使用了 QP 编码,但没有考虑主题与内容的区别。
在主题中编码时,每一行必须都要单独编码,不可将主题全部编码。
经过与 Outlook Express 对比,在 MailEncoding 类中增加一个方法,专门针对邮件主题进行 Base64 编码
public static string ConvertHeaderToBase64(string s, string charset)
{
int lineLength = 40; // 每行处理 40 个字节
Encoding encoding = Encoding.GetEncoding( charset ); // 取指定编码
byte[] buffer = encoding.GetBytes( s ); // 转换为字节码
{
int lineLength = 40; // 每行处理 40 个字节
Encoding encoding = Encoding.GetEncoding( charset ); // 取指定编码
byte[] buffer = encoding.GetBytes( s ); // 转换为字节码
StringBuilder sb = new StringBuilder(); // 保存最终结果
string linebase64 ;
string linebase64 ;
int block = buffer.Length%lineLength==0?buffer.Length/lineLength:buffer.Length/lineLength + 1;
for(int i=0; i< block; i++)
{
if( buffer.Length - i*lineLength >=lineLength )
linebase64 = Convert.ToBase64String( buffer, i*lineLength, lineLength );
else
linebase64 = Convert.ToBase64String( buffer, i*lineLength, buffer.Length - i*lineLength);
sb.Append( "=?" );
sb.Append( charset );
sb.Append( "?B?" );
sb.Append( linebase64 );
sb.Append( "?=\r\n\t" );
}
sb.Remove( sb.Length-3, 3); // 删除最后的换行符号
return sb.ToString();
}
{
if( buffer.Length - i*lineLength >=lineLength )
linebase64 = Convert.ToBase64String( buffer, i*lineLength, lineLength );
else
linebase64 = Convert.ToBase64String( buffer, i*lineLength, buffer.Length - i*lineLength);
sb.Append( "=?" );
sb.Append( charset );
sb.Append( "?B?" );
sb.Append( linebase64 );
sb.Append( "?=\r\n\t" );
}
sb.Remove( sb.Length-3, 3); // 删除最后的换行符号
return sb.ToString();
}
然后,将 MailMessage 类中的 ToString 方法中对邮件主题的处理修改为调用自定义的方法
// sb.Append("Subject: " + MailEncoder.ConvertHeaderToQP(cleanSubject.ToString(), charset) + "\r\n");
sb.Append("Subject: " + MailEncoder.ConvertHeaderToBase64( cleanSubject.ToString(), charset) + "\r\n");
重新编译即可