我们遇到了WebSphere 7 / IBM JDK 6的一个奇怪问题,其中一个节点有一些初始化问题。
我们有一些调用InitialContext.lookup的代码,在这个节点上我们有时会遇到以下异常:
Caused by: java.lang.ArrayIndexOutOfBoundsException
at java.lang.String.getChars(String.java:666)
at java.lang.StringBuilder.append(StringBuilder.java:207)
at javax.naming.spi.NamingManager.getURLContext(NamingManager.java:646)
at javax.naming.InitialContext.getURLOrDefaultInitCtx(InitialContext.java:422)
at javax.naming.InitialContext.lookup(InitialContext.java:436)
[...]
我们看了一下javax.naming.spi.NamingManager.getURLContext的源代码:
public static Context getURLContext(String schema, Hashtable<?, ?> envmt)
throws NamingException {
if (null == schema || 0 == schema.length() || null == envmt) {
return null;
}
// obtain pkg prefixes from hashtable
String pkgPrefixes[] = EnvironmentReader
.getFactoryNamesFromEnvironmentAndProviderResource(envmt, null,
Context.URL_PKG_PREFIXES);
for (String element : pkgPrefixes) {
// create factory instance
ObjectFactory factory;
String clsName = element + "." //$NON-NLS-1$
+ schema + "." //$NON-NLS-1$
+ schema + "URLContextFactory"; //$NON-NLS-1$
[...]
第646行是增强的for循环,但是下一个语句是字符串连接,并且可能被编译器替换为StringBuilder。
我们在StringBuilder上做了一些快速单元测试,但是没有引发ArrayIndexOutOfBoundsException。
如何抛出ArrayIndexOutOfBoundsException以及如何避免它?
修改:
我们使用以下java版本:
java version "1.6.0"
Java(TM) SE Runtime Environment (build pxa6460sr9fp2ifix-20110913_02(SR9 FP2+IV03622+IZ99243))
IBM J9 VM (build 2.4, JRE 1.6.0 IBM J9 2.4 Linux amd64-64 jvmxa6460sr9-20110912_90359 (JIT enabled, AOT enabled)
J9VM - 20110912_090359
JIT - r9_20101028_17488ifx31
GC - 20101027_AA)
JCL - 20110727_04
答案 0 :(得分:2)
这是IBM JVM的JIT编译器的已知错误。解决方法似乎是将getChars
排除在JIT编译之外:
-Xjit:exclude={ProgramClass.callStringGetChars*}
请参阅IZ78413: JIT-COMPILED STRING.GETCHARS THROWS UNEXPECTED ARRAYINDEXOUTOFBO UNDSEXCEPTION以供参考。
答案 1 :(得分:0)
如果传递空环境哈希表,上面的代码似乎会返回null
。
这让我想知道你是否用某种参数实例化InitialContext。
如果上面的代码确实是源代码,那么传递空环境哈希表会使getURLContext()在字符串连接循环之前返回null
。
如果您不需要指定JNDI环境变量(例如,使用默认的JNDI env),您可以尝试new InitialContext()
或new InitialContext(null)
吗?