我在SO周围挖了一个答案,到目前为止我能找到的最好的是here,但这是针对静态构造函数的实例;我只是静态使用这个类。
我的代码:
public static class MailHelper {
private static string mailHost;
static MailHelper() {
var mailSettings = ConfigurationManager.GetSection("MailSettings") as NameValueCollection;
if (null == mailSettings) {
throw new ConfigurationErrorsException("Missing Mail Settings in the configuration file");
}
mailHost = ConfigurationManager.AppSettings["mailHost"];
if (null == mailHost) {
throw new ConfigurationErrorsException("Missing mailHost setting in the configuration file");
}
}
public static void SendMail(MailMessage Message) {
...
}
}
try {
MailHelper.SendMail(Message);
}
catch (ConfigurationErrorsException exc) {
...
}
// ???
MailHelper.SendMail(Message);
.
因此,如果静态构造函数在第一次调用时抛出异常,那么第二次尝试访问静态SendMail()方法时会发生什么?
PS:很抱歉,如果你不喜欢Stroustrup版本的K& R支撑造型,但是不要编辑我的帖子只是为了将支架改为你喜欢的Allman风格。感谢。
答案 0 :(得分:87)
类型初始值设定项失败一次后,永远不会重试。该类型在AppDomain的生命周期内已经死亡。 (请注意,对于所有类型初始值设定项都是如此,不仅适用于具有静态构造函数的类型。具有初始化表达式但没有静态构造函数的静态变量的类型可能会在类型初始化程序执行的时间方面表现出细微的差异 - 但它会仍然只发生一次。)
演示:
using System;
public sealed class Bang
{
static Bang()
{
Console.WriteLine("In static constructor");
throw new Exception("Bang!");
}
public static void Foo() {}
}
class Test
{
static void Main()
{
for (int i = 0; i < 5; i++)
{
try
{
Bang.Foo();
}
catch (Exception e)
{
Console.WriteLine(e.GetType().Name);
}
}
}
}
输出:
In static constructor
TypeInitializationException
TypeInitializationException
TypeInitializationException
TypeInitializationException
TypeInitializationException
如您所见,静态构造函数只被调用一次。
答案 1 :(得分:24)
另外两个答案是对你的直接问题的好答案 - 这是一个metaanswer - 当你检测到配置元素没有被填充时,你应该在方法中抛出异常,而不是在构造函数中。恕我直言,“未配置”是构造函数阶段中这些元素的有效配置状态,而不是在SendMail时间。这将回避整个问题。
答案 2 :(得分:16)