0

我正在尝试编写代码以使用 FTL 生成 JVM 线程转储,但在遍历 TreeMap 时遇到问题。

这是我的动作类的片段:

- - -开始 - - -

public AppsnapAction() {
    traces = new TreeMap<Thread, StackTraceElement[]>(COMP);
    traces.putAll(Thread.getAllStackTraces());
}

有问题的方法。

public Map<Thread, StackTraceElement[]> getAppTraces() {
 //  log.debug("HelloMe..in getAppTraces"+ traces);
    return traces;
}

- - 结尾 - - - -

这是 FTL 的片段:

- - -开始 - - -

<h1>My Trace</h1>
<#list appTraces?keys as trace>
<h4><a name="${trace_index}">${trace}</a></h4>
<pre>
<#list appTraces[trace] as traceline>
   at ${traceline}
</#list>
</pre>
    </#list>

- - - 结尾 - - - -

代码在这一行中断:“<#list appTraces[trace] as traceline>”

除了以下例外:

2013-09-15 22:58:35,701 [http-8080-1] 错误 freemarker.runtime -

get(Thread[ContainerBackgroundProcessor[StandardEngine[Catalina]],5,main]) 在 java.util.TreeMap 实例上失败 有问题的指令:


==> 将 appTraces[trace] 列为跟踪线 [在 /resources/templates/td-Page.ftl 中的第 42 行第 5 列]


程序员的 Java 回溯:


freemarker.template.TemplateModelException: get(Thread[ContainerBackgroundProcessor[StandardEngine[Catalina]],5,main]) failed on instance of java.util.TreeMap
at freemarker.ext.beans.BeanModel.get(BeanModel.java:223)
at freemarker.core.DynamicKeyName.dealWithStringKey(DynamicKeyName.java:136)
at freemarker.core.DynamicKeyName._getAsTemplateModel(DynamicKeyName.java:94)
at freemarker.core.Expression.getAsTemplateModel(Expression.java:89)
at freemarker.core.IteratorBlock.accept(IteratorBlock.java:94)
at freemarker.core.Environment.visit(Environment.java:213)
Caused by: java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Thread
at com.jivesoftware.helloworld.action.AppsnapAction$1.compare(AppsnapAction.java:1)
at java.util.TreeMap.getEntryUsingComparator(TreeMap.java:351)
at java.util.TreeMap.getEntry(TreeMap.java:322)
at java.util.TreeMap.get(TreeMap.java:255)
at freemarker.ext.beans.MapModel.invokeGenericGet(MapModel.java:127)
at freemarker.ext.beans.BeanModel.get(BeanModel.java:185)
... 202 more

请对如何遍历内部循环有任何想法吗?

4

1 回答 1

1

您刚刚遇到了 FreeMarker 的最大烦恼之一:[]运算符不支持非字符串键,至少在 FreeMarker 2.4 之前是这样。因此,在此之前,最简单的解决方法可能是将对象包装器更改为BeansWrappersimpleMapWrapper打开的对象包装器。像, new BeansWrapper(); bw.setSimpleMapWrapper(true); Configuration.setObjectWrapper(bw);。这个包装器提供了这个技巧:appTraces(trace). myMap[stringKey]并且myMap.foo仍然可以工作。不利的一面是,如果您已经有很多模板,其中一些可能会损坏,因为数据模型现在看起来会略有不同。或者有很多不同,具体取决于您现在使用的对象包装器......无论如何,如果您来自默认对象包装器,它通常不会破坏任何东西,除非您在数据模型中有 XML。

于 2013-09-16T09:34:16.530 回答