我需要编写一个解析器。我想测试很多输入格式,输出非常复杂。我应该如何使用类似的测试来测试不同的输入和输出?
例如
public class Person
{
public string Name;
public int Age;
public string Comment;
}
public interface IParser
{
Person Parse(string input);
}
public class Parser : IParser
{
public Person Parse( string input )
{
var fields = input.Split(',');
var p = new Person
{
Name = fields[0],
Age = int.Parse(fields[1])
};
p.Comment = fields.Length == 3 ? fields[2] : "";
return p;
}
}
我写了这些测试......
public class ParserTests
{
protected string _input;
protected Person _expected;
private IParser _parser = new Parser();
private Person _actual { get { return _parser.Parse(_input); } }
[TestMethod]
public void Parse_Name()
{
Assert.AreEqual(_expected.Name, _actual.Name);
}
[TestMethod]
public void Parse_Age()
{
Assert.AreEqual(_expected.Age, _actual.Age);
}
[TestMethod]
public void Parse_Comment()
{
Assert.AreEqual(_expected.Comment, _actual.Comment);
}
}
[TestClass]
public class ParserTestsWithoutComment : ParserTests
{
public ParserTestsWithoutComment()
{
_input = "John,29";
_expected = new Person { Name = "John", Age = 29, Comment = "" };
}
}
[TestClass]
public class ParserTestsWithComment : ParserTests
{
public ParserTestsWithComment()
{
_input = "Brian,99,test";
_expected = new Person { Name = "Brian", Age = 99, Comment = "test" };
}
}
我是单元测试的新手,我不知道如何从更复杂的东西开始。我真正的输入文件更复杂,就像
PokerStars Hand #98451585362: Hold'em No Limit ($5/$10 USD) - 2013/05/12 9:25:04 CET [2013/05/12 3:25:04 ET]
Table 'Soyuz-Apollo II' 6-max Seat #4 is the button
Seat 1: Codrus426 ($1812.52 in chips)
Seat 2: JMBigJoe ($2299.10 in chips)
Seat 3: xinxin1 ($903.94 in chips)
Seat 4: moshmachine ($1107 in chips)
Seat 5: TopKat5757 ($1147 in chips)
Seat 6: LukaschenkoA ($1274.96 in chips)
TopKat5757: posts small blind $5
LukaschenkoA: posts big blind $10
*** HOLE CARDS ***
Codrus426: calls $10
JMBigJoe: raises $25 to $35
xinxin1: folds
moshmachine: folds
TopKat5757: folds
LukaschenkoA: folds
Codrus426: calls $25
*** FLOP *** [2h 3s 6h]
Codrus426: checks
JMBigJoe: bets $41
Codrus426: calls $41
*** TURN *** [2h 3s 6h] [2d]
Codrus426: bets $40
JMBigJoe: calls $40
*** RIVER *** [2h 3s 6h 2d] [Qh]
Codrus426: checks
JMBigJoe: checks
*** SHOW DOWN ***
Codrus426: shows [9d Ah] (a pair of Deuces)
JMBigJoe: mucks hand
Codrus426 collected $244 from pot
*** SUMMARY ***
Total pot $247 | Rake $3
Board [2h 3s 6h 2d Qh]
Seat 1: Codrus426 showed [9d Ah] and won ($244) with a pair of Deuces
Seat 2: JMBigJoe mucked
Seat 3: xinxin1 folded before Flop (didn't bet)
Seat 4: moshmachine (button) folded before Flop (didn't bet)
Seat 5: TopKat5757 (small blind) folded before Flop
Seat 6: LukaschenkoA (big blind) folded before Flop
我想把它解析为我正在研究的Hand
课......
public class Hand
{
public long ID;
public string Stakes;
public DateTime Date;
public IDictionary<Street, decimal> Pots;
public decimal FinalPot;
public decimal Rake;
public Player Hero;
public IDictionary<Player, PlayerInfo> Players;
public IList<Card> Board;
public IList<Decision> Actions;
public Hand()
{
this.Players = new Dictionary<Player, PlayerInfo>();
this.Board = new List<Card>();
this.Actions = new List<Decision>();
this.Pots = new Dictionary<Street, decimal>();
}
}
public class PlayerInfo
{
public Player Player;
public decimal Stack;
public decimal Summary;
public Position Position;
public Card Holecards;
}
答案 0 :(得分:2)
您的解决方案正在运行,但由于您将全局变量与继承相结合,因此难以理解。
如果您使用的是NUnit 2.5或更高版本,则可以使用Parameterized Tests with TestCaseAttribute。
[TestCase("John,29","John",29,"")]
[TestCase(",13","",13,"")]
public void ParserTest(Sting stringToParse, String expextedName, int expectedAge, String expectedComment)
{
IParser _parser = new Parser();
Person _actual = _parser.Parse(stringToParse);
Assert.AreEqual(expextedName, _actual.Name, stringToParse + " failed on Name");
Assert.AreEqual(expextedAge, _actual.Age, stringToParse + " failed on Age");
Assert.AreEqual(expextedComment, _actual.Comment, stringToParse + " failed on Comment");
}
我认为这更容易理解。
如果您需要继续使用mstest,则必须按how-to-rowtest-with-mstest
中的描述进行模拟答案 1 :(得分:0)
我认为你所拥有的结构非常好,记住TMTOWTDI。
但是,请确保明确地测试Parse
是否为空,空(即零长度)和空白字符串。更复杂/预期路径的情况当然是测试的关键,但像这样的简单情况也很重要。
此外,您正在使用的结构很可能会折叠到ParserTests
中,并使用常见的%MethodUnderTest%_With%Condition(s)%_Expect%ExpectedResult%
测试用例命名约定和可选的Test Data Builder模式,以使另一个类负责构建像您现在在ParserTests
子类