Подпись обоих классов одинакова, но реализация отличается. Для класса Object:
public boolean equals(Object obj) {
return (this == obj);
}
Для класса String реализация выглядит так:
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
Поэтому, когда вы вызываете метод equals объекта String, он вызывает его переопределенную версию метода, т.е. класс String