¿Cómo almacena Java las cadenas y cómo funciona la subcadena internamente? [cerrado]

class StringTesting {
    public static void main(String args[])
    {
        String str = "abcd";
        String str1 = new String("abcd");
        String str2 = str.substring(0,2);
        String str3 = str.substring(0,2);
        String str4 = str.substring(0,str.length());
        String str5 = str1.substring(0,2);
        String str6 = str1.substring(0,2);
        String str7 = str1.substring(0,str1.length());

        System.out.println(str2 == str3);
        System.out.println(str == str4);
        System.out.println(str5 == str6);
        System.out.println(str1 == str7);
    }
}

Aquí está la salida que obtengo en java 1.6.0_27:

false
true
false
true

¿Puede alguien por favor explicar la salida. Sé que Java distingue entre String almacenado en montón y String almacenado en String "grupo común" (que puede ser internado). Internamente, en qué se diferencia su representación. ¿Cómo cambia el algoritmo de subcadena? Cite amablemente el libro / artículo / blogs, etc., según corresponda.

Respuestas a la pregunta(2)

Su respuesta a la pregunta