如果有人问你: "子类继承父类所有非私有(private)的属性和方法这句话对吗?", 如果你回答对的, 那我只能说too young too simple!

关于代码块和成员变量初始化的问题你清楚吗?请看这里

修饰符的问题

  为什么说这句话不对了, 只要你搞清楚了4个访问修饰符的应用范围你就知道了

  public: 任何地方都能访问

  private: 类访问符(自己取得名字), 只要在本类中的都可以访问, 其他类中不可以访问

  默认: 即不加修饰符的时候, 这时候是包访问符, 只要在本包中的都可以访问, 其他包不可以访问(其他包中的子类也不可以访问, 这也正是和protected的区别)

  protected: 默认+非同包的子类, 即可以在本包和非同包的子类中访问, 在非同包的非子类中不可以访问

  可以自己写个案例验证一下, 也有助于自己更好的理解, 那我再问: "实现类'继承'父接口的所有非私有的属性和方法对吗?", 如果你受到上面的启发脱口而出'不对', 那你就要仔细思考一下了. 首先这个问题就有问题, 因为接口中的属性默认被public static final修饰, 接口中的方法默认被public abstract修饰, 所以接口中没有私有的属性和方法, 所以说"实现类'继承'父接口中所有的属性和方法"是对的.

  怎么验证呢?

    static: 验证在实现类中是否能被类名直接调用

    final: 验证在实现类中是否能修改属性的值

    abstract: 试着在接口中写一个具体的方法

    public: 这个就不用说了吧

  

代码块的问题

  掌握java中的代码块有时对我们是很有帮助的, 主要有静态块(格式为static{doSomething})和构造块(格式为{doSomething}), 他们都是定义在类中的, 很简单, 通过下面的一个例子相信你就会完全了解了

public class CodeBlock {
private String code = "code"; static{
System.out.println("我是静态代码块");
} {
System.out.println("我是构造代码块");
} public CodeBlock(){
System.out.println("我是无参构造方法");
} //输入说明一切, 当然用调试模式看的更清楚
public static void main(String[] args) {
System.out.println("开始执行main方法");
new CodeBlock();
System.out.println("-----------分割线-----------");
new CodeBlock();
}
}

  自己执行一下, 看一下输出结果你就会发现结论是这样的: 静态块在加载类的时候执行, 而且只执行一次. 构造块在每次调用构造方法的时候都会执行, 所以构造块在构造方法之前执行. 他们执行的优先级: 静态块 > main方法 > 构造块 > 构造方法.

  可是我为什么要在这里加一个成员变量呢?我不知道你有没有这样的疑惑, 因为这和我接下来要讲的有关. 在讲之前先扯点淡, 我们都知道java中的变量在使用之前是要初始化的, 那么你真的清楚成员变量是在什么时候初始化的么?(老鸟请无视), 不要问我局部变量何时初始化的, 那可不就是在执行到他的时候初始化的嘛!好, 回到刚才代码块的这个案例中.

  没有用static修饰的成员变量也叫'实例变量', 对, 你从他的名字可能已经猜到了实例变量初始化的时机: 实例变量和构造块类似, 每次调用构造方法的时候都会执行初始化, 更具体的是在调用构造方法并调用完父类的构造方法之后, 不过他的优先级比构造块高点, 所以结合代码块一起看, 他们的优先级是: 静态块 > main方法 > 实例变量 > 构造块 > 构造方法.

static理解

  但是如果成员变量被static修饰呢?这时候就应该叫类变量了, 顾名思义, 类变量在类加载的时候就被初始化了. 在类中被static修饰的成员(变量, 方法, 代码块)在类加载的时候就被初始化并放到内存中, 系统会在内存中专门开辟出一块区域来存储他们, 以后就算创建类的实例, 静态成员也还是在那儿. 也就是说, 静态成员在类初始化一次后, 系统就不会为他们开辟新的内存空间. 而每new一个类的对象, 系统就会重新在堆内存中开辟一个新空间来存放该类的实例对象, 并且栈中也会有一个新的引用变量去指向它. 一个类的静态成员是被这个类的所有实例所共享的. 讲到这又想起来一个问题: 在静态方法中不能直接调用非静态方法. 因为静态方法随着类的加载而被加载到了内存中, 而非静态方法(实例方法)是在创建类的实例后才被加到内存中的, 所以假如在静态方法中直接调用非静态方法, 静态方法: "咦, 我要调用的这个方法不存在啊!", 但是在静态方法中new一个对象来调用非静态方法是肯定可以的. 关于static可以参考这里, 我觉得解释的比较好. 如果你还是有点懵懵的, 那就来个例子, 再加深一下理解.

  碗类:

public class Bowl {
Bowl(int marker){
System.out.println("碗的数量"+marker);
} void f1(int marker){
System.out.println("f1方法"+marker);
}
}

  碗柜类:

public class Cupboard {
Bowl bowl3 = new Bowl(3);
static Bowl bowl4 = new Bowl(4); Cupboard(){
System.out.println("这是碗柜的无参构造");
bowl4.f1(2);
} void f3(int marker){
System.out.println("这是f3方法, 数量"+marker);
} static Bowl bowl5 = new Bowl(5);
}

  桌子类:

public class Table {
static Bowl bowl1 = new Bowl(1);
Table(){
System.out.println("table的无参构造");
bowl2.f1(1);
}
void f2(int marker){
System.out.println("f2的数量"+marker);
}
static Bowl bowl2 = new Bowl(2);
}

  测试:

/**
* 通过本例了解对象的创建过程及先后顺序
* 在加载类StaticInitialzation时, 静态成员变量table被初始化, 从而导致table所
* 对应的类Table被加载, table中也有静态成员变量, 所以也会被加载, 非静态成员变量(实例变量)是在
* 调用构造方法的时候初始化, 所以在Cupboard中调用构造方法时(调用了构造方法, 但还没有执行
* 构造方法里面的主体代码, 使用调试可以很清楚地看到)才初始化bowl3
*/
public class StaticInitialzation {
public static void main(String[] args) {
System.out.println("创建碗柜");
new Cupboard();
System.out.println("再创建一个碗柜");
new Cupboard();
table.f2(1);
cupboard.f3(1); } static Table table = new Table();
static Cupboard cupboard = new Cupboard(); static{
System.out.println("我是静态块");
}
}

  自己试一下, 看输出的结果和自己想的是否一样, 打断点调试会比较有帮助.

后面还会更新其他的……

java中易遗忘的知识,不定时更新……的更多相关文章

  1. java中的字符串相关知识整理

    字符串为什么这么重要 写了多年java的开发应该对String不陌生,但是我却越发觉得它陌生.每学一门编程语言就会与字符串这个关键词打不少交道.看来它真的很重要. 字符串就是一系列的字符组合的串,如果 ...

  2. [转]java中的字符串相关知识整理

    字符串为什么这么重要 写了多年java的开发应该对String不陌生,但是我却越发觉得它陌生.每学一门编程语言就会与字符串这个关键词打不少交道.看来它真的很重要. 字符串就是一系列的字符组合的串,如果 ...

  3. java中数组的相关知识

      1. 2.数组的命名方法 1)int[]ages=new int[5]; 2) int[]ages; ages=new int[5]; 3)int[]ags={1,2,3,4,5}; 4)int[ ...

  4. Java中的NIO基础知识

    上一篇介绍了五种NIO模型,本篇将介绍Java中的NIO类库,为学习netty做好铺垫 Java NIO 由3个核心组成,分别是Channels,Buffers,Selectors.本文主要介绍着三个 ...

  5. java中易错点

    1.A instanceof  B{这是没有好好利用java多态的表现} java中的二元操作符,测试A对象是否是B类的实例: 返回值:boolean类型 2.“==”与 “equals”的区别: = ...

  6. java中一些常考知识

    一.static的作用 static是修饰符,用于修饰成员变量(静态变量/类变量). static修饰的成员被所有对象共享. static优先于对象存在. static修饰的成员可以用类名.静态成员来 ...

  7. Java专业技能面试问题(不定时更新)

    刚看到园友五月的仓颉<面试感悟----一名3年工作经验的程序员应该具备的技能>感觉很不错,不论是为面试跳槽准备,还是打算深化精进自己的技术都可以参考一下.面向工资编程多少也有点道理,虽然技 ...

  8. Java学习笔记(二)不定时更新

    Java语言画图 package cn.witksy.dev; import javax.imageio.ImageIO; import java.awt.*; import java.awt.ima ...

  9. Java中浮点数的基础知识

    偶然查看Math.round的JDK public static int round(float a) { if (a != 0x1.fffffep-2f) // greatest float val ...

随机推荐

  1. dagger2系列之依赖方式dependencies、包含方式(从属方式)SubComponent

    本篇是实战文章,从代码的角度分析这两种方式.本文参考自下列文章: http://www.jianshu.com/p/1d42d2e6f4a5 http://www.jianshu.com/p/94d4 ...

  2. 那些年【深入.NET平台和C#编程】

    一.深入.NET框架 1..NET框架具有两个组件:CLR(公共语言运行时)和FCL(框架类库),CLR是.NET框架的基础 2.框架核心类库: System.Collections.Generic: ...

  3. Effective java笔记(二),所有对象的通用方法

    Object类的所有非final方法(equals.hashCode.toString.clone.finalize)都要遵守通用约定(general contract),否则其它依赖于这些约定的类( ...

  4. github免输用户名/密码SSH登录的配置

    从github上获取的,自己整理了下,以备后用. Generating an SSH key mac windows SSH keys are a way to identify trusted co ...

  5. FineReport:关于扩展行列求各种条件下的函数运用

    最简单的扩展列,扩展行的求"最大,最小,平均"值的例子 设计图 效果图 相关函数 =MAX(B2:E2) =MIN(B2:E2) =AVERAGE(B2:E2) 这个是(满足条件) ...

  6. Javascript实践技巧

    最近辞职了,准备北上.期待有个好结果~   本文以<Javascript高级程序设计>为基础,结合自身经验来总结下Javascript实际工作方面的知识.   一.可维护性 1.代码约定 ...

  7. Fedora 22中的用户和用户组管理

    The control of users and groups is a core element of Fedora system administration. This chapter expl ...

  8. DUILIB入门使用说明整理

    整理自博客上对DUILIB的操作使用的博客文章 一.DuiLib操作入门 1.DuiLib入门简明教程 -- 前言 2.DuiLib入门简明教程 -- VS环境配置 3.DuiLib入门简明教程 -- ...

  9. Python 资源大全中文版

    Python 资源大全中文版 我想很多程序员应该记得 GitHub 上有一个 Awesome - XXX 系列的资源整理.awesome-python 是 vinta 发起维护的 Python 资源列 ...

  10. XML技术之SAX解析器

    1.解析XML文件有三种解析方法:DOM SAX DOM4J. 2.首先SAX解析技术只能读取XML文档中的数据信息,不能对其文档中的数据进行添加,删除,修改操作:这就是SAX解析技术的一个缺陷. 3 ...