文章最初发表于我的个人博客非典型性程序猿
对于刚接触JAVA或者其他面向对象编程语言的朋友们来说,可能一开始都很难理解面向对象的概念以及类和对象的关系。笔者曾经带过一个短期培训班教授java入门基础,在最后结束课程的时候,还有很多同学不太理解面向对象的思维以及类与对象的意义。这几天有空,就想着整理整理自己的思路,谈谈自己对面向对象以及类与对象的理解。
面向对象
首先,一言不和先百度,得到如下定义:
一切事物皆对象,通过面向对象的方式,将现实世界的事物抽象成对象,现实世界中的关系抽象成类、继承,帮助人们实现对现实世界的抽象与数字建模。
我们知道,编写程序的目的是为了解决现实生活中的问题,编程的思维方式也应该贴近现实生活的思维方式。面向对象的编程方式就是为了实现上述目的二出现的。它使得编程工作更直观,更易理解。需要注意的是这里说的编程不光是coding还包括了设计的过程也是面向对象的
为什么说面向对象更贴近实际生活
想象一下,当我们向别人描述一样事物时,我们都是怎么说的?"它有像鸭子一样的嘴巴","它有4条退","爪子里还有蹼","它是哺乳动物但却是卵生"。
这种HAS A 和 IS A的表达方式往往可以简单而高效的描述一样事物。HAS A描述事物的属性或行为,IS A 则说明了事物的类属。
当我们把这一系列的属性组合起来便得到的鸭嘴兽这一类,同时哺乳动物一词简单精炼的表面了所有哺乳动物的特性而不用一一列出,这是继承特性的体现,同时卵生又是多态的体现。
这就是面向对象的思维特点,抽取(抽象)有用的属性和行为(抛弃哪些无需关系的)组织(封装)成一个类。这个过程中你也许会发现很多属性或方法是和另一个类相同的,那么你就可以采用继承的方式避免重复(当然这个过程也有可能是,当你设计完一个个类后,才发现他们有共同点,然后再抽取出基类)。更重要的是,继承是可以不原样照搬的,我们可以通过重载实现相同行为或属性的特有实现方式,这种特点称之为多态,例如同样的生产行为,实现方式由胎生变为卵生。请大声念出,并牢牢记住面向对象的四个特征:
- 抽象
- 封装
- 继承
- 多态
与早期结构化编程相比
早期结构化编程是面向过程的(功能),换句话说程序是由功能的集合组成,而调用者是作为功能的参数传入的。而在面向对象的程序中,对象是主体,程序是由对象的集合组成。一个对象中包含一系列符合设计的功能供其他对象调用。这么说可能还是比较抽象,
例如当我们设计一个五子棋游戏时,面向过程的设计思路就是首先分析问题的步骤:
1、开始游戏,2、黑子先走,3、绘制画面,4、判断输赢,5、轮到白子,6、绘制画面,7、判断输赢,8、返回步骤2,9、输出最后结果。
把上面每个步骤用分别的函数来实现,问题就解决了。
而面向对象的设计则是从另外的思路来解决问题。整个五子棋可以分为:
1、黑白双方,这两方的行为是一模一样的,2、棋盘系统,负责绘制画面,3、规则系统,负责判定诸如犯规、输赢等。
第一类对象(玩家对象)负责接受用户输入,并告知第二类对象(棋盘对象)棋子布局的变化,棋盘对象接收到了棋子的变化就要负责在屏幕上面显示出这种变化,同时利用第三类对象(规则系统)来对棋局进行判定。
(以上例子来自国内著名问答社区)
随便写点代码,大家看看就好,不要太认真....
- /**
- 玩家类
- **/
- public class Player {
- String name; //棋手名称
- boolean isFirst; //是否先手
- int color_flag; //代表颜色 0-白 1-黑
- Table table;//棋盘对象
- public Player(String name,boolean isFirst;int color_flag){
- this.name=name;
- this.isFirst=isFirst;
- this.color_flag=color_flag;
- }
- /**
- 下棋,x,y为落子坐标
- **/
- public void play(int x,int y) throws Exception{
- if(this.table==null){
- throw new IllegalArgumentException("玩家还未注册到棋盘!");
- }
- table.setNewPieces(x,y);
- }
- public void setTable(Table table){
- this.table=table;
- }
- }
- /**
- 棋盘类
- **/
- public class Table{
- List<Player> playerList=new ArrayList<Player>();
- Referee referee ;
- public Table(){
- referee =new Referee(this);
- }
- /**
- 注册玩家
- **/
- public void registPlayer(Player player) throws Exception {
- //检测棋盘中的玩家是否已满,先手玩家和玩家选色是否冲突。
- .......
- playerList.add(player);
- player.setTable(this);
- }
- /**
- 落子
- **/
- public void setNewPieces(int x , int y){
- //重新绘制棋盘
- ......
- //调用裁判对象,判断结果
- if(referee.isEnd){
- End();
- }
- }
- public void End(){
- .......
- }
- }
- /**
- 裁判类
- **/
- public class Referee(){
- Table table;
- public Referee(Table table){
- this.table=table;
- }
- public boolen isEnd(){
- //判断输赢
- ....
- return false;
- }
- }
然而事实上,通过上述示例代码,我们不难发现,即使我们使用面向对象的方式,上面例子里面向过程中提到的几个下棋过程我们还是都实现了的,只不过程被封装到了类的方法中。所以说其实面向对象和面向过程并不是编程的区别(需要实现的业务逻辑的量不会产生变化),而是设计的区别!
类与对象
类是抽象的,而对象是具体的
如何理解上面的话呢? 例如鸭嘴兽是类型,具体的鸭嘴兽A、鸭嘴兽B就是对象了。在JAVA中对象是通过new关键字声明的。 再例如,《红色警戒》中美国大兵是一类兵种,点击制造后从兵营里出来的那个会开枪的家伙就是对象了:
类的定义就是一个模板,它描述的一类对象的属性与行为。类往往是抽象的、没有实体的。哺乳动物是类的概念,是抽象的,现实中没有哺乳动物这一实体,只有具体的如老虎,狮子等。编程工作中套用这一思维模式,我们将程序中的实例抽象为类,例如一个系统中的用户有张三、李四我们会把他们抽象为Person类,或者称之为一个名为Person的数据类型。
对象则是根据所属类模板创造出来的实实在在的事物。在程序中我将这个实实在在的事物称之为实例,我们为它的属性赋上特定的值,让它成为张三或者李四。在内存里来说,对象是表示的就是具体数据。
前面说的都是概念性的东西,下面我们说说实际的运用过程中的理解。
从数据类型来说
以java为例,数据类型分为基本数据类型和引用数据类型。
基本数据类型就是byte,short,int,long,double,char,boolean;其它的,需要用到new关键字来赋值的都是引用数据类型。 类与对象指的便是引用数据的类型与其值(这里指的类不光是class,还包括接口、数组、枚举、注解)。 而引用指的是内存地址的引用,关于这点在后面说的内存时会细说。
看下面的代码:
- int a =1;
- Person b=new Person();
a 和 b 都是本身无意义的变量名。需要关注的是:a的类型是基本数据类型int值为1,而b的类型是Person属于引用类型,其引用的是new Person()这个对象。我们往往会说对象xx,比如这里的对象b。但实际上b只是对象的引用,真正的对象是后面的new Person()!
需要注意的是String也是引用数据类型,只不过因为使用率非常高,所以对于String,jvm支持其可 以像基本数据类型一样使用:String a = "abc"; 同等于 String a = new String("abc");
总之呢,简单来说类指的的引用数据的类型,对象是具体赋的值。为了更深入理解,我们下面需要解释下这个引用是如何体现的。
什么是引用(从内存来说)
要深入理解什么是类,什么是对象,什么又是引用,就离不开说说java的内存使用方式。
在java中内存被大致划分为栈(stack)与堆(heap) (之所以是大致,是因为还包括其它几部分就不在这细说)。
关于什么是栈与堆在这就不细说,有空我再整理一篇文章详细说明。
在这里我们只说一点:java中,基本数据类型以及对象的引用都保存在栈(stack),而对象则保存在堆(heap)中,例如当如下代码:
- int a=1;
- Person p;
内存中的状态大致如下:
int a = 1 是直接在栈中开辟空间,而对于未进行实例化的Person p因为没有有效的内存地址引用它的值是null。而当代码进行如下修改时:
- int a =1 ;
- Person p = new Person();
内存中的状态大致如下:
Person p=new Person();使得p的值=0x8da23也就是对象new Person();在堆中的地址。所以,到这里后就不难理解之前说的对象的引用了,所谓引用其实就是堆内存地址的引用。
总结
随着计算机技术的不断提高,现在计算机不单单是用来解决运算问题,而是被用于解决越来越贴近现实生活的复杂问题。面向对象就是这一发展进程的产物,它使得编程工作更贴近人的思维方式,从而大大提升编程效率。我们必须明白的是面向对象并不是一种编程方式,而是一种编程思维方式,这种思维方式涵盖了分析,设计,编码等。在面向对象编程中,程序的基本单元是对象,数据封装在对象中。类是对象模板,是预定义好的结构,所谓的实例化一个类,所指的就是将数据填入模板。
最后,本人文笔不是很好,有待提高。写文章和博客的最大目的是梳理自己的思路,其二是分享自己的想法,望大家多多吐槽,愿共同提高。