Java面向对象编程是围绕class、new、this、super展开的具体机制;new强制绑定构造方法,this/super有严格作用域,多态需继承关系、父类引用指向子类对象、调用重写实例方法三者缺一不可。
Java面向对象编程不是一套抽象理论,而是围绕 class、new、this、super 这几个关键词展开的一系列具体约束和协作机制。理解不到位,写出来的代码就容易出现 NullPointerException、方法调用错乱、子类无法复用父类逻辑等问题。
new 后必须跟构造方法?Java 中没有“直接创建对象”的语法糖;new 操作符强制绑定一个构造方法(哪怕是编译器自动生成的无参构造)。这决定了对象初始化的入口是明确且唯一的。
常见错误现象:
this(...) 或 super(...) 时位置不对(必须是第一行)实操建议:
public MyClass() {}
init() 或使用工厂方法overridable method call in constructor)this 和 super 的实际作用域边界this 不只是“当前对象引用”,它还承担字段/方法消歧义、链式构造调用、作为参数传递等角色;super 则严格限定为访问父类中被覆盖(@Override)或隐藏(字段同名)的成员。
关键区别:
this.field 访问的是当前类声明的字段(即使子类有同名字段也不会向上查找)super.method() 调用的是父类版本的方法,但该方法内部若又调用了 this.xxx(),仍会触发子类重写版本(动态绑定不变)super() 只能在构造方法首行调用,且每个构造方法最多调用一次典型陷阱:
class Parent {
String name = "parent";
void print() { System.out.println(name); }
}
class Child extends Parent {
String name = "child";
void print() { System.out.println(name); }
void test() {
System.out.println(this.name); // 输出 "child"
System.out.println(super.name); // 输出 "parent"
super.print(); // 输出 "child"(因为 print() 内部的 this.name 指向 Child 实例)
}
}
Java 多态不是“写了 extends 就自动生效”,它依赖编译期类型(引用类型)和运行期类型(实际 new 的类型)分离这一机制。缺一不可。
必须同时满足:
class A extends B 或 class C implements D)B obj = new A();)@Override)的实例方法(非 static、非 private、非构造方法)注意:
static 方法看编译期类型(B.staticMethod() 永远调用 B 中的版本)obj.field 总是取编译期类型的字段)List 和 List 在运行时都是 List,无法靠类型参数实现多态分发private”,而是控制变更影响范围把字段设为 private 只是手段,真正目标是让外部依赖不随内部实现细节变化而失效。很多团队误以为加了 getter/setter 就算封装好了,其实不然。
容易被忽略的点:
return this.items;),外部可随意修改,破坏封装setAge(-5)),等于把校验责任推给所有调用方Calendar、ArrayList)却不做防御性拷贝改进做法:
Collections.unmodifiableList(...) 或新副本IllegalArgumentException
真正难的从来不是记住“四大特性”名词,而是每次写 new、每次加 private、每次重写 toString() 时,脑子里是否清楚这个动作在内存布局、调用链路、生命周期上带来了什么连锁反应。