注意点

2019-01-08  本文已影响0人  luckee

不使用IDE怎么编译和指定classpath

private default protected public

image.png

注意:java的访问控制是停留在编译层的,也就是它不会在.class文件中留下任何的痕迹,只在编译的时候进行访问控制的检查。其实,通过反射的手段,是可以访问任何包下任何类中的成员,例如,访问类的私有成员也是可能的。

Arrays.asList的用法

使用工具类Arrays.asList()把数组转换成集合时,不能使用其修改集合相关的方法,它的add/remove/clear方法会抛出UnsupportOperationException异常
说明:asList的返回对象是一个Arrays内部类,并没有实现集合的修改方法。Arrays.asList体现的是适配器模式,只是转换接口,后台的数据仍是数组。
String[] str = new String[]{"1","2"};
List list = Arrays.asList(str);
第一种情况:list.add("x");//运行时异常
第二种情况:str[0] = "unv";//那么list.get(0)也随着修改。
此类包含用来操作数组(比如排序和搜索)的各种方法。此类还包含一个允许将数组作为列表来查看的静态工厂。 除非特别注明,否则如果指定数组引用为 null,则此类中的方法都会抛出 NullPointerException。一段有意思的代码如下:
Java代码

public static void main(String[] args) {

int[] data = {1,2,3,4,5};

List list = Arrays.asList(data);

System.out.println("列表中的元素数量是:" + list.size());

}

注意这里输出的数量是1,原因是,在Arrays.asList中,该方法接受一个变长参数,一般可看做数组参数,但是因为int[] 本身就是一个类型,所以data变量作为参数传递时,编译器认为只传了一个变量,这个变量的类型是int数组,所以size为1。基本类型是不能作为泛型的参数,按道理应该使用包装类型,但这里缺没有报错,因为数组是可以泛型化的,所以转换后在list中就有一个类型为int的数组

public class ClearRepeatTest {
     
    public static Integer[] clearRepeat(int [] array){
        Set<Integer> set = new HashSet<>();
        for(int i : array){
            set.add(i);
        }
        Integer[] newArray = set.toArray(new Integer[set.size()]);
        return newArray;
    }
     
     
    public static void main(String[] args) {
        //创建一个数组,可以看出 2和4 是重复的
        int [] array = {1,2,3,4,8,2,5,4};
        Integer[] newArray = clearRepeat(array);
        for(Integer i : newArray){
            System.out.println(i);
        }
        //1 2 3 4 5 8
         
    }
 
}

==和equals

==比较的是两个操作数的值是否相等,对于基本类型变量和引用类型变量都是比较变量的值是否相等。对于基本类型的变量好理解,对于引用类型的变量,其本身也是变量,有自己的内存空间,存的是其引用的对象的堆空间地址。equals比较的是对象的内容是否一样,并不一定是同一个对象内容才相同,不同的对象内容也可能相同。

&&和&的区别(||和|也一样)

都可以是逻辑运算操作符,此时&&具有短路功能;除此之外,当操作数不是boolean值时,&则表示按位与

数据库表可以没有主键吗

可以

HashMap的四种遍历

transient

Java中transient关键字的作用,简单地说,就是让某些被修饰的成员属性变量不被序列化,这一看好像很好理解,就是不被序列化,那么什么情况下,一个对象的某些字段不需要被序列化呢?如果有如下情况,可以考虑使用关键字transient修饰:

1、类中的字段值可以根据其它字段推导出来,如一个长方形类有三个属性:长度、宽度、面积(示例而已,一般不会这样设计),那么在序列化的时候,面积这个属性就没必要被序列化了;

2、其它,看具体业务需求吧,哪些字段不想被序列化;

PS,记得之前看HashMap源码的时候,发现有个字段是用transient修饰的,我觉得还是有道理的,确实没必要对这个modCount字段进行序列化,因为没有意义,modCount主要用于判断HashMap是否被修改(像put、remove操作的时候,modCount都会自增),对于这种变量,一开始可以为任何值,0当然也是可以(new出来、反序列化出来、或者克隆clone出来的时候都是为0的),没必要持久化其值。

对象复制

浅复制:将一个对象复制后,基本数据类型的变量都会重新创建,而引用类型,指向的还是原对象所指向的。

深复制:将一个对象复制后,不论是基本数据类型还有引用类型,都是重新创建的。简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。

public class Prototype implements Cloneable, Serializable {
 
    private static final long serialVersionUID = 1L;
    private String string;
 
    private SerializableObject obj;
 
    /* 浅复制 */
    public Object clone() throws CloneNotSupportedException {
        Prototype proto = (Prototype) super.clone();
        return proto;
    }
 
    /* 深复制 */
    public Object deepClone() throws IOException, ClassNotFoundException {
 
        /* 写入当前对象的二进制流 */
        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        ObjectOutputStream oos = new ObjectOutputStream(bos);
        oos.writeObject(this);
 
        /* 读出二进制流产生的新对象 */
        ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
        ObjectInputStream ois = new ObjectInputStream(bis);
        return ois.readObject();
    }
 
    public String getString() {
        return string;
    }
 
    public void setString(String string) {
        this.string = string;
    }
 
    public SerializableObject getObj() {
        return obj;
    }
 
    public void setObj(SerializableObject obj) {
        this.obj = obj;
    }
 
}
 
class SerializableObject implements Serializable {
    private static final long serialVersionUID = 1L;
}

一个类是否可以调用其他类的main

一个类是否可以调用其他类的main

基本类型变量存在哪

基本数据类型存在于虚拟机栈区,基本数据类型的存储形式既不是对象引用,也不是地址,而是就是值本身。
在虚拟机栈区中,除了基本数据类型以外,还存在着对象的引用。这些引用就指向了堆区中的实际对象值。

类A有个字段是类B类型的,那么A类型的对象在堆空间地址范围中包含一个B类型的对象还是对象引用

我觉得在为A的对象分配空间的时候,计算需要申请多大的内存,应该算的是基本类型和String类型(因为String是不可变类),引用类型的字段应该用一种能够表示内存地址的类型(比如用一个int来表示一个地址)来表示,存放的值为B类型的对象在堆中的地址

如何计算对象的占用空间大小

计算对象空间大小

  1. 通过java.lang.instrument.Instrumentation的getObjectSize(obj)直接获取对象的大小

  2. 通过sun.misc.Unsafe对象的objectFieldOffset(field)等方法结合反射来计算对象的大小

main方法结束了,程序就结束了吗

不是的,main方法所在的线程是主线程,各个线程之间没有关系,main方法结束了,其他线程完全可能还在运行

接口继承了Object吗

没有。下面这段话是Oracle官方描述

If an interface has no direct superinterfaces, then the interface implicitly declares a public abstract member method m with signature s, return type r, and throwsclause t corresponding to each public instance method m with signature s, return type r, and throws clause t declared in Object, unless a method with the same signature, same return type, and a compatible throws clause is explicitly declared by the interface.

如果一个接口定义是最顶级的(没有 super interfaces),那么这个接口会自动声明一个 abstract member method 结构体来代表所有来自 Object 类(一切类的superclass)中的public方法(包括这些方法的签名、返回类型以及抛出的异常)再换种说法——意思是接口隐含定义了一套与Object类中的方法签名完全相同的方法,所以,我们在程序中调用接口的那些与Object中具有相同签名的方法时,编译器不会报错!好处是让接口型的(interface Type)父类引用也可以拥有Object类的一般方法( toString()、equals() 等)

接口的默认

接口中的成员变量默认由 public static final 来修饰的, 而接口中成员方法前使用 public abstract 来修饰。 image.png

如何跳出当前的多重嵌套循环

为循环打标记,例如

outer:
for(int i = 0;i<10;i++) {
  inner:
  for(int j = 0;j<10;j++) {
      if(j == 5) 
        break outer;
  }
}

活用位运算

恰当的时候使用位运算来做乘法/除法,能提高速度,因为CPU直接支持位运算。

亦或运算

自己跟自己进行亦或运算结果为0。跟全1与运算得到自己,跟全0或运算得到自己

溢出

进行数值运算时要考虑溢出的可能

使用final修饰的变量

final修饰的变量,变量本身的值不能变,但是引用的对象的内容可以改变(不可变类除外)

ArrayList,Vector,LinkedList的存储性能和特性

ArrayList和Vector都是使用数组方式存储数据,此数组元素数大于实际存储的数据以便增加和插入元素,它们都允许直接按序号索引元素,但是插入元素要涉及数组元素移动等内存操作,所以索引数据快而插入数据慢,Vector由于使用了synchronized方法(线程安全),通常性能上较 ArrayList差,而LinkedList使用双向链表实现存储,按序号索引数据需要进行前向或后向遍历,但是插入数据时只需要记录本项的前后项即 可,所以插入速度较快

final, finally, finalize的区别。

final 用于声明属性,方法和类,分别表示属性不可变,方法不可覆盖,类不可继承。finally是异常处理语句结构的一部分,表示总是执行。finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的此方法,可以覆盖此方法提供垃圾收集时的其他资源回收,例如关闭文件等。

Overload和Override的区别。Overload的方法是否可以改变返回值的类型?

方法的重写Overriding和重载Overloading是Java多态性的不同表现。重写Overriding是父类与子类之间多态性的一种表现, 重载Overloading是一个类中多态性的一种表现。如果在子类中定义某方法与其父类有相同的名称和参数,我们说该方法被重写 (Overriding)。子类的对象使用这个方法时,将调用子类中的定义,对它而言,父类中的定义如同被“屏蔽”了。如果在一个类中定义了多个同名的方 法,它们或有不同的参数个数或有不同的参数类型,则称为方法的重载(Overloading)。Overloaded的方法是可以改变返回值的类型。

forward 和redirect的区别

接口是否可继承接口?抽象类是否可实现(implements)接口?抽象类是否可继承具体类(concrete class)?抽象类中是否可以有静态的main方法?

接口可以继承接口。抽象类可以实现(implements)接口,抽象类可以继承具体类。抽象类中可以有静态的main方法。只要记住抽象类与普通类的唯一区别:就是不能创建实例对象和允许有abstract方法(也可以没有)。

native

native方法表示该方法要用另外一种依赖平台的编程语言实现的,不存在着被子类实现的问题,所以,它也不能是抽象的,不能与abstract混用。例如,FileOutputSteam类要硬件打交道,底层的实现用的是操作系统相关的api实现,例如,在windows用c语言实现的,所以,查看jdk的源代码,可以发现FileOutputStream的open方法的定义如下:
private native void open(Stringname) throws FileNotFoundException;
如果我们要用java调用别人写的c语言函数,我们是无法直接调用的,我们需要按照java的要求写一个c语言的函数,用我们的这个c语言函数去调用别人的c语言函数。由于我们的c语言函数是按java的要求来写的,我们这个c语言函数就可以与java对接上,java那边的对接方式就是定义出与我们这个c函数相对应的方法,java中对应的方法不需要写具体的代码,但需要在前面声明native。

try {}里有一个return语句,那么紧跟在这个try后的finally {}里的code会不会被执行,什么时候被执行,在return前还是后

会执行,在return前执行

Iterator和ListIterator的区别是什么?

Iterator可用来遍历Set和List集合,但是ListIterator只能用来遍历List。
Iterator对集合只能是前向遍历,ListIterator既可以前向也可以后向。
ListIterator实现了Iterator接口,并包含其他的功能,比如:增加元素,替换元素,获取前一个和后一个元素的索引,等等。

内部类

内部类可以访问外部类的数据域,因为内部类包含了一个对外部类的引用“外部类.this”;外部类定义了一个内部类,并不意味着每个外部类对象都有一个内部类对象

根据位置不同分为下面几种情况(内部类包含了一个对外部类的引用“外部类.this”):

  1. (普通)成员内部类。类似于外部类的数据域/方法的位置,可以访问外部类的所有数据域和方法,可以用public,private修饰;如果是public修饰,在外面可以这样使用公有内部类,假设A内定义了一个B类,A a = new A(); A.B b = a.new B();

  2. 静态(成员)内部类。跟成员内部类不同的是,有static修饰符,只能访问外部类的静态数据域和静态方法

  3. 局部内部类。定义在外部类的方法内,不可以public,private修饰,而且只有在该方法内可见,对外部隐藏起来,就算是外部类的其他方法也不可见。局部内部类可以访问外部类的所有数据域和方法,但是只能访问所在方法中final修饰的局部变量,不能访问所在方法中的非final局部变量

  4. 匿名内部类。其实是临时类,因为只是在此时需要一个特定对象而建的一个临时类,用完即走,也就用这么一次,所以没必要单独写个类文件,而且实际上是创建了一个内部类对象,new SuperClass(parameters) {....}或者new Interface(){...},接口的时候没有参数

硬链接和软链接

视图: 视图是用户看到并与之交互的界面。视图向用户显示相关的数据,并接受用户的输入。视图不进行任何业务逻辑处理。

模型: 模型表示业务数据和业务处理。相当于JavaBean。一个模型能为多个视图提供数据(比如个人主页,自己模式和游客模式就是不同的视图)。这提高了应用程序的重用性

控制器: 当用户单击Web页面中的提交按钮时,控制器接受请求并调用相应的模型去处理请求。然后根据处理的结果调用相应的视图来显示处理的结果。
MVC的处理过程:首先控制器接受用户的请求,调用相应的模型来进行业务处理,并返回数据给控制器。控制器调用相应的视图来显示处理的结果。并通过视图呈现给用户。

上一篇 下一篇

猜你喜欢

热点阅读