Java4Android类和对象的初始化详解

1,成员初始化

Java尽力保证:所有变量在使用前都能够恰当的初始化。

 1)方法的局部变量。Java以编译时错误来贯彻这种保证。eg:

void f(){
    int i;
    i++; //Error , i not initialized
}

 2)类的数据成员。如果是基本类型,他们都会有一个初始值;如果是对象引用,那么这个引用将会被初始化为null。

指定初始化

如果想为某个变量赋值,该怎么做?

1)直接在定义类成员变量的地方为其赋值(注意,C++里面是不可以的,尽管C++新手们总想这么做)

class Depth{}

public class InitialValues{
    boolean bool = true;
    char ch = 'x';
    int i = 999;
    Depth d = new Depth();
}

2)调用某个方法来提供初始值。

public class MethodInit{
int i = f();
int f(){return 11;}
}

这个函数还可以带参数,但必须是初始化过的参数。

public class MethodInit2{
int i = f();
int j = g(i);
int f(){
    return 11;
}
int g(int n){
   return n*10;
}
}

但下面的写法确实错误的

public class MethodInit3{
// int j = g(i) ; //Illegal forward reference
int i = f();
int f(){return 11;}
int g(int n){return n*10;}
}

显然,上述程序的正确性取决于初始化的顺序。编译器会发出“向前引用”的警告。

上述这种在定义的地方就初始化,简单快捷,这样创造出来的每个对象都会具有相同的值。

2,构造器初始化

在C++里面叫做构造函数,这里我们叫做构造器,都一个东西。上一节讲到的自动初始化是一定会被执行的,而且会在构造器之前执行。

public class Counter{
int i;
Counter(){i = 8;}
}

上述代码,i首先会被置为0,然后变为7.

初始化顺序

在类的内部,变量定义的先后顺序决定了初始化的顺序。即使变量定义散布于方法定义之间,他们仍然会在任何方法(包括构造器)被调用之前得到初始化。

class Window{
Window(int marker){print("window("+marker+")");}
}
class House{
window w1 = new Window(1);
House(){
print("house()");
w3 = new Window(33);
}
Window w2 = new Window(2);
void f(){print("f()");}
Window w3 = new Window(3);

}
public class OrderOfInitialization{
public static void main(String[] args){
    House h = new House();
   h.f();
}
}

/**Output**/
/*
Window(1)
Window(2)
Window(3)
House()
Window(33)
f()*/

静态数据的初始化

class Bowl{
Bowl(int marker){print("Bowl("+marker+")");}
void f1(int marker){print("f1("+marker+")");}
}

class Table{
static Bowl bowl1 = new Bowl(1);
Table(){
    print("Table()");
    bowl2.f1(1);
}
void f2(int marker){
    print("f2("+marker+")");
}
static Bowl bowl2 = new Bowl(2);
}

class Cupboard{
Bowl bowl3 = new Bowl(3);
static Bowl bowl4 = new Bowl(4);
Cupboard(){
    print("Cupboard()");
    bowl4.f1(2);
}
void f3(int marker){
    print("f3("+marker+")");
}
static Bowl bowl5 = new Bowl(5);
}

public class StaticInitialization{
public static void main(String[] args){
print("Creating new Cupboard() in main");
new Cupboard();
print("Creating new Cupboard() in main");
new Cupboard();
table.f2(1);
cupboard.f3(1);
}
static Table table = new Table();
static Cupboard cupboard = new Cupboard();
}
/**output*/
/*
Bowl(1)
Bowl(2)
Table()
f1(1)
Bowl(4)
Bowl(5)
Bowl(3)
Cupboard()
f1(2)
Creating new Cupboard in main
Bowl(3)
Cupboard()
f1(2)
Creating new Cupboard in main
Bowl(3)
Cupboard()
f1(2)
f2(1)
f3(1)
*/

请务必细心的看上述代码和结果。最好自己能够运行一遍。

初始化的顺序是先静态对象,后非静态对象。

静态初始化只有在必要的时刻才会进行。如果不创建Table对象,也不引用Table.b1,Table.b2,那么静态的Bowl b1 和b2 永远都不会被创建。下面的步骤我们会解析为什么。

对象创建过程

总结下对象创建的过程。假如有一个名为Dog的类:

1)即使没有显式的使用static关键字,构造器实际上也是静态方法。因此,当首次创建类型为Dog的对象时(构造器可以看成是静态方法),或者Dog类的静态方法,静态域首次被访问的时候,Java解释器必须查找路径,以定位Dog.class文件。

2)然后载入Dog.class(这将创建一个Class对象),有关静态初始化的所有动作都会执行。因此,静态初始化只在Class对象首次加载的时候进行一次。

3)当用new Dog()创建对象的时候,首先将在堆上为Dog对象分配足够的存储空间。

4)这块存储空间会被清零。这就自动的将Dog对象中的所有基本类型数据都设置成了默认值,而引用则被设置成了null。(再次强调方法内部的局部变量不会被初始化,如果使用未初始化的局部变量会提示编译错误)。

5)执行所有出现于字段定义处的初始化动作。

6)执行构造器。

显式的静态初始化(静态块)

Java允许将多个静态初始化动作组织成一个特殊的“静态子句”(也叫静态块)。

public class Spoon{
static int i;
static {
    i = 47;
}
}

上述代码和静态初始化动作是一样的,代码仅在必要的时刻仅执行一次。

class Cup{
Cup(int marker){
    print("Cup("+marker+")");
}
void f(int marker){
    print("f("+marker+")");
}
}

class Cups{
static Cup cup1;
static Cup cup2;
static {
    cup1 = new Cup(1);
    cup2 = new Cup(2);
}
Cups(){
    print("Cups()");
}
}

public class ExplicitStatic{
    public static void main(String[] args){
     print("Inside main()");
     Cups.cup1.f(99);//(1)
}
    //static Cups cups1 = new Cups();//(2)
    //static Cups cups2 = new Cups();//(3)
}
/**output*/
/*
Inside main()
Cup(1)
Cup(2)
f(99)
*/

上述代码无论是运行(1)还是把(1)注释了运行(2),Cups的静态初始化都会得到执行。(3)也可以打开看看,无关紧要,因为静态初始化动作只进行一次。

非静态实例初始化

它在静态初始化之后,在构造函数之前,前面我们讲述过这个步骤。我们来看一段代码:

class Mug{
Mug(int marker){
    print("Mug("+marker+")");
}
void f(int marker){
    print("f("+marker+")");
}
}

public class Mugs{
Mug mug1;
Mug mug2;
{
    mug1 = new Mug(1);
    mug2 = new Mug(2);
    printf("mug1 & mug2 initialized");
}
Mugs(){
    print("Mugs()");
}
Mugs(int i){
    print("Mugs(int)");
}
public static void main(String[],args){
    print("Inside main()");
    new Mugs();
    print("new Mugs() completed");
    new Mugs(1);
    print("new Mugs(1) completed");
}
}
/** output*/
/*
Inside main()
Mug(1)
Mug(2)
mug1 & mug2 initialized
Mugs()
new Mugs() completed
Mug(1)
Mug(2)
mug1 & mug2 initialized
Mugs(int)
new Mugs(1) completed
*/

这种语法对于支持“匿名内部类”的初始化是必须的。

累了。今天学习到这里。

时间: 2024-08-30 12:45:50

Java4Android类和对象的初始化详解的相关文章

PHP类和对象函数实例详解

1. interface_exists.class_exists.method_exists和property_exists:         顾名思义,从以上几个函数的命名便可以猜出几分他们的功能.我想这也是我随着对PHP的深入学习而越来越喜欢这门编程语言的原因了吧.下面先给出他们的原型声明和简短说明,更多的还是直接看例子代码吧. bool interface_exists (string $interface_name [, bool $autoload = true ]) 判断接口是否存在

Java语言class类用法及泛化(详解)_java

这篇文章主要介绍了Java语言class类用法及泛化(详解),大家都知道Java程序在运行过程中,对所有的对象进行类型标识,也就是RTTI.这项信息记录了每个对象所属的类.虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类.Class类封装一个对象和接口运行时的状态,当装载类时,Class类型的对象自动创建,具体内容介绍如下: 说白了就是: Class类也是类的一种,只是名字和class关键字高度相似.Java是大小写敏感的语言. Class类的对象内容是你创

CSS中伪类及伪元素用法详解

原文:CSS中伪类及伪元素用法详解 伪类的分类及作用: 注:该表引自W3School教程 伪元素的分类及作用: 接下来让博主通过一些生动的实例(之前的作业或小作品)来说明几种常用伪类的用法和效果,其他的读者可以自己尝试:   :active  大致效果为用鼠标点击时,元素增加特效,鼠标松开时,特效消失.多用在按钮的点击上.  写法: 这里id为box的是一div块,在css中首先设置了他的基本样式,下面为加入:active伪类后需要修改的样式. 未点击时: 点击之后: :active.:hove

PHP中Closure类的使用方法及详解_php实例

Closure,匿名函数,又称为Anonymous functions,是php5.3的时候引入的.匿名函数就是没有定义名字的函数.这点牢牢记住就能理解匿名函数的定义了. Closure 类(PHP 5 >= 5.3.0)简介 用于代表 匿名函数 的类. 匿名函数(在 PHP 5.3 中被引入)会产生这个类型的对象,下面我们来看一下PHP Closure类的使用方法及介绍. PHP Closure类之前在PHP预定义接口中介绍过,但它可不是interface哦,它是一个内部的final类.Clo

JavaScript中子对象访问父对象的方式详解_javascript技巧

在传统面向对象的编程语言里,都会提供一种子类访问父类的特殊语法,引文我们在实现子类方法往往需要父类方法的额外辅助.在这种情况下,子类通常会调用父类中的同名方法,最终以便完成工作. javascript虽然没有类似上述的特殊语法,但我们可以造一个啊! function her(){}; her.prototype.name = 'Anna'; her.prototype.toString = function(){ var const = this.constructor; return cons

Java多线程编程中使用Condition类操作锁的方法详解_java

Condition的作用是对锁进行更精确的控制.Condition中的await()方法相当于Object的wait()方法,Condition中的signal()方法相当于Object的notify()方法,Condition中的signalAll()相当于Object的notifyAll()方法.不同的是,Object中的wait(),notify(),notifyAll()方法是和"同步锁"(synchronized关键字)捆绑使用的:而Condition是需要与"互斥

使用Java的Graphics类进行绘图的方法详解_java

Graphics类提供基本绘图方法,Graphics2D类提供更强大的绘图能力.本节讲解Graphics类,下节讲解Graphics2D. Graphics类提供基本的几何图形绘制方法,主要有:画线段.画矩形.画圆.画带颜色的图形.画椭圆.画圆弧.画多边形等. 1. 画线 在窗口画一条线段,可以使用Graphics类的drawLine()方法:   drawLine(int x1,int y1,int x2,int y2) 例如,以下代码在点(3,3)与点(50,50)之间画线段,在点(100,

PHP类的封装与继承详解_php技巧

封装        把成员方法和成员属性封装到类中,隐藏属性和方法实现的细节,通过public.protected.private等限定类成员的访问权限,数据被保护在内部,只通过被授权的成员方法才可以操作,尽可能的对成员进行封装. public:方法或者属性在任何作用域下都可以访问到,而且是默认的,如果没有为一个属性或方法指定访问修饰符,它将是public.protected:本类和子类可以访问,外部对象不可以调用.private:只能在本类访问,子类与外部对象都不可能调用.经过private标

常用Response对象的使用详解

response|对象|详解 每一个程序语言或开发工具都有一定的函数与用户进行沟通,Asp同样如此.在Asp中负责将信息传递给用户的对象就是Response对象.Response对象用于动态响应客户端请求(Request),并将动态生成的响应结果返回到客户端浏览器中,使用Response对象可以直接发送信息给浏览器.重定向浏览器到另一个URL或设置cookie的值等等.Response对象在asp编程中非常广泛,也是一种非常好用的工具.下面我们来具体看看Response对象: 语法: Respo