【Java编程思想笔记】第一章——对象导论

第一章 对象导论

抽象过程:建立解空间(问题建模) 和问题空间 (问题存在) 关联

  1. 万物皆对象
  2. 程序是对象的集合,通过发送消息来告知彼此要做的
  3. 每个对象都有自己的由其他对象构成的存储
  4. 所有对象都有类型
  5. 某一特定类型的所有对象可以接受相同的消息

对象具有状态、行为、表示

  • 每个对象都有一个接口:挑战问题空间的元素和加空间的对象建立一一的映射
  • 每个对象只能满足某些请求,这些请求由接口定义,决定接口的是类型,接口决定了对某一特定对象可以发出的请求,与满足请求的代码还有隐藏数据构成了实现
  • 每个对象都提供服务:对象是服务提供者,目标是创建提供理想的服务来解决问题的一些列对象
  • 被隐藏的具体实现:访问控制让客户端和服务端可以更专注于自己的领域
    • public任何元素都可以访问
    • protectde继承的类可以访问protected成员
    • private只有类型创建者和类型内部方法可以访问
作用域 自己 包内 子孙 其他
private
friendly
protected
public

封装
被隐藏(也即封装)的部分通常代表对象内部脆弱的部分,它们很容易被程序员所毁坏,因此将实现隐藏起来可以减少程序的bug。

隐藏是通过访问控制修饰符(public、protected、包访问、private)来实现的。

访问控制的第一个存在原因就是让调用者无法直接触及他们不应该触及的部分,但从另一方面来看,其实这不失为一项服务,因为他们可以很容易地看出哪些东西对他们来说很重要,而哪些东西可能不关心;访问控制的第二个存在原因就是允许库设计者可以改变类的内部的工作方式而不用担心会影响到调用者。

继承
代码复用:复用是面向对象程序设计所提供最了不起的优点之一。

  • 最简单的代码复用就是直接调用类的方法,此外,我们还可以将该类置于某个新类中,使它成为新类的属性成员。新的类也可由任意数量、任意类型的其他对象以任意可以实现新的类中想要功能的方式所组成,这种使用现有的类合成新的类方式称为组合复用。

  • 组合复用带来了极大的灵活性,使得它能在运行时动态的修改其实现行为,但继承并不具备这样的灵活性,因为继承是在编译时期就已经确定其行为,在运行时期是不能修改的。

  • 继承两种实现方式,第一种方式非常直接:直接在子类中添加新的方法,即扩展父类接口。第二种方式就是子类覆写父类方法,但不新增父类没有接口。

  • “is-a是一个”与“is-like-a像是一个”。继承时,我们使用第一种还是第二种方式呢?这可能引起争论:继承应该只覆盖基类的方法,而并不添加在基类中没有的新方法吗?如果这样做,就意味着子类与基类是完全相同的类型,因为它们具有完全相同的接口,结果可以用一个子类对象来完全替代一个基类对象,这可被认为是纯粹替代,通常称之为替代原则,这也是一种理想的方式,我们经常称这种情况下的子类与基类的关系是“is-a是一个”;有时必须在子类型中添加新的接口,这样也就扩展了接口,这个新的类型仍可以替代基类,但是这种替代并不完美,因为基类无法访问新添加的方法,这种情况下我们可以描述为“is-like-a像是一个”关系。

多态
一个非面向对象编程的编译器产生的函数调用会引起所谓的前期绑定,而向对象程序设计语言使用了后期绑定的概念。

  • 方法的调用就是编译器将产生对一个具体函数名字的调用,前期绑定是在运行时将这个调用解析到将要被执行的代码的绝对地址。然而在OOP中,程序直到运行时才能够确定代码的地址,为了执行后期绑定,Java编译器使用了一小段特殊代码来替代绝对地址调用,这段代码使用对象中存储的信息来计算方法体的地址,根据这段特殊代码,每个对象都可以具有不同的行为表现。

  • 在某些语言中,必须明确地声明某个方法具备后期绑定属性所带来的灵活性,如C++使用virtual关键字来实现,在默认情况下,C++不是动态绑定的,而在Java中,动态绑定是默认行为,不需要添加额外的关键字来实现多态。

Java语言支持四种类型

  • 接口(interface)、类(class)、数组(array)和基本类型(primitive)。前三种类型通常被称为引用类型(reference type),类实例和数组是对象(object),而基本类型的值则不是对象。类的成员(member)由它的域(field)、方法(method)、成员类(member class)和成员接口(member interface)组成。方法签名(signature)由它的名称和所有参数类型组成;签名不包括它的返回类型。

运行Java

  • 用javac命令编译一个打包的类时,如果没有加参数”-d”时,则编译出的类不会放在包中(即相应的文件夹中),是没有包路径的,除非用参数”-d”指定类存放的位置,–d 指示的是编译后的class文件放在哪个目录下,并且会自动创建包名文件夹。
  • 比如现有如下类:
package a.b;
class A{}
  • javac A.java 时会在当前工作目录下产生一个A.class文件,不会创建包目录结构。
    javac –d . A.java ,则会在当前工作目录产生 a/b/A.class 目录与文件。
    所以使用javac编译时要想产生相应的类包名上当结构,则需要带上“–d .”这样的参数。

  • 用java命令运行一个类时,如果该类是存放在包中的,则运行时一定要带上包名,并且在环境变量要有该包存放的路径。
      java -classpath . a.A
      注,如果用java命令运行时,没有配置classpath环境变量,则这里的classpath不能缺少。

类与类之间的关系
类和类、类和接口、接口和接口之间有如下几种关系:泛化关系、实现关系、关联关系(聚合、合成)、依赖关系。

  • 泛化:表示类与类之间的继承关系,使用extends关键字来表示。在图形上使用虚线三角形箭头表示。
  • 实现:表示类与接口之间的实现关系,使用implements关键字来表示。在图形上使用实线三角形箭头表示。
  • 关联:类与类之间的联接。关联可以是双向的,也可以是单向的,双向的关联可以有两个箭头或都没有箭头。单向的关联有一个箭头,表示关联的方向。在Java里,关联关系是使用实例变量实现的。在每一个关联的端点,还可以有一个基数,表时这一端的类可以有几个实例。常见的基数有:0..1(零个或者一个实例)、0..或者(没限制,可以是零)、1(只有一个实例)、1..*(至少有一个实例)。一个关联关系可以进一步确定为聚合与合成关系。在图形上使用实线的箭头表示。
  • 聚合:是关联关系的一种,是强的关联关系,聚合是整体和个体之间的关系。关联与聚合仅仅从Java语法是上是分辨不出的,需要考察所涉及的类之间的逻辑关系。如果不确定是聚合关系,可以将之设置为关联关系。图形是在实线的箭头的尾部多一个空心菱形。
  • 合成:是关联关系的一种,是比聚合关系强的关系。它要求普通的聚合关系中代表整体的对象负责代表部分的对象的生命周期。整体消亡,则部分与消亡。图形是在实线的箭头的尾部多一个黑心菱形。
  • 依赖:类与类之间的连接,依赖总是单向的。表示一个类依赖于另一个类的定义。一般而言,依赖关系在Java里体现为局部变量、方法的参数、以及对静态方法的调用。但如果对方出现在实例变量中,那关系就超过了依赖关系,而成了关联关系了。在图形上使用虚线的箭头表示。

你可能感兴趣的:(java学习,java编程思想笔记)