5

[编辑:我意识到失败的参数实际上是双精度数而不是整数。根据日志,没有一个整数计时器失败。大多数定时器和参数都是整数,但不是全部。双打不是原子的,毕竟缺乏锁定可能是问题所在。]

我有一个应用程序,它使用一个包含可配置值属性的类。应用程序中使用的大多数属性都是派生的。这些值在启动时设置,并且在应用程序的主要部分运行时不会更改。

private int _TimerInterval;
public int TimerInterval { get { return _TimerInterval; } }

private int _Factor1;
public int Factor1 { 
  set { 
    _Factor1 = value;
    _TimerInterval = _Factor1 * _Factor2;
  }
  get { return _Factor1; }
}

private int _Factor2;
public int Factor2 { 
  set { 
    _Factor2 = value;
    _TimerInterval = _Factor1 * _Factor2;
  }
  get { return _Factor2; }
}

我发现由于异常,返回的值很少明显为零。

Exception Message: '0' is not a valid value for 'Interval'. 'Interval' must be   greater than 0.
Exception Target Site: set_Interval

调用代码看起来像这样:

exitTimer.Interval = _config.TimerInterval;

该程序是多线程的,但对单个属性的调用仅在一个线程中使用。类的其他属性在其他线程中调用。我确实在具有类似属性的其他计时器上看到了这个问题。

如果我捕获异常并重试它的工作分配。

我的计时器是否发生了一些事情会导致除了属性返回零之外的执行?

更新 #1 - 请求更多代码

每个字段都被定义为一个cfXXX(配置字段)常量。这确保我们不会拼错字段名称。每个属性的相应默认值定义为DefXXXPareseXXX函数(在ParseInt此示例中)从配置查找中接受字符串值,并将其转换为相应的值类型或提供的默认值(如果失败)。失败可能是由于缺少 XML 记录(新配置选项)或编辑不正确造成的。

加载初始配置数据的代码:

// Main Form
public fMain()
{
  InitializeComponent();
  config = new ConfigData();
  config.LoadConfig();
  // Other initializations
}

//ConfigData Class

// XML config field names
private const string cfFactor1 = "Factor1";
private const string cfFactor1 = "Factor2";
private const string cfFactor3 = "Factor3";
private const string cfFactor4 = "Factor4";

//Default values
private const int DefFactor1 = 1;
private const int DefFactor2 = 50;
private const int DefFactor3 = 1;
private const int DefFactor4 = 25;

public void LoadConfig()
{
  Factor1 = ParseInt(ConfigurationManager.AppSettings[cfFactor1], DefFactor1);
  Factor2 = ParseInt(ConfigurationManager.AppSettings[cfFactor2], DefFactor2);
  Factor3 = ParseInt(ConfigurationManager.AppSettings[cfFactor3], DefFactor3);
  Factor4 = ParseInt(ConfigurationManager.AppSettings[cfFactor4], DefFactor4);
}

int ParseInt(string numberString, int aDefault = 0)
{
  int result;
  if (!int.TryParse(numberString, out result)) {
    result = aDefault;
  }
  return result;
}
4

3 回答 3

2

问题是前提是错误的。失败的属性实际上是浮点数而不是整数。尴尬。正如预期的那样,为这些项目的属性添加锁可以解决问题。整数计时器从未失败。

我会在这一点上删除这个问题,但它已经有了答案。所以让它成为一个关于双重检查假设的教训。

于 2013-12-17T13:49:47.267 回答
1

如果您想在使用实例之前为每个实例分配一次这些因素......在构造函数中执行此操作。

如果您想在创建任何实例之前为该类型分配一次这些因素......在静态构造函数中执行此操作。

不要在线程之间共享可变实例。

于 2013-10-23T14:36:22.120 回答
0

当仅设置 Factor1 或 Factor2 时,_TimerInterval 为 0。

x * 0 = 0

于 2013-09-30T13:56:08.550 回答