Why are field variables null after setting them in JUnitParams method?

2k Views Asked by At

I am writing a test class using JUnitParams and Mockito. I want to use a Mockito mock as parameter. In my test I have about ten mocks and I want to pass only one mock to define a special behaviour for it.

I reproduced the problem in a simple example.

My problem: I initialize the variable myList in the method parametersForTest, but when I debug into the test method myList is null, but param is my desired mock.

@RunWith(JUnitParamsRunner.class)
public class MockitoJUnitParamsTest {

    private List myList;

    @Test
    @Parameters
    public void test(List param) {
        assertThat(param).isEqualTo(this.myList);
    }

    public Object[] parametersForTest() {
        myList = Mockito.mock(List.class);
        return new Object[]{myList};
    }
}

I use

  • JUnitParams version 1.0.5
  • JUnit version 4.12
  • Mockito version 1.10.19

Why is myList null and how can I fix that?

2

There are 2 best solutions below

0
On BEST ANSWER

The instance of the MockitoJUnitParamsTest class running parametersForTest() method is not the same running test(). This is because Junit will create a different instance for each test method. In your case , making myList static

private static List myList;

can solve the problem partially, but It may fail if the tests run in parallel.

0
On

JUnit frameworks and runners often create separate instances of the test class for each method call. It looks like that's what JUnitParams does - you can see that using the following test:

@RunWith(JUnitParamsRunner.class)
public class JUnitParamsTest {

    @Test
    @Parameters
    public void test(JUnitParamsTest param) {
        Assert.assertNotNull(param);
        Assert.assertNotSame(this, param);
    }

    Object[] parametersForTest() {
        return new Object[]{this};
    }

}

So the problem with your example above is that the myList field being set by parametersForTest is a member of a different object from the one that test(List) is being called on.

How to fix

Well, I guess the key question is, what are you trying to achieve here? The whole point of JUnitParams is that parameters are injected into the test methods, so you don't have to use fields. It seems to me that modifying field values inside the parametersForTest() method is outside of intended usage, and I can't really see why you'd want to do this.

One quick and dirty fix would be to make myList static, which should work so long as your test class isn't accessed by multiple threads concurrently (some unit-test environments do run multi-threaded so there is a risk here).

A better solution would be to redesign your tests so that you're not modifying fields from inside of parameter-generating methods. The example you've given doesn't appear to be attempting to test anything other than JUnitParams itself, so I can't help you determine what a good design would be, as I son't know what you're trying to achieve. Are you saying you want several test methods to share a single Mockito mock? If so, why? I'm not sure Mockito would support this (I haven't tried). Unit tests are generally supposed to be isolated from each other, so the recommended action would be to create a new mock each time.