想知道是否有更好的方法来处理多个类似的条件语句和操作,例如下面的示例代码段。
private void AddCommonDictionaryItemsForAllAttributes(MyCustomType dc, string statusFlag)
{
if (dc.xmlAttributes == null) {
dc.xmlAttributes = new Dictionary<string, string>();
}
dc.xmlAttributes.Add(Constant.CD_1, statusFlag);
dc.xmlAttributes.Add(Constant.CD_2, statusFlag);
dc.xmlAttributes.Add(Constant.CD_3, statusFlag);
if (dc.primaryZone != null) {
dc.xmlAttributes.Add(Constant.CD_4, statusFlag);
}
if (dc.Mgr1 != null) {
dc.xmlAttributes.Add(Constant.CD_10, statusFlag);
}
if (dc.Mgr2 != null) {
dc.xmlAttributes.Add(Constant.CD_11, statusFlag);
}
if (dc.Mgr3 != null) {
dc.xmlAttributes.Add(Constant.CD_5, statusFlag);
}
if (dc.Producer != null) {
dc.xmlAttributes.Add(Constant.CD_6, statusFlag);
}
if (dc.CountTest > 0) {
dc.xmlAttributes.Add(Constant.CD_7, statusFlag);
}
if (dc.List1 != null && dc.List1.Count > 0) {
dc.xmlAttributes.Add(Constant.CD_8, statusFlag);
}
if (dc.List2 != null && dc.List2.Count > 0) {
dc.xmlAttributes.Add(Constant.CD_9, statusFlag);
}
}
if条件和字典操作的添加在我看来是多余的,所以寻找更有效和优雅的方法来编写代码。
谢谢!
更新:我使用的是.NET 3.5
答案 0 :(得分:4)
你可以创建一个帮助器类型,它提供了对MyCustomType
实例执行的测试,以及在xmlAttributes
字典中使用的键:
class Rule
{
private readonly Predicate<MyCustomType> _test;
private readonly string _key;
public Predicate<MyCustomType> Test { get { return _test; } }
public string Key { get { return _key; } }
public Rule(Predicate<MyCustomType> test, string key)
{
_test = test;
_key = key;
}
}
然后,您可以创建一组这些规则,并枚举它们:
private void AddCommonDictionaryItemsForAllAttributes(MyCustomType dc, string statusFlag)
{
var rules = new Rule[]
{
new Rule(x => x.Mgr1 != null, Constant.CD_4),
new Rule(x => x.Mgr2 != null, Constant.CD_10),
//...snip...
new Rule(x => x.List2 != null && x.List2.Count > 0, Constant.CD_9)
};
foreach(var rule in rules.Where(r => r.Test(dc)))
dc.xmlAttributes.Add(rule.Key, statusFlag);
}
答案 1 :(得分:2)
一种选择是拥有某种条件列表和这些条件所代表的常量。例如:
var list = new List<Tuple<Predicate<MyCustomType>, string>>
{
Tuple.Create(dc => true, Constant.CD_1),
Tuple.Create(dc => true, Constant.CD_2),
Tuple.Create(dc => true, Constant.CD_3),
Tuple.Create(dc => dc.primaryZone != null, Constant.CD_4),
Tuple.Create(dc => dc.Mgr1 != null, Constant.CD_5),
// etc
};
然后你可以遍历列表,只要谓词为真,就在字典中设置相关的status
:
foreach (var tuple in list)
{
if (tuple.Item1(dc))
{
dc.xmlAttributes.Add(tuple.Item2, statusFlag);
}
}
请注意,您可以静态设置列表一次,然后在任何地方重复使用,因为列表本身不会更改。
答案 2 :(得分:1)
你可以在for循环中执行,因为你可以cast into to enum。
答案 3 :(得分:0)
考虑在YourCustomClass中封装属性集合。这将保护您的属性免于意外更改,并且它将属性填充逻辑朝向它所属的数据。
好处:
因此,即使你的默认实现用法如下:
dc.SetStaus(string statusFlag)
所有脏工作都将在DC内完成(BTW我建议使用枚举CD而不是常量,但这取决于你):
public void SetStatus(string statusFlag)
{
if (_xmlAttributes == null)
_xmlAttributes = new Dictionary<CD, string>();
_xmlAttributes.Add(CD.CD_1, statusFlag);
_xmlAttributes.Add(CD.CD_2, statusFlag);
_xmlAttributes.Add(CD.CD_3, statusFlag);
if (_primaryZone != null)
_xmlAttributes.Add(CD.CD_4, statusFlag);
if (_mgr1 != null)
_xmlAttributes.Add(CD.CD_10, statusFlag);
if (_mgr2 != null)
_xmlAttributes.Add(CD.CD_11, statusFlag);
if (_mgr3 != null)
_xmlAttributes.Add(CD.CD_5, statusFlag);
if (_producer != null)
_xmlAttributes.Add(CD.CD_6, statusFlag);
if (_countTest > 0)
_xmlAttributes.Add(CD.CD_7, statusFlag);
if (_list1 != null && _list1.Count > 0)
_xmlAttributes.Add(CD.CD_8, statusFlag);
if (_list2 != null && _list2.Count > 0)
_xmlAttributes.Add(CD.CD_9, statusFlag);
}
之后,您可以轻松地重构实现:
private Dictionary<CD, Func<bool>> _statusSetConditions;
public MyCustomType()
{
_statusSetConditions = new Dictionary<CD, Func<bool>>();
_statusSetConditions.Add(CD.CD_1, () => true);
_statusSetConditions.Add(CD.CD_2, () => true);
_statusSetConditions.Add(CD.CD_3, () => true);
_statusSetConditions.Add(CD.CD_4, () => _primaryZone != null);
...
_statusSetConditions.Add(CD.CD_11, () => _mgr2 != null);
}
public void SetStatus(string statusFlag)
{
if (_xmlAttributes == null)
_xmlAttributes = new Dictionary<CD, string>();
foreach (CD cd in Enum.GetValues(typeof(CD)))
AddStatusAttribute(cd, statusFlag);
}
private void AddStatusAttribute(CD cd, string statusFlag)
{
Func<bool> condition;
if (!_statusSetConditions.TryGetValue(cd, out condition))
return; // or throw exception
if (condition())
_xmlAttributes.Add(cd, statusFlag);
}
客户端仍然只是调用dc.SetStatus(statusFlag);
也许在封装了这个状态集逻辑之后,你只需将状态保存在YourCustomClass的字段中。
答案 4 :(得分:0)
咩。它并不是多余的,除非你想考虑像java反射这样的东西。考虑辅助方法:
void addIfOk(int test, MyCustomType dc, String attr, string statusFlag) { if(test!=0) dc.xmlAttributes.Add(attr, statusFlag); } void addIfOk(Object test, MyCustomType dc, String attr, string statusFlag) { if(test!=null) dc.xmlAttributes.Add(attr, statusFlag); } void addIfOk(Collection test, MyCustomType dc, String attr, string statusFlag) { if(test!=null&&!test.isEmpty()) dc.xmlAttributes.Add(attr, statusFlag); }
然后代码变为:
addIfOk(dc.Mgr1, dc, Constant.CD_10, statusFlag); addIfOk(dc.Mgr2, dc, Constant.CD_11, statusFlag); addIfOk(dc.Mgr3, dc, Constant.CD_5, statusFlag); addIfOk(dc.Producer, dc, Constant.CD_5, statusFlag);
等等。也许这会更有意义作为自定义类型中的方法:setXmlStatusAttributes(statusfFlag)
答案 5 :(得分:-1)
有两种方法: 1.使用开关盒 2.使用三元运算符
两者都会让你的代码看起来干净,但在你的情况下,开关盒不起作用。