繁体   English   中英

如何模拟在另一个类中使用另一个静态最终变量的变量

[英]How to mock a variable that uses another static final variable in another class

我在junit中测试中设置此方法时遇到问题。 我正在尝试测试具有新时间戳记的updateConfigDates方法。 请在这里与我保持联系,因为这里提供了很多信息,而且看起来确实不太漂亮。

这是被测试的方法。

public static Document updateConfigDates(Document doc, Timestamp configDate)
{
    //Format timestamp to string
    String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";

    //Change configuration date for all nodes in GTR
    NodeList configIDNodes = doc.getElementsByTagName("ConfigDate");
    for (Element cidNode : new DOMUtil.ElementList(configIDNodes))
    {
        cidNode.setTextContent(configTimestampStr);
    }

    return doc;
}

问题就在网上:

    //Format timestamp to string
    String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";

这是使用公共最终类常量

public final class Constants
{
    /** GTR Timestamp formatter without micro-second */
    public static final FastDateFormat GTR_DATE_FORMAT 
                           = FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ss");
}

这是我当前的测试用例

@Test
public void testUpdateConfigDates() throws Exception
{
    // check if configDate in document is expected
    String docConfigDate = doc.getElementsByTagName("ConfigDate").item(0).getTextContent();
    assertEquals(docConfigDate, "2012-02-22T16:07:27Z");

    LOG.info("docConfigDate: " + docConfigDate);

    // variables
    Timestamp newConfigDate = Timestamp.valueOf("2009-07-29 13:24:11");

    // Mocking statics
    //PowerMockito.mockStatic(GTRConstants.class);

    // String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";
    //common.setFinalStatic(GTRUtility.class.getDeclaredField("configTimestampStr"), "yyyy-MM-dd'T'HH:mm:ss" + "Z");
    common.setFinalStatic(Constants.class.getDeclaredField("GTR_DATE_FORMAT"), FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ss"));

    /* setFinalStatic ExceptionInInitializerError information below
     * 
     * Field#set(Object, Object) can be used to set static fields.
     * If you try to set the field of an uninitialized class, the JVM will first try to initialize the class.
     * If a failure occurs, then set will throw a ExceptionInInitializerError.
     */

    // execute method under test
    doc = gtrUtility.updateConfigDates(doc, newConfigDate);

    // verify expectations
    docConfigDate = doc.getElementsByTagName("ConfigDate").item(0).getTextContent();
    assertEquals(docConfigDate, "2009-07-29 13:24:11");
}

使用此处发布setFinalStatic解决方案 ,该解决方案可以成功地与我正在进行的其他测试一起使用。

public static void setFinalStatic(Field field, Object newValue) throws Exception
{
    field.setAccessible(true);
    // remove final modifier from field
    Field modifiersField = Field.class.getDeclaredField("modifiers");
    modifiersField.setAccessible(true);
    modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
    field.set(null, newValue);
}

我尝试了几种解决方案:

  • 我的最新方法,使用setFinalStatic方法
  • 使用PowerMockito和Mockito产生模拟常量。
  • 使用PowerMockito和Mockito模拟GTR_DATE_FORMAT实例。
  • 使用PowerMockito和Mockito模拟FastDateFormat。

所有这些都失败了。 我认为setFinalStatic方法是最接近的方法,但是我目前遇到了ExceptionInInitializerError错误(也从此处的用户以及此处的Oracle Doc中进行了讨论

事实证明,对我的问题的答案是,Constants文件正在执行一些我必须模拟的初始化方法(在这种情况下:对来自数据库的对象的safeGetTypeId(x))。

当意识到使用常量类GTR_DATE_FORMAT的updateObjectId硬编码测试正在运行并且updateTimestamp正常运行时,我应该注意到这一点。

构造函数是空的,我没想到会执行其他任何操作。

这是一个极端而复杂的问题,因此我可以删除此问题,但是如果将来可以帮助某个人,我将不予理会。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM