Макет объекта и интерфейса - PullRequest
3 голосов
/ 11 мая 2010

Я новичок в модульном тестировании с Mock Object. Я использую EasyMock. Я пытаюсь понять этот пример:

import java.io.IOException;

public interface ExchangeRate {

    double getRate(String inputCurrency, String outputCurrency) throws IOException;

}

import java.io.IOException;


public class Currency {

    private String units;
    private long amount;
    private int cents;


    public Currency(double amount, String code) {
        this.units = code;
        setAmount(amount);
    }

    private void setAmount(double amount) {
        this.amount = new Double(amount).longValue();
        this.cents = (int) ((amount * 100.0) % 100);
    }

    public Currency toEuros(ExchangeRate converter) {
        if ("EUR".equals(units)) return this;
        else {
            double input = amount + cents/100.0;
            double rate;
            try {
                rate = converter.getRate(units, "EUR");
                double output = input * rate;
                return new Currency(output, "EUR");
            } catch (IOException ex) {
                return null;
            }
        }
    }

    public boolean equals(Object o) {
        if (o instanceof Currency) {
            Currency other = (Currency) o;
            return this.units.equals(other.units)
                    && this.amount == other.amount
                    && this.cents == other.cents;
        }
        return false;
    }

    public String toString() {
        return amount + "." + Math.abs(cents) + " " + units;
    }

}

import junit.framework.TestCase;
import org.easymock.EasyMock;
import java.io.IOException;

public class CurrencyTest extends TestCase {

    public void testToEuros() throws IOException {
        Currency testObject = new Currency(2.50, "USD");
        Currency expected = new Currency(3.75, "EUR");
        ExchangeRate mock = EasyMock.createMock(ExchangeRate.class);
        EasyMock.expect(mock.getRate("USD", "EUR")).andReturn(1.5);
        EasyMock.replay(mock);
        Currency actual = testObject.toEuros(mock);
        assertEquals(expected, actual);
    }

}

Итак, мне интересно, как использовать валюту ExchangeRate в методе toEuros(..).

rate = converter.getRate(units, "EUR");

Поведение метода getRate(..) не указано, поскольку ExchangeRate является интерфейсом.

/********************************************************************************/

Так что я пытаюсь сделать себе пример. Ниже приведены мои коды:

/**
 *Interface to access data
 */
public interface Dao {
    public boolean getEntityById(int id) throws SQLException;
}

/**
 *Business class do something in business layer
 */
public class Bussiness {
    private Dao dao;

    public Dao getDao() {
        return dao;
    }

    public void setDao(Dao dao) {
        this.dao = dao;
    }

    public boolean doSomeThing(int id) throws SQLException {
        if(dao.getEntityById(id)) {
            return true;
        } else {
            return false;
        }
    }

    public static void main(String[] args) throws SQLException {
        Bussiness b = new Bussiness();
        b.doSomeThing(3);
    }
}


package tunl;

import java.sql.SQLException;

import org.easymock.EasyMock;
import org.testng.Assert;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.Test;

    /**
     * This is my unit Test
     */
    @Test
    public class MyUnitTest {
        private Bussiness bussiness;
        private Dao mock;

        @BeforeTest
        public void setUp() {
            bussiness = new Bussiness();
            mock = EasyMock.createMock(Dao.class);// interface not class
            bussiness.setDao(mock);
        }

        public void testDoSomeThing() throws SQLException {
            EasyMock.expect(mock.getEntityById(3)).andReturn(true);
            EasyMock.replay(mock);
            Assert.assertTrue(bussiness.doSomeThing(3));
        }
    }

Итак, аппарат Тесс работает правильно

Но когда я хочу запустить метод main в Business Object:

public static void main(String[] args) throws SQLException {
            Bussiness b = new Bussiness();
            b.doSomeThing(3);
}

Мне нужно добавить конструктор для бизнеса.

public Bussiness() {
     dao = new DaoImpl();
}

Итак, мой бизнес-класс:

package tunl;

import java.sql.SQLException;

public class Bussiness {
    private Dao dao;

    public Bussiness() {
        dao = new DaoImpl();
    }
    public Dao getDao() {
        return dao;
    }

    public void setDao(Dao dao) {
        this.dao = dao;
    }

    public boolean doSomeThing(int id) throws SQLException {
        if(dao.getEntityById(id)) {
            return true;
        } else {
            return false;
        }
    }

    public static void main(String[] args) throws SQLException {
        Bussiness b = new Bussiness();
        b.doSomeThing(3);
    }
}

Также я должен реализовать интерфейс Dao:

package tunl;

import java.sql.SQLException;

public class DaoImpl implements Dao {

    @Override
    public boolean getEntityById(int id) throws SQLException {
        if(id == 3) {
            System.out.println("System input 3 ");
            return true;
        }
        System.out.println("You have to input  3 ");
        return false;
    }

}

В дизайне вы всегда создаете интерфейс для всех классов, которые будут тестироваться (например, DaoImpl) !!! Так это правильно?

1 Ответ

3 голосов
/ 11 мая 2010

EasyMock создает макет объекта на основе интерфейса. Объект mock реализует все методы интерфейса и для тех методов, которые вы укажете (например, с expect), он «воспроизводит» указанное поведение при их вызове.

Когда создается фиктивный объект, он находится в режиме записи . Линия

EasyMock.expect(mock.getRate("USD", "EUR")).andReturn(1.5);

указывает, что когда mock.getRate вызывается с данными параметрами, он должен возвращать 1,5. Затем объект переводится в режим воспроизведения с вызовом

EasyMock.replay(mock);

Все это более подробно объясняется в документации .

Обновление: к вашему комментарию - Currency передается экземпляр ExchangeRate здесь:

public Currency toEuros(ExchangeRate converter) { ... }

Все, что его волнует, это то, что он получает объект, реализующий этот интерфейс, так что

rate = converter.getRate(units, "EUR");

можно назвать. Затем тестовый метод передает созданный имитационный объект объекту валюты:

Currency actual = testObject.toEuros(mock);

Надеюсь, это поможет; если нет, возможно, вы могли бы прочитать некоторый вводный текст по ООП, интерфейсам и наследованию, чтобы лучше понять.

В примере кода в вашем ответе объект Dao должен быть передан в Bussiness, а не создан внутри, поскольку последний эффективно предотвращает модульное тестирование.

public static void main(String[] args) throws SQLException {
        Bussiness b = new Bussiness();
        b.setDao(new DaoImpl());
        b.doSomeThing(3);
}

Вы также можете добавить параметризованный конструктор к Bussiness, чтобы выполнить инициализацию за один шаг вместо двух.

...