Java基础知识陷阱(五)

本文发表于本人博客

今天我来说说关于静态变量初始化、数组、==与equals的问题,看下面代码:

public class Test{
    private final int age;

    private String name;

    public Test(){
        age = 30;
    }

    public Test(String name){
        this.name = name;
    }
}

我想上面的代码好多人都知道了会编译报错的,原因是在代餐构造函数中未对age进行复制操作。

对于终态变量final程序中是可以再2个地方赋值的,一是申明后赋值,二是构造函数中赋值,上面的代码带参构造修改成如下:

    public Test(String name){
        age = 30;
        this.name = name;
    }

这样编译就没问题可以通过了。那现在我们把这个age变量的修饰符增加static,看代码:

public class Test{
    private static final int age;

    private String name;

    public Test(){
        age = 30;
    }

    public Test(String name){
        age = 30;
        this.name = name;
    }
}

说说这样编译能通过吗!这样也是不行的,分析下当这个age变量变味static时它已经不属于任何对象了是属于类Test的了,那么如果在构造对象的时候还去赋值修改原有的值那肯定不行了,它只有在申明时候赋值,代码修改如下:

public class Test{
    private static final int age = 30;

    private String name;

    public Test(){
    }

    public Test(String name){
        this.name = name;
    }
}

继续看下面代码,看看这个能编译吗?

public class Test{
    public static final StringBuffer sb = new StringBuffer();
    public static void main(String[] args){
        sb.append("www.luoliang.me");
    }
}

这个比较简单,竟然sb为类常量,那么由于是类,它的引用地址是不能再修改了,但是这个指向的地址的内容是可以被修改的。

继续看下面代码,试问编译通过吗?如果不通过,哪里有问题?

interface I{}
public class demo{
    public static void main(String[] args){
        I[] iArray = new I[2];
        I i = new I();
    }
}

上面的代码大家看了就知道,接口interface是不能实例化的,那么new I()肯定是不行的。至于这个new来实例化I的数组,这个可以吗,先解析下,由于数组里面的元素是对象,那么对象是可以为null的,这样的话new I[2]表示长度为2的I对象数组,至于数组的元素一定是要继承接口I了,比如:

class Test implements I{
}
iArray[0] = new Test();

现在我们来看看这个==与equals的问题,在网上也好多了,现在自己再复习下,看下面代码:

public class Test{
    public static void main(String[] args){
        String str1 = new String("luoliang.me");
        String str2 = new String("luoliang.me");
        System.out.println(str1 == str2);
        System.out.println(str1.equals(str2));
    }
}

试问输出什么?这个倒是简单很多人都知道了,如果是基本类型==比较的是其值,如果是引用类型比较的是其地址。那equals这个方法呢,所有对象都是Object的子类,那我们查看下jdk源码中的Object,可以看到:

    public boolean equals(Object obj) {
        return (this == obj);
    }

他们是直接比较的地址啊,是==。那字符串呢,先不说我们看看String类的equals方法:

    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = count;
            if (n == anotherString.count) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = offset;
            int j = anotherString.offset;
            while (n-- != 0) {
                if (v1[i++] != v2[j++])
                return false;
            }
            return true;
            }
        }
        return false;
    }

可以看到起源码是先比较是否是同一个字符串,如果是的话返回true,如果不是的话继续比较他们各次的字符。那现在我们就知道答案了,输出如下:

false
true

接下来我们来看如下代码:

class Person{
    public String name;
    public Person(String name){
        this.name = name;
    }
    public String getName(){
        return this.name;
    }
}

现在如果2个Person对象进行比较,如果是名称name相同则判断是同一个人,怎么做?先分析一下:判断是否同一个人,这个对象是继承基类Object的,本身有equals方法,那按照String类的方法重写Override一下应该就可以解决了,看下面:

class Person{
    public String name;
    public Person(String name){
        this.name = name;
    }
    public String getName(){
        return this.name;
    }
    @Override
    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof Person) {
            Person person = (Person)anObject;
            if(person.name.equals(this.name)){
                return true;
            }
        }
        return false;
    }
}

调用如下:

        Person p1 = new Person("luoliang");
        Person p2 = new Person("luoliang");
        System.out.println(p1 == p2);
        System.out.println(p1.equals(p2));

输出结果:

false
true

这次先到这里。坚持记录点点滴滴!

时间: 2024-10-06 23:54:09

Java基础知识陷阱(五)的相关文章

Java基础知识陷阱系列

Java基础知识陷阱系列 今天抽空把Java基础知识陷阱有关的文章汇总于此,便于大家查看. Java基础知识陷阱(一) Java基础知识陷阱(二) Java基础知识陷阱(三) Java基础知识陷阱(四) Java基础知识陷阱(五) Java基础知识陷阱(六) Java基础知识陷阱(七) Java基础知识陷阱(八)

Java基础知识陷阱(九)

本文发表于本人博客. 今天我来说说关于JAVA多线程知识,有错误请指出.大家都知道JAVA在服务端上处理也有很大优势,很多公司也有在服务器跑JAVA进程,这说明JAVA在处理这个多线程以及并发下也有一定有优点的(这说法有点坑了).下面来看看 那java中,不具备直接操作像操作系统一样的PV信号,然而它提供了synchronized来实现同步机制,可能这样说不够严谨.JAVA的基类Object中有以下几个方法: public final native void notify(); public f

Java基础知识陷阱(十)

上个星期由于时间比较紧所以未能继续写下去,今天再接再厉,专心 + 坚持这样离目标就越来越近了!废话少说说正题,今天我们还是来说说java中比较基础的知识,大家知道编写java程序中很多时候都用到了xml文件,有 些是框架本身支持配置的,有些是自定义配置的,这样就要求我们要对这个xml原理要比较了解,其中加载xml文件转换节点元素时有个核心:递归调用转换. 我们可以通过下面方法来查查这个实现类有关的源码: 1 2 3 4 DocumentBuilderFactory factory = Docum

Java基础知识(五)

1.      在HashTable中同步和如何实现HashMap的同步 1. 同步意味着在一个时间点只能有一个线程可以修改hash表,任何线程在执行HashTable的更新操作前都需要获取对象锁,其他线程则等待锁的释放. 2.  HashMap可以通过Map m=Collection.synchronizedMap(new HashMap())来达到同步的效果.具体而言,该方法返回一个同步的Map,该Map封装了底层的HashMap的所有方法,使得底层的HashMap即使是在多线程的环境中也是

Java基础知识陷阱(三)

本文发表于本人博客. 之前都讲了有关字符串的陷阱,那今天来说下关于静态这个东西,这分为静态变量.静态方法,先看下面的代码请问结果输出是什么?: class Person01{ private static Person01 person = new Person01(); public static int age; public static int hight = 0; private Person01(){ age++; hight++; } public static Person01

Java基础知识陷阱(七)

本文发表于本人博客. 上次说了下HashSet和HashMap之间的关系,其中HashMap这个内部有这么一句: static final float DEFAULT_LOAD_FACTOR = 0.75f; 这一句表示一个常量,作用是当容器数量大小达到0.75%的时候就进行重新在构建一个2倍大小的数组.竟然这2个是集合,那我们今天看看其他的集合类,比如:ArrayList.Vector.LinkedList,开始咯. 首先查看下ArrayList的源码,这几个集合类都是在java.util包下

Java基础知识陷阱(四)

本文发表于本人博客. 今天我们来说说关于java继承以及反射有关的问题,大家先看下下面代码,试问可以编译通过不,为什么具体说说原因? public class Test{ public static void main(String[] args){ Person p = new Man(); p.talk(); } } class Person{ public void talk() throw NullPointerException{ System.out.println("Person

Java基础知识陷阱(六)

本文发表于本人博客. 上次说了下equals跟==的问题,今天再来认识一下这个equals()跟hasCode().上次的代码如下: class Person{ public String name; public Person(String name){ this.name = name; } public String getName(){ return this.name; } @Override public boolean equals(Object anObject) { if (t

Java基础知识陷阱(一)

本文发表于本人博客. 事隔好多年了,重新拿起来Java这门语言,看似熟悉其实还很陌生,想想应该梳理下顺便提高下自己.这次先来看看Java里面的String这个对象. 经典的先看下面一段代码,请问最终创建几个对象,分别在哪里? String s0 = new String("luoliang.me"); String s1 = "luoliang.me"; String s2 = new String("luoliang.me"); 想要回答这个问