又懵逼了,getModel() 和 getView() 里面还是会调用安卓的代码,怎么使用Junit做测试呢?
  引入一个强大的测试框架:Mockito,接下来可以开始使用Junit & Mockito做Java代码的单元测试了,这种方式的单元测试可以直接运行与JVM上,使用Mockito隔离Android相关代码。
  然后可以为CreditCardPresenter 写单元测试了, 为了方便,静态导入了Mockito的所有方法 :
import static org.mockito.Mockito.*;
@RunWith(MockitoJUnitRunner.class)
public class CreditCardPresenterTest {
CreditCardPresenter creditCardPresenter;
@Mock
CreditCardContract.View creditCardView;
@Mock
CreditCardContract.Model creditCardModel;
List<CreditCard> creditCards;
@Before
public void setUp() throws Exception {
creditCardPresenter = new CreditCardPresenter();
creditCardPresenter.attachView(creditCardView);
creditCardPresenter.setModel(creditCardModel);
creditCards = new ArrayList<>();
}
public void testGetCreditCards() {
when(creditCardModel.getCreditCards()).thenReturn(Observable.create(new Observable.OnSubscribe<List<CreditCard>>() {
@Override
public void call(Subscriber<? super List<CreditCard>> subscriber) {
subscriber.onNext(creditCards);
subscriber.onCompleted();
}
}));
creditCardPresenter.getCreditCards();
verify(creditCardView).showCreditCards(creditCards);
verify(creditCardView).loadCompleted();
}
public void testGetCreditCardsOnError() {
final RuntimeException exception = new RuntimeException();
when(creditCardModel.getCreditCards()).thenReturn(Observable.create(new Observable.OnSubscribe<List<CreditCard>>() {
@Override
public void call(Subscriber<? super List<CreditCard>> subscriber) {
throw exception;
}
}));
creditCardPresenter.getCreditCards();
verify(creditCardView).showError(exception);
}
}
  这样为上述两种情况写了两个单元测试。
  其中使用@Mock注解来生成mock对象,也可以setUp方法中使用Mockito.mock()来生成mock对象,当使用注解的时候在类上必须加上注解@RunWith(MockitoJUnitRunner.class)
  mock出来的对象的方法都是空实现,void方法声明也不做,有返回值的方法返回null(int 类型返回0,boolean类型返回false等)。
  然后我们可以通过when(…).thenReturn(…)来为mock对象实现方法返回值。
  when(creditCardModel.getCreditCards()).thenReturn(Observable.create(new Observable.OnSubscribe<List<CreditCard>>() {
  @Override
  public void call(Subscriber<? super List<CreditCard>> subscriber) {
  subscriber.onNext(creditCards);
  subscriber.onCompleted();
  }
  }));
  上面代码的意思是说当调用creditCardModel.getCreditCards()的时候返回值是:
  Observable.create(new Observable.OnSubscribe<List<CreditCard>>() {
  @Override
  public void call(Subscriber<? super List<CreditCard>> subscriber) {
  subscriber.onNext(creditCards);
  subscriber.onCompleted();
  }
  })
  后使用verify()方法来校验某个方法是否被执行:
  verify(creditCardView).showCreditCards(creditCards);
  上面的代码意思是说 creditCardView.showCreditCards(creditCards)方法被执行了,并且参数是creditCards,并且只执行了一次。如果有一个条件不符合会报测试失败。
  verify()还有很多重载方法,默认其实是这样的 verfy(creditCardView, times(1)).showCreditCards(creditCards); 校验只执行了一次,times(1) 可以传入不同的参数来校验方法被执行了几次。还可以替换了nerver(),表示某方法一次也不执行。
  当然Mockito的功能远不止这么点,还有很多高级用法不继续介绍了。
  Mockito也有一些美中不足之处,不能mock静态方法,final方法等,比如项目中会有这样的方法 SelfApplication.getContext() 来获取自定义的Application,如果在测试代码中出现这类代码肯定会测试失败,因为JVM环境中没有Application,怎么办呢?
  再引入一个配合Mockito使用的库:PowerMock
  他弥补了Mockito的不足,可以mock静态方法和final方法,可以使用PowerMock来mock出SelfApplication.getContext(),从而不会调用到真正的Application对象:
  PowerMockito.mockStatic(SelfApplication.class);
  PowerMockito.when(SelfApplication.getContext()).thenReturn(mock(SelfApplication.class));
  另外,在方法上要声明@PrepareForTest(SelfApplication.class), 在类上要声明  @RunWith(PowerMockRunner.class) 来支持上述mock。
  这样当 调用SelfApplication.getContext()的时候将拿到一个mock对象,我们可以继续使用when().thenReturn()方法来处理方法返回值了。
  具体关于Mockito 和 PowerMock 的更多用法这里不做过多介绍了,官网才是好的教程。
  这只是一个简单的例子,实际项目中会出现好的复杂的情况。这要求我写的代码方法要短,耦合要低。写单元测试逼迫我们写更优雅的代码,也为我们下次修改需求或者重构代码提供了一道安全保障。还有其他更多的好处大家自己在实践中体会吧。