7

在过去的几天里,我开始玩 roboguice、robolectric 和 mockito。我有一个带有登录屏幕的小型 Android 应用程序,其中包含 AutoCompleteTextView 以便更快地输入用户名。AutoCompleteTextView 的用户名存储在 sqlite 数据库中。

public class MainActivity extends RoboActivity implements View.OnClickListener {
@InjectView(R.id.startScreen_Login_Button) private Button loginButton;
@InjectView(R.id.startScreen_Cancel_Button) private Button cancelButton;
@InjectView(R.id.startScreen_forgotPwd_TextView) private TextView forgotPWTextView;
@InjectView(R.id.startScreen_Username_AutoCompleteTextView) private AutoCompleteTextView loginUsernameAutoCompleteTextView;
@InjectView(R.id.startScreen_Password_EditText) private EditText loginPasswordEditText;
@Inject private SharedPreferences sharedPreferences;
@Inject SQLiteDBAdapter dbAdapter;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    loginButton.setOnClickListener(this);
    cancelButton.setOnClickListener(this);
    forgotPWTextView.setOnClickListener(this);

    // Creating List for startScreen_Username_AutoCompleteTextView
    List<User> userList = dbAdapter.getUserList();
    ListIterator<User> it = userList.listIterator();
    List<String> userStringList = new ArrayList<String>();
    User user;
    while (it.hasNext()) {
        user = it.next();
        userStringList.add(user.getName());
    }

    loginUsernameAutoCompleteTextView.setAdapter(new ArrayAdapter<String>(this, R.layout.select_page_row, userStringList));
    }
...
}

我想使用 robolectric 测试 MainActivity,尝试用 mockito 模拟数据库。这是我的测试课:

@RunWith(CustomRobolectricTestRunner.class)
public class MainActivityTest {

@Mock
SQLiteDBAdapter dbAdapter;

@Before
public void setUp() throws Exception {
    MockitoAnnotations.initMocks(this);
}

@Test
public void shouldHaveApplicationName() throws Exception {
    String appName = new MainActivity().getResources().getString(R.string.app_name);
    assertThat(appName, equalTo("OperationReport"));
}

@Test
public void testButtonsVisible()
{
    MainActivity mainActivity = new MainActivity();
    mainActivity.onCreate(null);
}
}

调用 mainActivity.onCreate(null); 正在启动错误级联,以行结束 Cursor cursor = db.rawQuery(SQL_QUERY, null); 我的 SQLiteDBAdapter 中的 getUserList 方法:

public List<User> getUserList() {
    SQLiteDatabase db = getReadableDatabase();
    List<User> userList = new ArrayList<User>();

    String SQL_QUERY = "SELECT * FROM User;";
    Cursor cursor = db.rawQuery(SQL_QUERY, null);
    cursor.moveToFirst();
    while (!cursor.isAfterLast()) {
        User user = new User();
        user.setUserUUID(cursor.getString(0));
        user.setName(cursor.getString(1));
        user.setPassword(cursor.getString(2));
        user.setDateOfBirth(cursor.getString(3));
        user.setStaffNumber(cursor.getString(4));
        user.setActive(cursor.getInt(5));
        user.setUserClass(cursor.getInt(6));
        userList.add(user);
        cursor.moveToNext();
    }
    cursor.close();
    db.close();
    return userList;
}

我读到,一个 Mock 正在返回 void 方法的空存根,并在任何其他方法上返回 null。当我在模拟 SQLiteDBAdapter 类时,我期望在模拟的 SQLiteDBAdapter 上调用 getUserList 会返回 null。我不太清楚,为什么他要访问原始方法。我猜它仍然使用原始的 SQLiteDBAdapter 而不是 Mock。我该怎么做才能解决这个问题,它是如何工作的?我的想法用完了,所以任何帮助表示赞赏。

4

1 回答 1

3

模拟数据库来测试 DAO 对我来说毫无意义。你在测试什么?数据库。为什么要消除它?

一旦所有的 DAO 测试都通过了,那么模拟数据库就很有意义,并且是时候测试使用它来完成工作单元的服务了。您已经测试了 DAO 和数据库,并且您的服务单元测试不必是集成测试。在那种情况下一定要嘲笑。

我不太了解你在嘲笑什么,但是当我嘲笑它是为了我制作的界面。模拟为我的客户端/测试正在使用的接口类型引用提供了一个替代实现。

如果您尝试模拟一个具体的类,我建议将该适配器包装在基于接口的实现中。这将是一个更好的抽象,并且您可以更轻松地模拟您的界面。

于 2013-04-10T22:18:29.363 回答