SMTP 用法(100分)

  • 主题发起人 主题发起人 madaha
  • 开始时间 开始时间
M

madaha

Unregistered / Unconfirmed
GUEST, unregistred user!
有谁知道,如何使用DELPHI6中DEMOS/FASTNET/SMTP/SMTPDEMO.EXE的用法
即:如何使用此程序中的SMTP功能发邮件。
比如我要从邮箱:a2332@vip.sina.com 向邮箱b23322@163.com
发一邮件。如何设定其中的参数?
Host : ?
Port : ?
user ID : ?

Name : ?
E-Mail Adress : ?

以及:UserName to Verify : ?
 
FASTNET不支行SMTP认证吧,你再怎样设也是不行的。
换成INDY或别的支行SMTP认证的控件吧
 
SMTP认证?能否解释一下?
 
你看一下Indy的MailClient吧,使用IDSMTP控件
那个控件发163发不过去,你在这儿查以下,可以解决的
 
用FASTNET的SMTP完全可以发邮件,且支持认证, 当然要自编认证功能。

以下是认证过程:
procedure TSmail.NMSMTP1Connect(Sender: TObject);
begin
if nmsmtp1.ReplyNumber =250 then nmsmtp1.Transaction('auth login');
if nmsmtp1.ReplyNumber =334 then nmsmtp1.Transaction(EncodeBase64('这里是userid'));
if nmsmtp1.ReplyNumber =334 then nmsmtp1.Transaction(EncodeBase64('这里是邮箱认证密码'));
if nmsmtp1.ReplyNumber =235 then AuthSucc:=true;
end;


以下是发送过程:
if NMSMTP1.Connected then NMSMTP1.Disconnect;
Nmsmtp1.Host:='sina 的 host 地址 ';
nmsmtp1.Port:=25;
nmsmtp1.UserID:=a2332;
nmsmtp1.ReportLevel:=1;
Nmsmtp1.TimeOut:=10000;
nmsmtp1.Connect;
if AuthSucc=true then
begin
nmsmtp1.PostMessage.FromAddress :=a2332@vip.sina.com ;
nmsmtp1.PostMessage.ToAddress.Text :=b23322@163.com ;
nmsmtp1.PostMessage.Body.Text :=form1.memo1.Text;
nmsmtp1.PostMessage.Subject :='这里是主题';
try
nmsmtp1.SendMail;
finally
nmsmtp1.Disconnect;
end;

注: 以上并不完整,还有EncodeBase64函数太长, 若要可与我联系。
 
redboy :
你能否将源码发给我?
我的E-Mail是:madahaa@21cn.com ,最好将 .dcu .dfm .pas .dpr .res文件都发过来
 
沒有必要給自己找麻煩.
Indy多好,什麼都有了!用Indy吧?
 
Richard3000: Indy好象也没有SMTP认证的,我试了一下好象不行。
 
var
SMTP:TidSMTP;

SMTP.AuthenticationType := atLogin; {Simple Login}
SMTP.UserID := UserID;
SMTP.Password := Pasword;
SMTP.Host := Host;
SMTP.Port := 25;
 
用这个PAS, 然后在主程序里调LOGIN()

unit UnitSmtpLogin;

interface

uses NMSMTP,classes,Sysutils;

type
EIdException = class(Exception);
TIdCardinalBytes = record
case Integer of
0: (
Byte1: Byte;
Byte2: Byte;
Byte3: Byte;
Byte4: Byte;);
1: (Whole: Cardinal);
end;

function login(smtp:TNMSMTP;user,password:string):boolean;


var
//MIME:
CodingTable: string = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
//UUE: CodingTable: string = '`!"#$%&''()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[/]^_';
//XXE: CodingTable: string = '+-0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';

//MIME:
FillChar: char = '=';
//UUE: CodingTable[1] or '~'
//XXE: CodingTable[1] or '~'

implementation

procedure EncodeUnit(const AIn1, AIn2, AIn3: Byte; var VOut: Cardinal);
var
LUnit: TIdCardinalBytes;
begin
LUnit.Byte1 := Ord(CodingTable[((AIn1 SHR 2) and 63) + 1]);
LUnit.Byte2 := Ord(CodingTable[(((AIn1 SHL 4) or (AIn2 SHR 4)) and 63) + 1]);
LUnit.Byte3 := Ord(CodingTable[(((AIn2 SHL 2) or (AIn3 SHR 6)) and 63) + 1]);
LUnit.Byte4 := Ord(CodingTable[(Ord(AIn3) and 63) + 1]);
VOut := LUnit.Whole;
end;

function Encode2(ASrcStream: TStream; const ABytes: integer = MaxInt): string;
var
LIn1, LIn2, LIn3: Byte;
LSize: integer;
LStartPos: integer;
LUnit: TIdCardinalBytes;
begin
//TODO: Make this more efficient. Profile it to test, but maybe make single calls to ReadBuffer
//then pull from memory
if (ABytes <> MaxInt) and ((ABytes mod 3) > 0) then begin
raise EIdException.Create('Uneven size in Encode.');
end;
Result := '';
LStartPos := ASrcStream.Position;
while (ASrcStream.Position < ASrcStream.Size) and (ASrcStream.Position - LStartPos < ABytes)
do begin
ASrcStream.ReadBuffer(LIn1, 1);
if ASrcStream.Position < ASrcStream.Size then begin
ASrcStream.ReadBuffer(LIn2, 1);
if ASrcStream.Position < ASrcStream.Size then begin
ASrcStream.ReadBuffer(LIn3, 1);
LSize := 3;
end else begin
LIn3 := 0;
LSize := 2;
end;
end else begin
LIn2 := 0;
LSize := 1;
end;
EncodeUnit(LIn1, LIn2, LIn3, LUnit.Whole);
Result := Result + Chr(LUnit.Byte1) + Chr(LUnit.Byte2) + Chr(LUnit.Byte3) + Chr(LUnit.Byte4);
if LSize < 3 then begin
Result[Length(Result)] := FillChar;
if LSize = 1 then begin
Result[Length(Result) - 1] := FillChar;
end;
end;
end;
end;

function Encode(const ASrc: string): string;
var
LSrcStream: TStringStream;
begin
LSrcStream := TStringStream.Create(ASrc); try
Result := Encode2(LSrcStream);
finally FreeAndNil(LSrcStream); end;
end;

function login(smtp:TNMSMTP;user,password:string):boolean;
const crlf = #13#10;
var
s:string;
begin
result:=false;
try
s:='auth LOGIN'+crlf;
smtp.SendBuffer(pchar(s),length(s));
s:=smtp.ReadLn;
if copy(s,1,3)<>'334' then exit; //Fail

s:=Encode(user)+crlf;
smtp.SendBuffer(pchar(s),length(s));
s:=copy(smtp.ReadLn,1,3);
if (s<>'334')and(s<>'235') then exit; //Fail

s:=Encode(password)+crlf;
smtp.SendBuffer(pchar(s),length(s));
s:=copy(smtp.ReadLn,1,3);
if (s<>'235') then exit; //Fail
result:=true;
except
end;
end;

end.
 
indy的最新版本有SMTP的认证功能,或者使用ICS,非常的好用。
 
你修改以下Indy控件把
打开IdMessage.pas单元
搜索Values['X-Library'],把这行注释掉,
保存,重新编译indy包,
在把你的程序编译以便,搞定,呵呵


不要谢我、谢谢twos
 
同意:
你看一下Indy的MailClient吧,使用IDSMTP控件
那个控件发163发不过去,你在这儿查以下,可以解决的
 
twos: “邮件特快专递”如何实现加带附件?

type
TForm1 = class(TForm)
IdDNSResolver: TIdDNSResolver;
IdAntiFreeze1: TIdAntiFreeze;
btnSend: TButton;
IdSMTP: TIdSMTP;
IdMsgSend: TIdMessage;
mmContent: TMemo;
Label1: TLabel;
edtTo: TEdit;
Label4: TLabel;
Label5: TLabel;
edtFrom: TEdit;
Label6: TLabel;
edtSubject: TEdit;
procedure btnSendClick(Sender: TObject);
private
{ Private declarations }
procedure GetMxList(AMxList: TStringList; AQName: string);
public
{ Public declarations }
end;

var
Form1: TForm1;

implementation

{$R *.dfm}

{ TForm1 }

{ 这个过程是用来得到邮件特快专递目的地服务器名称及优先级别数,参数AMXList是
用来接收结果值,AQName代表传递过来的域名 }
procedure TForm1.GetMxList(AMxList: TStringList; AQName: string);
var
i: Integer;
begin
with IdDNSResolver do
begin
Host := '202.101.107.55'; { Host属性用来指定域名服务器的地址,此处为笔者所在地
的主域名服务器地址,你也可以指定任一可以快速访问到的Internet上域名服务器
地址,要知道自己所在地的域名服务器地址,win98下通过winipcfg命令,win2000下
通过ipconfig /all即可查出。}
ReceiveTimeout := 10000; // 在指定的时间内得不到域名服务器的反馈,则视为失败。
ClearVars; // 清除前一次查询所反馈回来的资源记录

{ 构建此次查询的头部结构 }
with DNSHeader do
begin
Qr := False; // False 代表查询
Opcode := 0; // 0代表标准域名查询
RD := True; //域名服务器可以进行递归查询
QDCount := 1; //查询的数量
end;

{ 构建要查询的问题 }
DNSQDList.Clear;
with DNSQDList.Add do
begin
QName := AQName; //要查询的域名
QType := cMX; //QTYPE指定要查询的资源记录的种类,值为cMX代表邮件交换记录
QClass := cIN;
end;

ResolveDNS; //向域名服务器发出请求

{ 从域名服务器接收反馈的结果,将反馈回来的邮件服务器名称放在AMXList列表的Name
部分,
邮件服务器的优先级别数放在Value部分。 }
for i := 0 to DNSAnList.Count - 1 do
AMxList.Add(DNSAnList.RData.MX.Exchange + '=' +
IntToStr(DNSAnList.RData.MX.Preference));
end;
end;

{ 单击"发送"按钮时发送专递邮件 }
procedure TForm1.btnSendClick(Sender: TObject);
var
MxList: TStringList;
i: Integer;
QName, ThoughAddress: string;
begin
{ 根据用户所填写的内容创建邮件 }
with IdMsgSend do
begin
Body.Assign(mmContent.Lines); //邮件正文
From.Address := Trim(edtFrom.Text); //发件人地址
Recipients.EMailAddresses := Trim(edtTo.Text); //收件人地址
Subject := edtSubject.Text; //邮件主题
end;

{ 从输入的收件人地址中取出邮箱域名,利用前面的GetMxList过程得到目的地地址 }
QName := TrimRight(copy(edtTo.Text, Pos('@', edtTo.Text) + 1, Length(edtTo.Text)));
MxList := TStringList.Create;
try
GetMxList(MxList, QName);
ThoughAddress := MxList.Names[0]; {取反馈回来的第一个服务器为目的地,读者可
根据实际需要改进,比如说考虑到信件的优先级或当你选择的服务器因繁忙而暂时
不能处理你的信件时,换用其它服务器试试 }
finally
MxList.Free;
end;

{ 发送邮件 }
with IdSMTP do
begin
Host := ThoughAddress; // 将Host赋值为目的地,这就是特快专递与普通邮件的区别
Port := 25; // smtp服务默认的端口为25
Connect; //连接到服务器
try
Send(IdMsgSend); //发送刚才创建的邮件
ShowMessage('发送完毕'); //发送完毕后提示
finally
Disconnect; //断开服务器连接
end;
end;
end;

end.

//本程序在Windows 2000 + Delphi 6 下调试通过,点击这里下载。
//http://www.ccw.com.cn/htm/app/down/1204Source.zip
 
手把手教你学会用delphi制作发送邮件程序,并送源码
http://www.delphibbs.com/delphibbs/dispq.asp?lid=1702834
 
多人接受答案了。
 
后退
顶部