Como o Java armazena Strings e como a substring funciona internamente? [fechadas]

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);
    }
}

Aqui está a saída que recebo no java 1.6.0_27:

false
true
false
true

Alguém pode por favor explicar a saída. Eu sei que o Java diferencia entre String armazenada no heap e String armazenada em String "common pool" (que pode ser internada). Internamente, como é sua representação diferente? Como isso altera o algoritmo de substring. Por favor, cite livro / artigo / blogs, etc., sempre que apropriado.

questionAnswers(2)

yourAnswerToTheQuestion