虽然这种问题应该很难遇到,但是遇到了也会感到莫名其妙。不知道大家有没有遇到那种在java代码里用字符串写sql语句的情况,但是如果sql语句字符串的长度太长的话就会报错。

代码如下:

  • 代码A
String str = "567890123456789...0123456789";//由于字符串长度太长,所以省略一部分,长度是65535
System.out.println(str.length());   //编译报错:Error:(14, 22) java: 常量字符串过长
  • 代码B
String str = "67890123456789...0123456789";//由于字符串长度太长,所以省略一部分,长度是65534
System.out.println(str.length());   //编译通过,运行结果为:65534


  • 代码C
String str = "67890123456789...0123456789";//由于字符串长度太长,所以省略一部分,长度是65534
str = "5" + str;
System.out.println(str.length());   //编译通过,运行结果为:65535

String内部是以char数组的形式存储,数组的长度是int类型,那么String允许的最大长度就是Integer.MAX_VALUE = 2^31 - 1 = 2147483647。又由于java中的字符是以16位存储的,因此大概需要4GB的内存才能存储最大长度的字符串。 

  不过这仅仅是对字符串变量而言,如果是字符串常量,如“abc”、”1234”之类写在代码中的字符串str,那么允许的最大长度取决于字符串在常量池中的存储大小,也就是字符串在class格式文件中的存储格式:

CONSTANT_Utf8_info {
    u1 tag;
    u2 length;
    u1 bytes[length];
}

u2是无符号的16位整数,因此理论上允许的string str的最大长度是2^16-1=65535。然而实际测试表明,允许的最大长度仅为65534,超过就编译错误。