C# Dotnet core SmtpClient-在SendCompletedEventHandler上的数据库中插入数据
我有一个.NET Core 3.1应用程序。应用程序应发送电子邮件,并在电子邮件成功(或未成功)发送到SQL Server数据库时记录 我已在C# Dotnet core SmtpClient-在SendCompletedEventHandler上的数据库中插入数据,c#,dependency-injection,callback,smtp,dbcontext,C#,Dependency Injection,Callback,Smtp,Dbcontext,我有一个.NET Core 3.1应用程序。应用程序应发送电子邮件,并在电子邮件成功(或未成功)发送到SQL Server数据库时记录 我已在Startup.cs中注册了数据库: services.AddDbContext<DataBaseAppContext>(options => options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection"))); 每当我调用EmailSender.S
Startup.cs
中注册了数据库:
services.AddDbContext<DataBaseAppContext>(options => options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection")));
每当我调用EmailSender.SendEmailAsync
时,我都会遇到以下错误:
无法访问已释放的对象。此错误的一个常见原因是处理通过依赖项注入解析的上下文,然后在应用程序的其他位置尝试使用相同的上下文实例。如果对上下文调用Dispose(),或将上下文包装到using语句中,则可能会发生这种情况。如果您使用的是依赖项注入,那么应该让依赖项注入容器处理上下文实例
我理解这是因为服务注册为AddTransient
,并且当回调发生时,\u上下文
已经被释放
不确定解决此问题的最佳方法是什么,尝试将
AddTransient
更改为AddScoped
或AddSingleton
,只是为了看看会发生什么,但随后在程序中出现错误。cs
是否可以使用SendMailAsync
方法可以等待此方法,容器不应处理您的上下文
我想如果你坚持使用回调方法,你可以注入工厂,在你需要的时候为你创建上下文(在回调中),你可以在回调方法中自己处理上下文 您可以使用
SendMailAsync
方法吗?-可以等待此方法,容器不应处理您的上下文
我想如果你坚持使用回调方法,你可以注入工厂,在你需要的时候为你创建上下文(在回调中),你可以在回调方法中自己处理上下文
services.AddTransient<IEmailSender, EmailSender>();
public class EmailSender : IEmailSender
{
private readonly DataBaseAppContext _context;
public EmailSender(DataBaseAppContext context)
{
_context = context;
}
public Task SendEmailAsync(string subject, string message, string email)
{
return Execute(subject, message, email);
}
public Task Execute(string subject, string message, string email)
{
try
{
MailAddress fromAddress = new MailAddress(...);
SmtpClient smtp = new SmtpClient
{
Host = ...,
Port = ...,
...
};
MailMessage md = new MailMessage
{
From = ...,
IsBodyHtml = true,
Subject = subject,
Body = message,
To = ...
};
smtp.SendCompleted += new SendCompletedEventHandler(SendCompletedCallback);
void SendCompletedCallback(object sender, AsyncCompletedEventArgs e)
{
if (e.Cancelled)
{
//Want to use this statement as well, to log errors
}
if (e.Error != null)
{
//Want to use this statement as well, to log errors
}
else
{
try
{
//Object that should be saved in DB
EmailLogger emailLogger = new EmailLogger()
{
Email = email,
Subject = subject,
Message = message,
DateTime = DateTime.Now
};
_context.EmailLogger.Add(emailLogger);
_context.SaveChanges();
}
catch(Exception ex)
{
//Here I got an error - the *error below
}
}
}
return smtp.SendMailAsync(md);
}
catch (Exception e)
{
throw e;
}
}
}