2013-04-10 106 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