Search code examples
jspcustom-tags

Design pattern for jsp attributes to avoid name collision


You're making a custom tag library and have defined an attribute:

test.tag:

<%@ attribute name="variableName" required="false" type="java.lang.String" %>
${variableName}

and have a class:

Test.java

public class Test {
     public String getVariableName() {...};
     public List<Test> getTests() {...};
}

and finally a jsp:

<s:iterator value="getTests">
  <t:test />
</s:iterator>

While you have not specified a variableName in the <t:test /> tag, the tag is grabbing a value for variableName from the Test object because it happens to share the same variable name.

How do you avoid this namespace collision?


Solution

  • It seems the issue was ${variableName}. When the jsp compiled the java class uses:

    pageContext.findAttribute("variableName");
    

    According to javadocs, findAttribute "Searches for the named attribute in page, request, session (if valid), and application scopes in order and returns the value associated or null."

    Since a value for that variable is not passed in the tag, it won't be found in page scope, but it will subsequently be found in request scope as provided by my class. To limit visibility I should have been using

    ${pageScope.variableName}
    

    Uses this function instead:

    pageContext.getAttribute("variableName");
    

    According to javadocs, getAttribute "Returns the object associated with the name in the page scope or null if not found." That is the desired behavior. Use the variable only if supplied as an attribute to the tag.

    The function names are too similar for my tastes. By the name alone you wouldn't know the difference between the two.