如何模拟getApplicationContext

| 我有一个存储应用程序上下文信息的应用程序。应用程序上下文信息在MyApp类中的活动之间共享,该类扩展了Application类。 我正在为我的活动编写单元测试,我想检查一下,当用户单击活动中的按钮时,应用程序状态是否会更改。像这样:
@Override
public void onClick(View pView) {
    ((MyApp)getApplicationContext()).setNewState();
}   
问题是我不知道如何模拟该应用程序上下文。我正在使用ActivityUnitTestCase作为测试用例库。当我调用setApplication时,它将更改Activity类的mApplication成员的值,但不会更改应用程序上下文。我也尝试过setActivityContext,但这似乎是错误的(不是应用程序上下文而是活动上下文),它会在startActivity内部触发断言。 所以问题是-如何模拟getApplicationContext()?     
已邀请:
由于方法
getApplicationContext
在您要扩展的类内部,因此变得有些问题。有几个问题需要考虑: 您真的无法模拟正在测试的类,这是对象继承(即子类化)的众多缺点之一。 另一个问题是
ApplicationContext
是一个单例,这使得测试变得更加邪恶,因为您无法轻易地模拟出被编程为不可替代的全局状态。 在这种情况下,您可以做的是优先于继承而不是继承。因此,为了使您的“ 3”可测试,您需要对逻辑进行一些分解。假设您的3英镑称为5英镑。它需要由一个逻辑组件(或类)组成,命名为“ 6”。这是一个简单的类图图: 为了解决单例问题,我们让逻辑与应用程序上下文“注入”,以便可以使用模拟对其进行测试。然后,我们仅需要测试
MyActivity
对象是否已将正确的应用程序上下文放入
MyActivityLogic
中。我们基本上如何解决这两个问题是通过另一层抽象(由Butler Lampson解释)。在这种情况下,我们添加的新层是将活动逻辑移到活动对象之外。 为了您的示例,类需要看起来像这样:
public final class MyActivityLogic {

    private MyApp mMyApp;

    public MyActivityLogic(MyApp pMyApp) {
        mMyApp = pMyApp;
    }

    public MyApp getMyApp() {
        return mMyApp;
    }

    public void onClick(View pView) {
        getMyApp().setNewState();
    }
}

public final class MyActivity extends Activity {

    // The activity logic is in mLogic
    private final MyActivityLogic mLogic;

    // Logic is created in constructor
    public MyActivity() {
        super(); 
        mLogic = new MyActivityLogic(
            (MyApp) getApplicationContext());
    }

    // Getter, you could make a setter as well, but I leave
    // that as an exercise for you
    public MyActivityLogic getMyActivityLogic() {
        return mLogic;
    }

    // The method to be tested
    public void onClick(View pView) {
        mLogic.onClick(pView);
    }

    // Surely you have other code here...

}
它应该看起来像这样: 要测试
MyActivityLogic
,您只需要一个简单的jUnit
TestCase
,而不是
ActivityUnitTestCase
(因为它不是Activity),并且您可以使用选择的模拟框架来模拟您的应用程序上下文(因为手动滚动自己的模拟点有点拖动)。示例使用Mockito:
MyActivityLogic mLogic; // The CUT, Component Under Test
MyApplication mMyApplication; // Will be mocked

protected void setUp() {
    // Create the mock using mockito.
      mMyApplication = mock(MyApplication.class);
    // \"Inject\" the mock into the CUT
      mLogic = new MyActivityLogic(mMyApplication);
}

public void testOnClickShouldSetNewStateOnAppContext() {
    // Test composed of the three A\'s        
    // ARRANGE: Most stuff is already done in setUp

    // ACT: Do the test by calling the logic
    mLogic.onClick(null);

    // ASSERT: Make sure the application.setNewState is called
    verify(mMyApplication).setNewState();
}
要测试
MyActivity
,您可以照常使用
ActivityUnitTestCase
,我们只需要确保它创建的
MyActivityLogic
带有正确的
ApplicationContext
即可。粗略的测试代码示例可以完成所有这些工作:
// ARRANGE:
MyActivity vMyActivity = getActivity();
MyApp expectedAppContext = vMyActivity.getApplicationContext();

// ACT: 
// No need to \"act\" much since MyActivityLogic object is created in the 
// constructor of the activity
MyActivityLogic vLogic = vMyActivity.getMyActivityLogic();

// ASSERT: Make sure the same ApplicationContext singleton is inside
// the MyActivityLogic object
MyApp actualAppContext = vLogic.getMyApp();
assertSame(expectedAppContext, actualAppContext);
希望这对您有意义并可以帮助您。     

要回复问题请先登录注册