Достъп до вложени колекции със скорост

Имам този шаблон за скорост:

<userTestData>

#foreach( $user in $userList )
<user>
    <imsi>$user.imsi</imsi>
    #foreach ( $subscriptionToGroupMap in $user.subscriptionToGroupsList )
    <subscriptionToGroup>
        #foreach ( $key in subscriptionToGroupMap.keySet() )
        <$key>$subscriptionToGroupMap.get($key)</$key>
        #end
    </subscriptionToGroup>
    #end
</user>

#end
</userTestData>

Но когато го опитвам, получавам следното съобщение за грешка:

16:23:58,872 ERROR [org.jboss.as.ejb3.invocation] (http-/127.0.0.1:8080-1) JBAS014134: EJB Invocation failed on component ... for method public void ...) ...: javax.ejb.EJBException: org.apache.velocity.exception.ParseErrorException: Invalid arg #2 in directive #foreach at userTestDataTemplate.vm[line 10, column 21]

Което основно сочи към този ред:

#foreach ( $subscriptionToGroupMap in $user.subscriptionToGroupsList )

Останалата част от кода изглежда така:

ArrayList userList = new ArrayList();

    for(...) {
        Map user = new HashMap();
        user.put("imsi", i);

        ArrayList subscriptionToGroupsList = new ArrayList();

        Random rnd = new Random();
        int numberOfSubscriptionsToGroups = (rnd.nextInt(1000) % (BucketType.values().length)) + 1;
        LOG.log(Level.INFO, "...: numberOfSubscriptionsToGroups {0}", numberOfSubscriptionsToGroups);
        for(int j = 0; j < numberOfSubscriptionsToGroups; j++) {
            Map subscriptionToGroupMap = new HashMap();

            subscriptionToGroupMap.put("name", ...);

            subscriptionToGroupMap.put("priority", rnd.nextInt(Integer.MAX_VALUE) % (Integer.MAX_VALUE));

            ...
            Date startDate = new Date(dateGeneratorHelper);
            String stringFormattedStartDate = new SimpleDateFormat("dd-MM-yyyy'T'HH:mm:ss").format(startDate);
            subscriptionToGroupMap.put("startDate", stringFormattedStartDate);

            Date endDate = new Date(dateGeneratorHelper + dateGeneratorHelperHelper + 1);
            String stringFormattedEndDate = new SimpleDateFormat("dd-MM-yyyy'T'HH:mm:ss").format(endDate);
            subscriptionToGroupMap.put("endDate", stringFormattedEndDate);

            subscriptionToGroupsList.add(subscriptionToGroupMap);                
        }

        user.put("subscriptionToGroupsList", subscriptionToGroupsList);

        userList.add(user);           
    }

    String debugOutput = null;
    for (Iterator it = userList.iterator(); it.hasNext();) {
        HashMap userMap = (HashMap)it.next();
        debugOutput += "\n" + userMap.get("imsi") + "\n";
        ArrayList stgsList = (ArrayList)userMap.get("subscriptionToGroupsList");
        for (Iterator jt = stgsList.iterator(); jt.hasNext();) {
            Map stgMap = (HashMap)jt.next();
            for (Iterator nt = stgMap.entrySet().iterator(); nt.hasNext();) {
                Map.Entry<String, ArrayList> entry = (Map.Entry<String, ArrayList>)nt.next();
                debugOutput += "\t<" + entry.getKey() + ">" + entry.getValue() + "</" + entry.getKey() + ">\n";
            }
        }
    }

    ServletContext servletContext = (ServletContext)webServiceContext.getMessageContext().get(MessageContext.SERVLET_CONTEXT);
    VelocityEngine ve = new VelocityEngine();
    ve.setApplicationAttribute("javax.servlet.ServletContext", servletContext); 
    ve.setProperty("resource.loader", "webapp"); 
    ve.setProperty("webapp.resource.loader.class", "org.apache.velocity.tools.view.servlet.WebappLoader"); 
    ve.setProperty("webapp.resource.loader.path", "/WEB-INF/classes"); 
    VelocityContext velocityContext = new VelocityContext();
    velocityContext.put("userList", userList);
    Template t = ve.getTemplate("userTestDataTemplate.vm");
    StringWriter writer;
    writer = new StringWriter();
    t.merge(velocityContext, writer);

Получавам правилния изход за отстраняване на грешки от частта Java код, така че информацията е там.

Имате ли представа защо velocity не приема достъп до списъка в $user Map?


person soltzu    schedule 22.08.2014    source източник
comment
Правописна грешка ли е или липсва $ в #foreach ( $key in subscriptionToGroupMap.keySet() ) (трябва да е $subscriptionToGroupMap.keySet())?   -  person Serge Ballesta    schedule 22.08.2014
comment
Знам, че нямам кода със себе си, така че не мога да го проверя, това би било неудобно, но в същото време и голямо облекчение. Значи кодът изглежда добре за вас?   -  person soltzu    schedule 23.08.2014
comment
трябва да можете да използвате $user.get("subscriptionToGroupsList") така ли е?   -  person man.2067067    schedule 18.11.2016


Отговори (1)


Е, вашият контекст на скоростта съдържа един обект "userList", който е List, и вие получавате вашето $user от него: добре. $user е Map, което на свой ред съдържа списък с карти.

Така че грешката е нормална: velocity използва нотацията . за достъп до методи на обект или атрибути на bean, а не елементи на карта.

Трябва да напишете:

#foreach ( $subscriptionToGroupMap in $user["subscriptionToGroupsList"] )

и с изключение на липсващите $ до редове по-късно (вижте коментара ми по-горе), останалата част от шаблона изглежда добре (но не го тествах ...)

person Serge Ballesta    schedule 23.08.2014
comment
Благодаря! Ще го пробвам. - person soltzu; 23.08.2014
comment
Сега работи, с вашето предложение, благодаря много. Между другото, липсваше ми знакът „$“, но бих искал да мисля, че щях да поправя това, когато бях уведомен за грешката. - person soltzu; 25.08.2014