- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
环境:NUnit 2.6.4; NUnit 测试适配器 2.0.0;最小起订量 4.2.1; VisualStdio 4.6;
简介:我实现了一个 HomeSecuritySystem,其中有供应商的传感器、警报、电源、显示器的接口(interface)。我想测试它的实现。我想测试“SystemCheckPass”子例程。
我模拟一个 ISensor 列表并构造“SecurityController”,然后运行“SystemCheckPass”子例程。当它执行到“SystemCheckPass”子例程时,“foreach”无法正常运行,它显示“NullReferenceException”。
在 ISensors 的 MockList 上的“foreach”循环的测试函数中一切正常。简单地将 ref 传递给“SecurityController”后,“list”的“foreach”“循环”无法找到其元素的“ref”。我对此很困惑。
谁能告诉我为什么?
安全 Controller 的实现。
using System;
using System.Collections.Generic;
using HomeSecuritySystem.Sensors;
using HomeSecuritySystem.Comms;
using HomeSecuritySystem.Power;
using HomeSecuritySystem.Display;
using HomeSecuritySystem.Alarm;
using HomeSecuritySystem.Report;
using System.Threading;
using System.Diagnostics;
namespace HomeSecuritySystem
{
public class SecurityController : ControllerBase
{
public int Value;
public ICollection<ISensor> sensors;
public IComms comms;
public IPowerSupply powerSupply;
public IAlarm alarm;
public IDisplay display;
public SecurityController(ICollection<ISensor> sensors, IComms comms, IPowerSupply powerSupply, IAlarm alarm, IDisplay display)
: base(sensors, comms, powerSupply, alarm, display)
{
this.sensors = sensors;
this.comms = comms;
this.powerSupply = powerSupply;
this.alarm = alarm;
this.display = display;
// bind the power down event when system initials, because it never changes.
powerSupply.OnNoPower += new Events.NoPowerEvent(PowerSupplyNoPower);
// initialize
IsArmedLastSate = IsArmed;
IsStayLastSate = IsStay;
}
public bool SystemCheckPass()
{
// <---- begin system check
bool systemCheckPass = true;
// check the device power except sensors
if (alarm.IsOn == false || comms.IsOn == false || powerSupply.IsOn == false)
{
// part of system check, power is off, system check fail
systemCheckPass = false;
}
// check power of sensors
foreach (ISensor sensor in sensors)
{
if (sensor.IsOn == false)
systemCheckPass = false;
}
/*
// check the battery of sensors
foreach (ISensor sensor in sensors)
{
switch (sensor.Type)
{
// part of system check, motion sensor's battery is low, system check fail
case Report.SensorType.Motion:
IMotionSensor motionSensor = (IMotionSensor)(sensor);
if (motionSensor.IsLowBattery == true)
{
systemCheckPass = false;
}
break;
// part of system check, smoke sensor's battery is low, system check fail
case Report.SensorType.Smoke:
ISmokeSensor smokeSensor = (ISmokeSensor)(sensor);
if (smokeSensor.IsLowBattery == true)
{
systemCheckPass = false;
}
break;
default:
// so far, only two kinds of sensors
throw new Exception("no exit sensor type!");
}
}
//check the battery of power supply
if (powerSupply.IsLowBattery)
{
systemCheckPass = false;
}
*/
// system check over ---->
return systemCheckPass;
}
public override void SystemCheck()
{
// <---- begin system check
bool systemCheckPass = true;
List<int> lowBatterySensorIDList = new List<int>();
// check the device power except sensors
if (alarm.IsOn == false || comms.IsOn == false || powerSupply.IsOn == false)
{
// part of system check, power is off, system check fail
systemCheckPass = false;
}
// check power of sensors
foreach (ISensor sensor in sensors)
{
if (sensor.IsOn == false)
systemCheckPass = false;
}
bool lowBatterySensorExist = false;
// check the battery of sensors
foreach (ISensor sensor in sensors)
{
switch (sensor.Type)
{
// part of system check, motion sensor's battery is low, system check fail
case Report.SensorType.Motion:
IMotionSensor motionSensor = (IMotionSensor)(sensor);
if (motionSensor.IsLowBattery == true)
{
lowBatterySensorIDList.Add(motionSensor.Id);
systemCheckPass = false;
lowBatterySensorExist = true;
}
break;
// part of system check, smoke sensor's battery is low, system check fail
case Report.SensorType.Smoke:
ISmokeSensor smokeSensor = (ISmokeSensor)(sensor);
if (smokeSensor.IsLowBattery == true)
{
lowBatterySensorIDList.Add(smokeSensor.Id);
systemCheckPass = false;
lowBatterySensorExist = true;
}
break;
default:
// so far, only two kinds of sensors
throw new Exception("no exit sensor type!");
}
}
//check the battery of power supply
bool BatteryOfPowSupplyIsLow = false;
if (powerSupply.IsLowBattery)
{
systemCheckPass = false;
BatteryOfPowSupplyIsLow = true;
}
// system check over ---->
// <---- report begin
if (systemCheckPass)
{
display.ShowSystemReady();
}
else
{
display.ShowSystemNotReady();
}
if (lowBatterySensorExist)
{
display.ShowSensorLowBattery(lowBatterySensorIDList);
}
if (BatteryOfPowSupplyIsLow)
{
display.ShowPowerSupplyLowBattery();
}
// report end --->
}
public override void ClearMemory()
{
display.ClearSentReport();
display.ClearAlarmSound();
display.ClearSystemArmed();
foreach (ISensor sensor in sensors)
{
display.ClearSensorDetected(sensor.Id);
}
}
// bind to event sensor.OnDetectionStateChanged
public void ArmSensorDetected(ISensor sensor)
{
if (sensor.Detected)
{
alarm.SoundAlarm();
display.ShowAlarmSound();
display.ShowSensorDetected(sensor.Id);
Report.Report report = new Report.Report();
report.SensorId = sensor.Id;
report.SensorType = sensor.Type;
report.Time = new DateTime();
report.Type = Report.ReportType.Intrusion;
display.ShowSentReport("sensor detected");
comms.InformSecurity("sensor detected");
}
}
public void ArmStaySensorDetected(ISensor sensor)
{
if (sensor.Type == Report.SensorType.Motion)
{
IMotionSensor motionSensor = (IMotionSensor)(sensor);
if (motionSensor.Detected && motionSensor.IsPerimeterSensor)
{
alarm.SoundAlarm();
display.ShowAlarmSound();
display.ShowSensorDetected(sensor.Id);
display.ShowSentReport("sensor detected");
comms.InformSecurity("sensor detected");
}
}
}
public void SmokeSensorDetected(ISensor sensor)
{
if (sensor.Type == Report.SensorType.Smoke)
{
ISmokeSensor smokeSensor = (ISmokeSensor)(sensor);
if (smokeSensor.Detected)
{
Report.Report report = new Report.Report();
report.SensorId = sensor.Id;
report.SensorType = sensor.Type;
report.Time = new DateTime();
report.Type = Report.ReportType.Smoke;
alarm.SoundAlarm();
display.ShowAlarmSound();
display.ShowSensorDetected(sensor.Id);
display.ShowSentReport("sensor detected");
comms.InformSecurity("sensor detected");
}
}
}
public void PowerSupplyNoPower()
{
Report.Report report = new Report.Report();
report.Time = new DateTime();
report.Type = Report.ReportType.NoPower;
comms.InformSecurity("power down");
display.ShowSentReport("power down");
}
private bool IsArmedLastSate;
private bool IsStayLastSate;
private void DelegateHandling()
{
foreach (ISensor sensor in sensors)
{
sensor.OnDetectionStateChanged += new Events.SensorDetectionStateChangeEvent(ArmSensorDetected);
}
if (IsArmed == true && IsStay == false)
{
foreach (ISensor sensor in sensors)
{
sensor.OnDetectionStateChanged += new Events.SensorDetectionStateChangeEvent(ArmSensorDetected);
}
}
else if (IsArmed == true && IsStay == true)
{
foreach (ISensor sensor in sensors)
{
sensor.OnDetectionStateChanged += new Events.SensorDetectionStateChangeEvent(ArmStaySensorDetected);
}
}
else if (IsArmed == false && IsStay == false)
{
alarm.StopAlarm();
}
foreach (ISensor sensor in sensors)
{
sensor.OnDetectionStateChanged += new Events.SensorDetectionStateChangeEvent(SmokeSensorDetected);
}
}
public void Run()
{
// initial delegete bind
DelegateHandling();
//delegate sensor
for (;;)
{
Thread.Sleep(100);
SystemCheck();
if (IsArmedLastSate != IsArmed || IsStayLastSate != IsStay)
{
// when Security Controller change mode, rebind all sensors' delegete.
DelegateHandling();
IsArmedLastSate = IsArmed;
IsStayLastSate = IsStay;
}
}
}
static int Main(string[] args)
{
//...
return 0;
}
}
}
我使用 NUnit 和 Moq 进行单元测试。系统检查通过的单元测试。
using HomeSecuritySystem.Alarm;
using HomeSecuritySystem.Sensors;
using Moq;
using System.Collections.Generic;
namespace HomeSecuritySystem
{
using Comms;
using Display;
using NUnit.Framework;
using Power;
[TestFixture]
public class SecurityControllerTest
{
[Test]
public void Test_System_Check()
{
//arrange
Mock<IMotionSensor> mockMotionSensor = new Mock<IMotionSensor>();
mockMotionSensor.SetupGet(t => t.IsLowBattery).Returns(true);
//mock sensors
Mock<List<ISensor>> mockSensors = new Mock<List<ISensor>>();
mockSensors.Object.Add(mockMotionSensor.Object);
foreach (ISensor sensor in mockSensors.Object)
{
if (sensor.IsOn == false)
;
}
Mock<IComms> mockComms = new Mock<IComms>();
mockComms.SetupGet(t => t.IsOn).Returns(true);
Mock<IPowerSupply> mockPowerSupply = new Mock<IPowerSupply>();
mockPowerSupply.SetupGet(t => t.IsOn).Returns(true);
Mock<IAlarm> mockAlarm = new Mock<IAlarm>();
mockAlarm.SetupGet(t => t.IsOn).Returns(true);
Mock<IDisplay> mockDisplay = new Mock<IDisplay>();
SecurityController securityController = new SecurityController(mockSensors.Object, mockComms.Object, mockPowerSupply.Object, mockAlarm.Object, mockDisplay.Object);
//act
Assert.AreEqual(securityController.SystemCheckPass(), true);
}
}
}
NUnit 测试框架给出的错误:
Test Name: Test_System_Check
Test FullName: HomeSecuritySystem.SecurityControllerTest.Test_System_Check
Test Source: C:\Users\OEM\Documents\Visual Studio 2015\Projects\HomeSecurityController\HomeSecurityController.UnitTest\SecurityControllerTest.cs : line 17
Test Outcome: Failed
Test Duration: 0:00:00.569
Result StackTrace:
at HomeSecuritySystem.SecurityController.SystemCheckPass() in C:\Users\OEM\Documents\Visual Studio 2015\Projects\HomeSecurityController\HomeSecurityController\SecurityController.cs:line 58
at HomeSecuritySystem.SecurityControllerTest.Test_System_Check() in C:\Users\OEM\Documents\Visual Studio 2015\Projects\HomeSecurityController\HomeSecurityController.UnitTest\SecurityControllerTest.cs:line 42
Result Message: System.NullReferenceException : Object reference not set to an instance of an object.
最佳答案
这主要是因为GetEnumerator
被要求为您的 IList<T>
,你没有 mock 。
我建议不要模拟类似集合的对象,如 IList<T>
.只需传递它们的实际实现即可。否则就像你在测试 foreach
有效。
关于C# 模拟测试 - Moq 列表不能是 'foreach',告诉 'NullReferenceException',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35120534/
这是一个假设性问题。如果我有来自 3 个单独的 sql db 查询的 3 个数组,这些查询都与另一个数组相关。例如…… //db schools id | school_name classes id
在我的应用程序中,我使用 scrape(string url) 方法从网页中抓取链接。可以说它每次都返回我 10 个 url。 我想从每个抓取的 url 中抓取 10 个链接。 长话短说: (第 1
我的java7代码: final Map result = new HashMap<>(); final Set> classes = getClasses(co.glue()); for (fina
是否可以在 SwiftUI 中设置变量,例如在这样的 ForEach 中: struct ContentView: View { var test: Int var body: som
在 D、int、uint 中使用 foreach 时,循环索引的首选类型是什么?或者只是通过省略类型自动实现? 最佳答案 一般来说,索引应该是size_t。与长度相同。如果您尝试使用 int 或 ui
根据 http://dlang.org/statement.html 的“Foreach 限制”部分以下代码 int[] a; int[] b; foreach (int i; a) { a
在什么情况下我们应该在 JDK 8 中使用旧的 foreach 循环遍历新的 collection.forEach() 还是最好的做法是转换 every foreach 循环?是否存在任何重要的性能差
获得类似东西的惯用方法是什么? ((fn [coll] (function-body)) [:a :b :c :d]) -> [[:a :b][:a :c][:a :d][:b :c][:b :d][
我正在创建一个基于 who is it? 的 Java 应用程序。现在我正在制作一种方法,在回答问题时我需要其他卡片。 我有两个列表: 列表是一个 ImageView 列表,其中我有卡片必须代表的 2
我希望有人能在我发疯之前帮助我。 我有 3 张 table : Table A SELECT companypk, companyname, logo, msscope FROM global_com
我正在尝试将多个字符串添加到 C# 中的 MailAddress。 如果我使用ForEach,我的代码会是这样 foreach (var item in GetPeopleList()
我没有太多的 C# 经验,所以如果有人能指出正确的方向,我将不胜感激。我有一个引用对象变量的 foreach 循环。我希望在主循环中创建另一个 foreach 循环,将当前变量与对象数组中的其余变量进
下面的代码每 60 秒删除文件夹“Images”中的文件,它可以工作,但是当文件夹为空时它会显示:警告:为 foreach() 提供的参数无效如果没有文件,如何解决这个问题,说“文件夹为空而不是那个警
我需要在两种不同的模式下运行,因此“if”(第二个稍后构建一个大的 csv) 下面对于单个实例运行正常,但在第二个 (*) 的加载时间上失败,因为在前 7k 行中的每一行上运行。 我想避免可怕的事情
我们可以使用以下两种方法实现类数组对象的迭代: let arrayLike = document.getElementsByClassName('dummy'); [].forEach.call(ar
我有这个代码 ... 它说: Attribute value invalid for tag forEach according to TLD 最佳答案 forEach标签不支持 valu
我在 SwiftUI 中有一个像这样的 ForEach: ForEach(entries) { (e: MyType) in NavigationLinkItem(entry: e) } 现在我
我无法在一个 Foreach 或 Foreach-Object 循环中使用多个命令 我的情况是—— 我有很多文本文件,大约 100 个。 所以他们被阅读 Get-ChildItem $FilePath
我必须从 json 文件(实际上是 2 个 json 文件)执行 ForEach,因此我执行 2 forEach,代码是 table { font-family: arial, sans-
我对编程很陌生,当我执行 forEach 函数时,我的应用程序返回错误。我的controller.js中有以下代码 $scope.ajaxRequest = A.Game.get({action: '
我是一名优秀的程序员,十分优秀!