Thinking in Java——笔记(8)
Polymorphism
- The polymorphic method call allows one type to express its distinction from another, similar type, as long as they’re both derived from the same base type.
Upcasting revisited
- Taking an object reference and treating it as a reference to its base type is called upcasting.
Forgetting the object type
- The compiler won’t give you any error messages if you forget to overload one of your methods and the whole process of working with types becomes unmanageable.
- Wouldn’t it be nice if you could forget that there are derived classes, and write your code to talk only to the base class?
The twist
Method-call binding
- Connecting a method call to a method body is called binding.
- When binding is performed before the program is run (by the compiler and linker, if there is one), it’s called early binding.
- late binding means that the binding occurs at run time, based on the type of object.
- There must be some mechanism to determine the type of the object at run time and to call the appropriate method.
- All method binding in Java uses late binding unless the method is static or final(private methods are implicitly final).
- Declaring a method final effectively “turns off” dynamic binding, or rather it tells the compiler that dynamic binding isn’t necessary.
Producing the right behavior
- Once you know that all method binding in Java happens polymorphically via late binding, you can write your code to talk to the base class and know that all the derived-class cases will work correctly using the same code.
Extensibility
- You can add new functionality by inheriting new data types from the common base class.
- Changes in your code don’t cause damage to parts of the program that should not be affected.
- Polymorphism is an important technique for the programmer to “separate the things that change from the things that stay the same.”
Pitfall: “overriding” private methods
- A private method is automatically final, and is also hidden from the derived class.
- Only non-private methods may be overridden.
- You should watch out for the appearance of overriding private methods, which generates no compiler warnings.
- You should use a different name from a private base-class method in your derived class.
Pitfall: fields and static methods
- Only ordinary method calls can be polymorphic.
- Any field accesses are resolved by the compiler, and are thus not polymorphic.
- If a method is static, it doesn’t behave polymorphically.
Constructors and polymorphism
- Constructors are not polymorphic (they’re actually static methods, but the static declaration is implicit).
Order of constructor calls
A constructor for the base class is always called during the construction process for a derived class.
The constructor has a special job: to see that the object is built properly.
Only the base-class constructor has the proper knowledge and access to initialize its own elements.
It’s essential that all constructors get called; otherwise the entire object wouldn’t be constructed.
It will silently call the default constructor if you don’t explicitly call a base-class constructor in the derived-class constructor body.
The order of constructor calls:
1.The base-class constructor is called.
2.Member initializers are called.
3.The body of the derived-class constructor is called.You must be able to assume that all the members of the base class are valid when you’re in the derived class.
Inside the constructor, however, you must be able to assume that all members that you use have been built.
The only way to guarantee this is for the base-class constructor to be called first. Then when you’re in the derived-class constructor, all the members you can access in the base class have been initialized.
Whenever possible, you should initialize all member objects at their point of definition in the class.
Inheritance and cleanup
- Most of the time you won’t have to worry about cleaning up.
- The order of disposal should be the reverse of the order of initialization, in case one subobject is dependent on another.
- This technique(reference counting) requires extra diligence to use, but if you are sharing objects that require cleanup you don’t have much choice.
Behavior of polymorphic methods inside constructors
- If you call a dynamically-bound method inside a constructor, the overridden definition for that method is used.
- The effect of this call can be rather unexpected because the overridden method will be called before the object is fully constructed.
- If the constructor is only one step in building an object of a class that’s been derived from that constructor’s class, the derived parts have not yet been initialized at the time that the current constructor is being called.
- In actual initialization, the storage allocated for the object is initialized to binary zero before anything else happens.
- Do as little as possible to set the object into a good state, and if you can possibly avoid it, don’t call any other methods in this class.
- The only safe methods to call inside a constructor are those that are final in the base class.
Covariant return types
- An overridden method in a derived class can return a type derived from the type returned by the base-class method.
Designing with inheritance
- It’s possible to dynamically choose a type (and thus behavior) when using composition, whereas inheritance requires an exact type to be known at compile time.
- You can’t decide to inherit differently at run time; that must be completely determined at compile time.
- Use inheritance to express differences in behavior, and fields to express variations in state.
Substitution vs. extension
- pure substitution: derived class objects can be perfectly substituted for the base class.
- The base class can receive any message you can send to the derived class because the two have exactly the same interface.
- “is-like-a” relationship: the derived class is like the base class—it has the same fundamental interface—but it
has other features that require additional methods to implement. - The extended part of the interface in the derived class is not available from the base class, so once you upcast, you can’t call the new methods.
Downcasting and runtime type information
- With a downcast, you don’t really know that a shape (for example) is actually a circle.
- In Java, every cast is checked!
- This act of checking types at run time is called runtime type identification (RTTI).
- You can try to downcast. If it’s the correct type, it will be successful. Otherwise, you’ll get a ClassCastException.
Thinking in Java——笔记(8)的更多相关文章
- Effective Java笔记一 创建和销毁对象
Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...
- java笔记00-目录
--2013年7月26日17:49:59 学习java已久,趁最近有空,写一个总结: java笔记01-反射:
- java笔记整理
Java 笔记整理 包含内容 Unix Java 基础, 数据库(Oracle jdbc Hibernate pl/sql), web, JSP, Struts, Ajax Spring, E ...
- 转 Java笔记:Java内存模型
Java笔记:Java内存模型 2014.04.09 | Comments 1. 基本概念 <深入理解Java内存模型>详细讲解了java的内存模型,这里对其中的一些基本概念做个简单的笔记 ...
- servlet(6) - servlet总结 - 小易Java笔记
垂阅前必看: 这都是我总结的我觉得是学习servlet应该掌握的,我在学习期间也做了一个博客项目来让所学的知识得以巩固.下面就是博客项目链接.前面的servlet相关的笔记总汇,还有就是我把觉得在学习 ...
- Java笔记 —— 继承
Java笔记 -- 继承 h2{ color: #4ABCDE; } a{ text-decoration: none!important; } a:hover{ color: red !import ...
- Java笔记 —— 方法重载和方法重写
Java笔记 -- 方法重载和方法重写 h2{ color: #4ABCDE; } a{ text-decoration: none !important; } a:hover{ color: red ...
- Java笔记 —— 初始化
Java笔记 -- 初始化 h2{ color: #4ABCDE; } a{ text-decoration: none !important; } a:hover{ color: red !impo ...
- Java笔记 —— this 关键字
Java笔记 -- this 关键字 h2{ color: #4ABCDE; } a{ color: blue; text-decoration: none; } a:hover{ color: re ...
- Java 笔记 —— java 和 javac
Java 笔记 -- java 和 javac h2{ color: #4ABCDE; } a{ text-decoration: none !important; } a:hover{ color: ...
随机推荐
- 【转】linux network namespace 学习
原文地址:https://segmentfault.com/a/1190000004059167 介绍 在专业的网络世界中,经常使用到Virtual Routing and Forwarding(VR ...
- 矩阵快速幂 ZOJ 3497 Mistwald
题目传送门 题意:看似给了一个迷宫,每个点能传送到4个地方,问在P时间能否到达终点 分析:其实是一个有向图,可以用邻接矩阵存图,连乘P次看是否能从1到n*m,和floyd的传递背包思想一样 #incl ...
- iOS instancetype or id ?
The id type simply says a method will return a reference to an object. It could be any object of any ...
- BZOJ2758 : [SCOI2012]Blinker的噩梦
首先将包含关系建树. 方法是将每个图形拆成上半边和下半边,从左往右扫描线,用Splay从下到上维护扫描线上所有图形. 每次加入一个新的图形$x$的时候,看看它下方第一个图形$y$,如果$y$是上半边, ...
- BZOJ3569 : DZY Loves Chinese II
这回是真·强制在线了,首先这道题就是AHOI2013连通图的加强版,那道题k最大只有4 那道题的做法是: 取一个生成树,对每条非树边取一个随机权值, 对每条树边设为“覆盖它的所有非树边”的权值的xor ...
- BZOJ 1355 & KMP
BZOJ放这种丝帛我也是醉了... 不过来填一下求最小循环节的坑... 以这道题为例,相同文本串粘起来的串中取一小节,可以把任意一个字符看做文本串头. 那么我们一次KMP求出next函数然后显见,最后 ...
- Leetcode Word Break
Given a string s and a dictionary of words dict, determine if s can be segmented into a space-separa ...
- 用存储过程 将大段的SQL藏起来
在日常工作中,当面对比较复杂的数据库操作时不免要写一些比较长的SQL,由于某系SQL有些长(目前我写的最长的貌似有30多行吧),这时候长会面临这个 方法 优点 缺点 用"+"串 ...
- JAVA_用Java来获取访问者真实的IP地址
在jsp里,获取客户端的ip地址的方法是:request.getRemoteAddr(),这种方法在大部分情况下都是有效的.但是在通过了Apache,Squid等反向代理软件就不能获取到客户端的真实I ...
- Delphi中对BCD码的直接支持 (转)
最近在Delphi下写软件,需要将数据转换为BCD码和将BCD码转换为其它数据类型,从网上搜索了一下,没有发现好的函数,于是就想自定义函数来完成BCD与其它格式的数据转换功能.但最终没有动手写,先查查 ...