黑马程序员----------面向对象三个权限修饰符

来源:互联网 发布:手机版收银软件 编辑:程序博客网 时间:2024/06/05 14:13
                                                               ---------------------- android培训、java培训、期待与您交流! ----------------------  


static

静态变量
    被static修饰的变量其实是一个全局变量。
    当这个类被虚拟机第一次加载的时候,就会为该变量分配了内存空间。
    当该类创建实例时,并不会生成对static变量的拷贝。而是多个该类的实例共享使用该变量。所有该类的对

象都可以操作这块存储空间。如果用final修饰就另当别论了。
    
    创建完成就需要进行初始化
    1. 定义时直接初始化
    2. 如果需要通过计算来初始化你的static变量,可以声明一个static块,Static 块仅在该类被加载时执行

一次,且在类被第一次装载时。
    【注意】static定义的变量的初始化会优先于任何其它非static变量,不论其出现的顺序如何。(代码1)
        在涉及到继承的时候,会先初始化父类的static变量,然后是子类的,依次类推。(自己验证)
    
    注意:
    1. 可以使用”类名.变量名“直接使用,并且被该类所有实例化对象共享
    2. 可以被类中所有方法使用(static与非static)
    3. 该类中某一个对象修改了变量的值,其他所有该类对象中的对应值都会随之改变
    4. 定义时初始化,或者通过静态代码块初始化
    
    ---------------------- android培训java培训、期待与您交流! ----------------------

静态方法
    
    被static修饰的方法我们称之为类方法。可以死通过类直接调用该方法,而没必要创建该类的实例后调用该

方法。
    1. 可以使用”类名.方法名“直接使用
    2. 只能调用其他Static方法
    3. 只能使用static成员变量
    4. 不能以任何形式引用this和super

    用途:静态方法常常为应用程序中的其它类提供一些实用工具,在Java的类库中大量的静态方法正是出于此

目的而定义的。Arrays和Collections

静态类

    通常一个普通类不允许声明为静态的,只有一个内部类才可以。这时这个声明为静态的内部类可以直接作为

一个普通类来使用,而不需实例一个外部类。(代码2)

    

补充:
    
    static表示“全局”或者“静态”的意思,用来修饰成员变量和成员方法,也可以形成静态static代码块,

但是Java语言中没有全局变量的概念。

    被static修饰的成员变量和成员方法独立于该类的任何对象。也就是说,它不依赖类特定的实例,被类的所

有实例共享。只要这个类被加载,Java虚拟机就能根据类名在运行时数据区的方法区内定找到他们。因此,static对

象可以在它的任何对象创建之前访问,无需引用任何对象。

    用public修饰的static成员变量和成员方法本质是全局变量和全局方法,当声明它类的对象时,不生成

static变量的副本,而是类的所有实例共享同一个static变量。

    static 变量前可以有private修饰,表示这个变量可以在类的静态代码块中,或者类的其他静态成员方法中

使用(当然也可以在非静态成员方法中使用--废话),但是不能在其他类中通过类名来直接引用,这一点很重要。实

际上你需要搞明白,private是访问权限限定,static表示不要实例化就可以使用,这样就容易理解多了。static前面

加上其它访问权限关键字的效果也以此类推。

    static修饰的成员变量和成员方法习惯上称为静态变量和静态方法,可以直接通过类名来访问,访问语法为


    类名.静态方法名(参数列表...)
    类名.静态变量名

    用static修饰的代码块表示静态代码块,当Java虚拟机(JVM)加载类时,就会执行该代码块(用处非常大,

呵呵)。

    static变量
    按照是否静态的对类成员变量进行分类可分两种:一种是被static修饰的变量,叫静态变量或类变量;另一

种是没有被static修饰的变量,叫实例变量。两者的区别是:
    对于静态变量在内存中只有一个拷贝(节省内存),JVM只为静态分配一次内存,在加载类的过程中完成静态

变量的内存分配,可用类名直接访问(方便),当然也可以通过对象来访问(但是这是不推荐的)。
    对于实例变量,没创建一个实例,就会为实例变量分配一次内存,实例变量可以在内存中有多个拷贝,互不

影响(灵活)。

    static方法
    静态方法可以直接通过类名调用,任何的实例也都可以调用,因此静态方法中不能用this和super关键字,不

能直接访问所属类的实例变量和实例方法 (就是不带static的成员变量和成员成员方法



final

    final特点:
    1. 用final修饰的变量表示常量,只能被赋一次值,不能修改。
        final修饰的基本类型变量:值不能被修改
        final修饰的引用类型变量(对象):对象地址不能被修改,对象内部的成员可以被修改
        被定义为final的对象引用只能指向唯一一个对象,不可以将它再指向其他对象,但是一个对象内部

的值却是可以改变的。

        被final修饰的变量是一个常量,必须被赋值后才能使用。可以在定义时赋值,也可在构造方法中赋

值。(只要在构造方法结束前给赋值就OK。)

    2. 用final修饰的方法不能被子类的方法覆盖;
        
    3. 用final修饰的类不能被继承,没有子类;
          final类不能被继承,因此final类的成员方法没有机会被覆盖,默认都是final的。
          但是final类中的成员变量可以被定义为final或非final形式。
          在设计类时候,如果这个类不需要有子类,类的实现细节不允许改变,那么就设计为final类。
    4. final不能用来修饰构造方法。


static和final
    static final用来修饰成员变量和成员方法,可简单理解为“全局量”!
    对于变量,表示一旦给值就不可修改,并且通过类名可以访问。
    对于方法,表示不可覆盖,并且可以通过类名直接访问。



抽象类与接口

    
    抽象类:

    含有abstract修饰符的class即为抽象类,abstract 类不能创建的实例对象。含有abstract方法的类必须定

义为abstract class,abstract class类中的方法不必是抽象的。abstract class类中定义抽象方法必须在具体

(Concrete)子类中实现,所以,不能有抽象构造方法或抽象静态方法。如果的子类没有实现抽象父类中的所有抽象方

法,那么子类也必须定义为abstract类型。
    接口:
    可以说成是抽象类的一种特例,接口中的所有方法都必须是抽象的。接口中的方法定义默认为public

abstract类型,接口中的成员变量类型默认为public static final。
    
    下面比较一下两者的语法区别:
    1. 抽象类可以有构造方法,接口中不能有构造方法。
    2. 抽象类中可以有普通成员变量,接口中没有普通成员变量
    3. 抽象类中可以包含非抽象的普通方法,接口中的所有方法必须都是抽象的,不能有非抽象的普通方法。
    4. 抽象类中的抽象方法的访问类型可以是public,protected,但接口中的抽象方法只能是public类型的,

并且默认即为public abstract类型。
    5. 抽象类中可以包含静态方法,接口中不能包含静态方法
    6. 抽象类和接口中都可以包含静态成员变量,抽象类中的静态成员变量的访问类型可以任意,但接口中定义

的变量只能是public static final类型,并且默认即为public static final类型。
    7. 一个类可以实现多个接口,但只能继承一个抽象类。
    

    
    接口和抽象类的概念不一样(通俗的讲)。

    
    接口是对动作的抽象,抽象类是对根源的抽象。

    抽象类表示的是,这个对象是什么。接口表示的是,这个对象能做什么。比如,男人,女人,这两个类(如

果是类的话……),他们的抽象类是人。说明,他们都是人。铁(iron)门、玻璃门这两个类的抽象类是门,说明他

们都是门。
    
    人可以吃东西,狗也可以吃东西,你可以把“吃东西”定义成一个接口,然后让这些类去实现它.

    所以,Java中一个类只能继承一个类(抽象类)(正如男人不可能同时是人和门),但是可以实现多个接口(吃

饭接口、走路接口)。
    
    当你关注一个事物的本质的时候,用抽象类;当你关注一个操作的时候,用接口。
    
    接口可以实现也可以继承,抽象类不行
    抽象类的功能要远超过接口,但是,定义抽象类的代价高。因为高级语言来说(从实际设计上来说也是)每

个类只能继承一个类。在这个类中,你必须继承或编写出其所有子类的
    所有共性。虽然接口在功能上会弱化许多,但是它只是针对一个动作的描述。而且你可以在一个类中同时实

现多个接口。在设计阶段会降低难度的。






代码1:
package cn.itcast.employment.staticdemo;

class Person {
    //年龄
    public static int age = 0;
    
    //出生
    Person(){}
    
    //克隆人
    Person(int age){
        this.age = age;
    }
    
    //过了一年
    public void grow(){
        age++;
    }
}

public class staticDemo {
    
    Person p = new Person(10);
    
    static Person p1,p2;
    
    static{
        System.out.println("p1.age =" + p1.age + ",p2.age =" + p2.age);
        p1 = new Person(20);
        System.out.println("p1.age =" + p1.age + ",p2.age =" + p2.age);
        p2 = new Person(30);
        System.out.println("p1.age =" + p1.age + ",p2.age =" + p2.age);
    }
    
    public static void main(String[] args) {
        
        staticDemo sd = new staticDemo();
        
        System.out.println("p.age =" + sd.p.age);
        System.out.println("p1.age =" +p1.age + ",p2.age =" + p2.age);
        p1.grow();
        System.out.println("p1.age =" +p1.age + ",p2.age =" + p2.age);
        System.out.println(sd.p.age);
    }
}


代码2:
    public class StaticCls {
        public static void main(String[] args) {
            OuterCls.InnerCls oi = new OuterCls.InnerCls();
        }
    }

    class OuterCls {
        public static class InnerCls {
            InnerCls() {
                System.out.println("InnerCls");
            }
        }
    }

在JAVA中如何完全跳出当前的多重嵌套循环?

代码:
方式一:可以在外面的循环语句前定义一个标号,然后在里层循环体的代码中使用带有标号的break 语句,即可跳出

外层循环。
    ok:for(int i=0;i<arr.length ;i++)
    {
        for(int j=0;j<arr[i].length;j++)
        {
            
            if(arr[i][j]  == 5)
            {
                break ok;
            }
            System.out.println(“i=” + i + “,j=” + j);
        }
    }  

方式二:让外层的循环条件表达式的结果可以受到里层循环体代码的控制,例如,要在二维数组中查找到某个数字。
int arr[][] = {{1,2,3},{4,5,6,7},{9}};
boolean found = true;
for(int i=0;i<arr.length && found;i++)
    {
        for(int j=0;j<arr[i].length;j++)
        {
            
            if(arr[i][j]  == 5)
            {
                found = true;
                break;
            }
            System.out.println(“i=” + i + “,j=” + j);
        }
    }

---------------------- <a href="http://www.itheima.com"target="blank">ASP.Net+Unity开发</a>、<a

href="http://www.itheima.com"target="blank">.Net培训</a>、期待与您交流! ----------------------
0 0
原创粉丝点击