2013-04-10 60 views
7

在过去的几天里,我开始玩弄roboguice,robolectric和mockito。我有一个带有一个包含AutoCompleteTextView的登录屏幕的小型Android应用程序,可以更快地输入用户名。 AutoCompleteTextView的用户名存储在sqlite数据库中。
在Robolectric测试Activity时嘲笑SQLite-Database

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)); 
    } 
... 
} 

我想用MainActivity robolectric,试图用嘲笑的的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 = db.rawQuery(SQL_QUERY,null)结束;我getUserList法在我SQLiteDBAdapter的:

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; 
} 

我读,一个是模拟返回的空隙方法空存根,以及任何其他方法返回null。由于我嘲笑SQLiteDBAdapter类,我期望调用我的模拟SQLiteDBAdapter getUserList返回null。我不清楚,他为什么要访问原始方法。我想它仍然使用原始的SQLiteDBAdapter而不是模拟。我需要做些什么来解决这个问题,它是如何工作的?我跑出了想法,所以任何帮助表示赞赏。

回答

2

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

一旦你所有的DAO测试都通过了,模拟数据库就有意义了,现在是时候测试用户完成一个工作单元的服务了。您已经测试了DAO和数据库,并且您的服务单元测试不需要进行集成测试。在这种情况下通过一切手段模拟。

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

如果你想模拟一个具体类,我建议在基于接口的实现中包装该适配器。这将是一个更好的抽象,你会更容易地嘲笑你的界面。

+0

我想测试我的登录活动。此活动正在使用数据库来获取AutoCompleteTextview的userList。我想用Mock替换数据库,所以我可以测试登录活动(按钮等)而不依赖于数据库。我考虑测试一些按钮和EditText一个很好的练习,因为我只是玩了几天robolectric和mockito。 – Frank 2013-04-10 22:32:21

+0

好的,这是不同的 - 你正在测试用户界面。我会假设你已经测试过DAO,所以我对这项服务的评论会适合你。 – duffymo 2013-04-10 22:58:24

+0

嘲讽第三方代码让我对世界一切有意义。你为什么要测试别人的代码?为什么你需要ACTUALLY有一个数据库的开销。当您测试DAO时,您未测试数据库或其驱动程序的功能,因为您正在测试与数据库的交互是否符合您的期望。这正是模拟的目的。例如重复的更新在敲击真实的数据库时不会被发现,但通常不是预期的或想要的。这是通过模拟发现的。不幸的是,你无法控制什么是可嘲笑的,所以抽象是唯一的方法。 – 2013-07-17 19:48:12