美文网首页
Java学习之面向对象与类之概述(匿名对象、封装、构造函数、th

Java学习之面向对象与类之概述(匿名对象、封装、构造函数、th

作者: 玉圣 | 来源:发表于2017-05-27 12:01 被阅读113次

    一、概述

    1、面向对象的概述:

           java是一种面向对象的编程语言,也就是说对象是这种语言的基础,没有对象了,就没有了java。任何功能都是通过对象来实现的,就是将功能封装进对象,让对象去调用这些功能。这种思想是将数据作为第一位,而方法(功能或者说是算法)作为其次。我个人认为,这是对数据的一种优化,安全性更高,操作起数据来一更方便。
          那么将这种思想提升到一种境界就是:万物皆对象。

    1、对面向对象的理解:
    1)面向对象是相对面向过程而言的,且基于面向过程的。
    2)面向对象是一种思想。
    3)面向对象将功能封装进对象中,强调了具备功能的对象,主体是对象,将过程简化
    4)在实际开发中,以对象为核心,先明确好对象,在实现具体功能,或者可是使用已有的对象。
    5)面向对象的三个特征:封装性,继承性,多态性。

    2、举例说明:
    人是一个对象,人有吃饭,睡觉以及学习等等的行为(可称为功能),那么就是将吃饭、睡觉以及学习等功能封装进人这个的事物中,让人去执行这些功能,是人在调用这些方法,从而简化了过程。

    二、类与对象

    1、类与对象概述

    1、类(class):可以理解为是构造对象的一个蓝图或者模板,是抽象的概念;反过来说,对象是以类为模型创造的具体实例,是对类的一种具体化、形象化。
    类:对生活中事物的描述
    对象:对类的具体实现,是实实在在存在的实体。

    2、例如:汽车的设计
    类:指的是汽车的设计图纸
    对象:指实际生产出来的汽车。

    示例:

    /** 
    需求|:定义一个汽车类,要求设计出汽车的型号(即名字),颜色,轮胎个数,行驶速度,并且让汽车行驶起来(即打印出每个汽车的颜色、轮胎数、速度) 
             并比较两辆汽车之间那个性能更好|:即速度快慢 
    思路|: 
        1创建一个构造函数,即一个汽车类,对其的颜色,轮胎数。行书速度,速度,以及其方法,即行驶等进行初始化 
        2在main方法中创建一个汽车对象,并实现其功能 
        3构造一个方法,比较两辆车之间的速度大小 
    */class Car   
    {  
        String name;  
        String color;  
        int nums;  
        double speed;  
        Car(String name,String color,int nums,double speed)  
        {  
            this.name = name;  
            this.color = color;  
            nums = 4;  
            this.speed = speed;  
            System.out.println(name + "是一辆" + color + ",轮胎数是:" + nums + ",可以行驶的速度是:" + speed);  
        }  
          
        public void compare(Car c)  
        {  
            if(this.speed > c.speed)  
            {  
                System.out.println(this.name + "行驶得更快。");  
                return;  
            }  
            System.out.println(c.name + "行驶得更快。");  
            return;  
        }  
    }  
      
    class CarDemo  
    {  
        public static void main(String[] args)   
        {  
            Car a = new Car("奔驰","黑色的",4,500);  
            Car b = new Car("宝马","蓝色的",4,600);  
            a.compare(b);  
        }  
    }
    

    2、成员变量与局部变量:

    在类中的不同位置定义变量,作用范围是不同的,下面简单区分一下,两种变量的不同:

    1、作用范围:
    a.成员变量:作用于整个类中
    b.局部变量:作用于函数中,或者作用于语句块中。

    2、在内存中的位置:
    a.成员变量:在堆内存中,因为对象的存在才在内存中存在。
    b.局部变量:在栈内存中,随着函数的结束而消亡

    3、初始化方式:
    a.成员变量:随着类的初始化而初始化,在堆内存中被加载,有默认值,可直接参与运算
    b.局部变量:随着方法的加载而加载进栈内存中,无初始化值,必须被初始化才能参与运算

    上面汽车的例子中,a、b和c是局部变量,定义在了方法中,存在于栈内存中;而name、color、nums和speed都是成员变量,存在于堆内存中,随着类的加载而加载。

    3、匿名对象

    1、简述:所谓匿名对象,就是创建的对象没有名字,直接使用。

    2、使用方式:
    1)使用方式一:当对对象的方法只调用一次时,可以使用匿名对象来完成,这样写比较简化。
          如果对一个对象进行多个成员调用,必须给这个对象起个名字
    2)使用方式二:可以讲匿名对象作为实际参数进行传递,从而可以不用在main方法中创建一个变量,提高了编程效率,减少了代码书写。

          但是这个对象实体在方法结束后,垃圾回收机制会将其作为垃圾回收。而非匿名对象则不同,当不使用了,会在某一时刻被回收,或是随着主函数的结束而被回收。

    二、封装

    1、概述

    1、定义:封装是指隐藏对象的属性和实现细节,仅对外提供公共访问方式。

    2、好处:
    a.将变化隔离
    b.便于使用
    c.提高重用性
    d.提高安全性

    3、原则:
    a.将不需要对外提供的内容隐藏起来
    b.把属性都隐藏,提供公共方法对其访问

    4、说明:
    a.私有仅仅是封装的一种表现形式,如包也是一种封装形式
    b.之所以对外提供访问方式,就是因为可以在访问方式中加入逻辑判断等语句,对访问的数据进行操作,提高了代码的健壮性。

    2、private 关键字

    1、private是一个权限修饰符
    2、用于修饰成员(成员变量和成员函数)
    3、被私有化的成员只在本类中有效。
    4、常用之一:将成员变量私有化,对外提供对应的set和get 方法对其进行访问,提高了对数据访问的安全性。
    5、当把类中的所有构造函数私有化后,代表着该类是不能创建对象的,因为对象不能进行初始化操作的。

    示例:

    /**需求:定义一个private变量,并定义两个方法,一个返回其值,另一个提供新值位于60-100之间的数给变量 
    思路:定义一个类,其属性包括一个变量,以及包括两个方法 
          一个方法返回变量的值,另一个方法提供60-100之间的值给变量 
     */  
      
    class Virus  
    {  
        private int newSeconds = 0;  
        public int getSeconds()  
        {  
            return newSeconds;  
        }  
      
        public void setSeconds(int x)  
        {  
            if(x>60 || x<100)  
            {  
                newSeconds = x;  
            }  
        }  
    }  
      
    class VirusText  
    {  
        public static void main(String [] args)  
        {  
            Virus a = new Virus();  
            a.setSeconds(78);  
            System.out.println(a.getSeconds());  
        }  
    }
    

    3、成员访问权限比较

    权限大小

    成员修饰符 public protected default(默认) private
    同一个类中 OK OK OK OK
    同一个包中 OK OK OK NO
    子类访问 OK OK NO NO
    不同包中 OK NO NO NO

    4、public类与源文件名

    一个编译单元(个人理解为执行main函数所调用到的所有文件)中只能有一个public类,且这个类的文件名必须要和其类名相同,包括大小写也必须一样。

    原因:
    1、一个编译单元只能有一个public类的原因:
    第一、public的意思是所有类都能访问,包括包以外的类。public是作为这个编译单元的公开接口存在的。
    第二、java程序的入口是main方法,所以被定为public的这个类一定是main方法的类,且这个类的名称要和文件名一直,因为虚拟机是要开始找main方法这个入口的。
    第三、你可以根据需要,添加任意辅助功能的public权限的类,但是如果这个编译单元(注意是编译单元)里面有两个或以上public类的话,那么编译器就会报错。

    建议:
    第一、不要在一个源文件中写多个类。在标准的java代码编写时,无论代码量是多少,最好一个源文件只有一个类或接口(即使是接口也要如此),因为java是面向对象的语言,每个类都是抽象的结果,所以每个类要单独写在一个源文件里。
    第二、只要要有一个是public类,虽然可以在编译单元中没有public类,即没有公开的接口,可在同一个包中访问,但是这样就将这个包都封闭了,是没意义的。如果没public,就可以随意给文件起名,可以不和类名相同。

    2、被public修饰的类与文件名必须同名的原因:
    第一、java是被编译执行的,它在运行时并不是将你写的所有类都先加载一遍的,而是当遇到import或使用到了其他类的时候,才会去在文件目录中找相应的class文件的。
    第二、对于一个public类。上面也说了,是可以被项目中的任何一个类引用的,只需通过import导入即可。既然是作为虚拟机入口的main函数要用public修饰而成为一个公共接口,那么将类名和文件名一一对应就可以方便虚拟机在相应的路径(包名)中找到相关的信息;但是你如果不这么做,虚拟机很难去找,开销也会跟着增大的。

    简单总结:
    public作为一个公共接口(此接口非interface这个接口),修饰作为虚拟机入口的main函数,就是为了方便虚拟机找到相应的类,从而节省开销。

    三、对象与函数

    1、main函数:

    我们刚开始接触java的时候就是用到了main函数,那么主函数是什么呢?在此分别对主函数的各个关键字及修饰符进行简单说明:

    public static void main(String [] args){....}
    

    1、主函数:是一个特殊的函数,作为程序的入口,可以被JVM识别并调用,它的格式是固定的。

    2、主函数的定义:
    1)public:代表着该函数的访问权限是最大的。
    2)static :代表着主函数随着类的加载就已经存在了,一边执行函数中的代码。
    3)void :主函数没有具体的返回值,只是作为执行程序的入口。
    4)main :不是关键字,和其他的函数名类似,只不过别定义为一个特殊的单词,可以被JVM识别。
    5)(String [] args)):函数的参数,参数类型是一个String类数组,元素为字符串。字符串类类型是数组,args是一个变量名,是约定俗成的,早期被写成arguments;如果改写成其他名称也是可以的。

    3、注意:
    可以重载主函数,但是虚拟机只识别固定格式的主函数,即public static void main(String [] args){....}

    2、构造函数

    先看一个简单的小程序

    //创建一个简单的雇员类  
    public class Employee   
    {  
        //构造Employee函数  
        public Employee(String name,int age,double salary)  
        {  
            //将变量定义为private  
            private this.name = name;  
            private this.age = age;  
            private this.salary = salary;  
        }  
          
        {  
            System.out.println("我来工作了");  
        }  
      
        //访问器方法,获取name  
        public String getName()  
        {  
            return name;  
        }  
        //获取age  
        public int getAge()  
        {  
            //限制age的值  
            if (age<18)  
            {  
                System.out.println("不好意思啦,我们不接收未成年人!嘿嘿");  
                return;  
            }  
            return age;  
        }  
        //获取salary  
        public double getSalary()  
        {  
            return salary;  
        }  
        //修改雇员工资:涨byPercent个百分点的工资  
        public void setSalary(double salary,double byPercent)  
        {  
            salary + = salary*byPercent/100;  
        }  
    } 
    

    其中的public Employee(String name,int age,double salary)就是对这个类Employee的构造函数

    1、特点:
    a.函数名与类名相同
    b.不用定义返回类型,括号中的参数可以没有,也可以有多个。
    c.不可以写return语句
    d.构造函数总是伴随着new操作一起被调用。
    e.细节:当一个类中没有定义构造函数时,那么系统会默认给该类加入一个空参数的构造函数,即:类名(){}。但当类中自定义了构造函数后,默认的构造函数就不存在了。也就是说,每一个类中一定含有一个构造函数。

    2、作用:给对象进行初始化

    3、注意:
    a.默认构造函数的特点
    b.多个构造函数时以重载形式存在的。

    3、构造函数与一般函数的区别:运行上不同
    构造函数时在对象一建立就运行,是给对象初始化的。且一个对象建立后,构造函数只运行一次。
    一般函数时对象调用才执行,是给对象添加对象具备的功能的。一般函数可以被该对象多次调用。
    注:
    一般函数是不能调用构造函数的,因为一般函数中不能定义this,而构造函数中可能存在this。

    4、何时定义构造函数:
    当分析事物时,该事物存在具备一些特性或行为,那么将这些内容定义在构造函数中。也就是说,一类事物一出现就应该存在的特性,这就需要构造函数对其进行初始化。

    5、构造代码块:
    如上面的小程序,其中打印的“我来工作了”这个语句块就是构造代码块。它和构造函数作用类似,只不过仅用一对花括号括起来即可。
    1)作用:给对象进行初始化,对象一建立就运行,且优于构造函数执行,即在构造函数前加载。
    2)与构造函数区别:
    构造代码块是给所有对象进行统一初始化,是所有对象的共性初始化方式内容。如任何孩子出生都要哭几声。

    构造函数是给对应的对象初始化,不同的对象有各自的特性,需要选择不同的构造函数初始化。

    <<<<<------------------------------------------------------------------------------------------------------------------->>>>>
    补充:
    this关键字:
    1、this:表面上,是用于区分局部变量和成员变量同名的情况。即this代表着当前对象的引用。

    2、this的用法:
    a.用于区分同名变量的情况。当局部已经定义了变量,当需要找成员中的变量,且变量名相同时,为了区分局部变量和成员变量的相同变量名,可以使用this加以区分。
    b.用于构造函数间的调用。

    3、特点:
    1)this代表了本类的对象,即代表了它所在函数所属对象的引用,也就是说,哪个对象调用这个函数,this就代表哪个对象。
    2)一般情况下,this都是被省略的,需要使用的时候才需加上。

    如上面构造函数中的this的使用。

    4、this应用:
    当定义类中的功能时,该函数内部要用到调用该函数的对象时,这时用this来表示调用这个函数的对象。
    只要本类功能内部使用了本类对象,都用this表示。

    5、在构造函数中的应用:
    this语句:用于构造函数之间的互相引用。且只能放在构造函数的第一行,否则编译失败。
    原因:初始化的动作要先执行,因为自身的特性要先具备;如果初始化中还有初始化,要限制性更细节的操作,然后再执行自己所需求的初始化。

    注意:
    不允许两个构造函数间相互调用this语句,否则会出现死循环。
    一般函数中不能定义this语句。

    <<<<<------------------------------------------------------------------------------------------------------------------->>>>>

    3、static之静态函数

    我们在学习java的最初就接触到了static这个修饰main方法的修饰符,那么static在java中有什么特点和作用呢?下面是对static的总结的几点:

    一)总体来说static有这么几个特点:

    1、随着类的加载而加载,在类中生命周期最长

    2、优先于对象而存在

    3、可以被所有对象共享

    4、可以直接被类名调用

    二)static的用法:
    1、修饰成员:包括成员变量和成员方法
    当成员被static修饰后,就多了一种调用方式,即可以被对象和类名调用

    需要注意的一点:static绝对不能修饰局部变量。为什么呢?

    局部变量的作用域就是它所在的方法或代码块中,而static的变量刚是定义在类中方法体外,是作为整个类共同使用的,它从类加载开始就存在,而局部变量在它所在的方法或代码块结束后就要被回收的。所以是不能修饰局部变量的

    1)静态常量:
    例如:在Math类中定义了一个静态常量PI

    public class Math    
    {    
        ...    
        public static final double PI = 3.14159265358979323846;//final将PI设置为不可再定义的常量  
        ...    
    }
    

    如果省略了static,PI就变成了一个实例常量,那么,每一个Math对象就都有自己的一个PI拷贝了,这样的话,对内存也是一种占用。
    2)静态变量
    如果将变量定义为static,那么每个类中只有这样一个变量。例如:

    class Student  
    {  
        private int id;  
        private static int nextId = 1;  
        //获取id的访问器  
        public int getId()  
        {  
            return id;  
        }  
        public void setId()  
        {  
            id = nextId;  
            nextId++;  
        }  
    }
    

    每个学生都有自己的一个id号,但这个Student的所有对象都共享一个nextId,即使不存在对象,也仍存在nextId,因为它是属于类的,而不属于任何独立的对象,所以它是随着类的加载而加载的,随着类的消亡而消亡的。

    3)静态方法:
    静态方法是一种不向对象进行操作的方法。当方法被static修饰的时候,此方法是用类名.方法名的方式使用的,当然也可以用对象名.方法名,但是这样就会产生误解,会让别人以为这个方法是非静态的,这样就有些不合理了。但是静态方法有一点需要注意的是:静态方法只能访问静态成员,因此静态方法也就不能定义this和super等关键字了。

    那么在什么时候用到静态方法呢?
    第一、当一个方法不需要访问对象是,其中所需的参数都是通过显示参数提供的。比如说Math.sqrt(double n)
    第二、一个方法只需要访问累的静态变量时,如Person.getCountry();

    private static String country = "CN";//因为国家是共有的,共享  
    public static void getCountry()  
    {  
        return country;  
    }
    

    3、静态的应用:
    当每个应用程序都有共同之处,可将其封装,提高其复用性。
    需要注意的是:
    a.对象时用于封装数据的;
    b.对数据的操作方法,若没用到方法中特有的数据,则无需创建对象而占用多余的内存。

    4、静态代码块
    特点:a.随着类的加载而执行,且优先于主函数;b.只执行一次,类再创建对象,则不再执行,已经存在于内存中。
    格式:

    static  
       {语句}
    

    5、静态是用注意事项:
    1)静态方法只能访问静态成员,非静态方法既能访问静态也可以访问非静态
    2)静态方法中不可定义this,super等关键字,因为静态优先于对象存在,所以静态方法中不可以出现this
    3)主函数是静态的。

    6、静态有利有弊:
    好处:对对象的共享数据惊醒单独空间的存储,节省空进。没必要每个对象中都要存储一份,可以直接被类名调用
    弊端:生命周期过长。访问出现局限性,因为静态只能访问静态。

    举例:

    class Student  
    {  
        private static String country = "CN";//因为国家是共有的,共享  
        private int id;  
        private static int nextId = 1;  
        //静态代码块  
        static  
        {  
            System.out.println("Hello");  
        }  
        public static String getCountry()  
        {  
            return country;  
        }  
        public int getId()  
        {  
            return id;  
        }  
        public void setId()  
        {  
            id = nextId;  
            nextId++;//前一个人获取id后,然后一个人获取下一个id时就加1  
        }  
    }  
      
    class StudentText  
    {  
        public static void main(String [] args)  
        {  
            Student st1 = new Student();  
            st1.setId();  
            System.out.println("st1'country = " + Student.getCountry() + "; st1id = " + st1.getId());  
            Student st2 = new Student();  
            st2.setId();  
            System.out.println("st2'country = " + Student.getCountry() + "; st2id = " + st2.getId());  
        }  
    }
    

    运行结果如下:

    Paste_Image.png

    四、单例设计模式

    在java中存在很多通用的设计模式,今天我简单总结一下单例设计模式:
    解决问题:解决一个类在内存中只存在一个对象的问题(比如说一个软件中的配置文件)

    1、如何保证对象的唯一性:

    1、为避免建立过多的该类对象,应首先禁止其他应用程序创建该类对象。
    2、为让其他应用程序访问到该对象,在本类中自定义一个对象,为避免直接访问该对象,要对其进行私有化。
    3、提供访问方式,便于其他程序对自定义对象的访问,提供的访问方法是公有的。
    对象保证是惟一的了,那么该如何具体实现呢?

    2、使对象唯一性的步骤:

    1、将构造函数私有化
    2、在类中创建一个本类对象,并设置为私有的
    3、提供一个公有的方法获取该对象,便于使用

    3、单例设计模式的具体表现形式

    具体代码如下:
    1、饿汉式:先初始化对象,类一进内存就加载

    //饿汉式  
    class Single   
    {  
        private Single(){}  
        private static Single s = new Single();  
        public static  Single getSingle()  
        {  
            return s;  
        }  
    }  
      
    class SingleText  
    {  
        public static void main(String [] args)  
        {  
            Single s1 = Single.getSingle();  
            Single s2 = Single.getSingle();  
            if (s1==s2)  
                System.out.println(true);  
            else  
                System.out.println(false);        
        }  
    }
    

    2、懒汉式:类进内存,对象还没有存在,只有调用了getSingle方法时,才建立对象

    //懒汉式  
    class Single   
    {  
        private Single(){}  
        private static Single s = null;  
        public static Single getSingle()  
        {  
            if (s==null)   
                s = new Single();  
            return s;  
        }  
    }  
      
    class SingleText  
    {  
        public static void main(String [] args)  
        {  
            Single s1 = Single.getSingle();  
            Single s2 = Single.getSingle();  
            if (s1==s2)  
                System.out.println(true);  
            else  
                System.out.println(false);        
        }  
    }
    

    运行的结果是:true;这是因为s1和s2引用的是同一个对象,所以符合条件。

    但是对于第二种懒汉式的单例设计模式,会出现一些小小的问题,当一个线程调用时,是没什么问题的,如果多个线程调用此种方式,那么就会出现问题。

    class Single  
    {  
        private static Single s = null;  
        private Single(){}  
        public static Single getInstance()  
        {  
            if (s == null)  
            {  
                synchronized(Single.class)  
                {  
                    if (s == null)  
                        s = new Single();  
                }  
            }  
            return s;  
        }  
    }
    

    比如说,当A调用时,当读到if(s1==null) 时,可能就停在这了,然后cpu再调用B,B也读到if(s1==null)这停下了,cpu再切换到A,接着创建一个对象,A就执行完了;之后B也向下执行,又创建一个对象;此时,对象就不唯一了,就破坏了对象的唯一性的初衷。那么解决方案是这样的:
    这利用了锁的机制。synchronized是java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。这涉及到了多线程的问题。在这个例子中,比如说,当A调用时,当读到第二个if(s1==null) 时,可能就停在这了,然后cpu再调用B,B读到第一个if(s1==null)这停下了,因为加上synchronized后,A进去就相当于将其他的调用锁在外面的语句上了,要先执行完A,那么A执行完后,就已经创建了一个对象;当B再读到第二个if(s1==null)的时候不符合就直接结束了。如果再有其他C或D等调用的时候,就直接不符合第一个(s1==null)的条件,所以直接返回s。在这里,我们再来看看关于懒汉式的多线程问题:

    上面的懒汉式的写法,是效率比较高的,先看看下面一段代码,比较一下,就会清晰很多:

    class Single  
    {  
        private static Single s = null;  
        private Single(){}  
        public static synchronized Single getInstance()  
        {  
            if (s == null)  
                s = new Single();  
            return s;  
        }  
    }
    

    在这两种方式中,含有双重判断(称为第一种,无双重判断的为第二种)的效率更高,为什呢?虽然第一种和第二种都要先判断一下,但是对于第一种,第一个线程执行完后,s不为null了,那么后面只需要判断s是否为null即可,而对于第二种,要先判断锁,锁里没有线程,再进入,然后再判断一下s是否为null,这样一来,就要判断两次,所以,效率会更低。所以,对于双重判断,是可以提高效率的。

    问题是解决了,但是相比之下,还是第一种饿汉式的单例设计模式更好一些,是一种建议使用的方式。

    相关文章

      网友评论

          本文标题:Java学习之面向对象与类之概述(匿名对象、封装、构造函数、th

          本文链接:https://www.haomeiwen.com/subject/qjtqfxtx.html