关于java基础的一些东西

 最近觉得自己基础方面忘了很多东西,总结 一下

 

 一、关于java类加载器。

      一般会知道java的类加载器是ClassLoader,但是ClassLoader也是java类编写,谁来加载它?肯定不可能是java了,它是

C++编写的BootStrap.BootStrap是嵌套在java虚拟机内核中的,jvm启动的时候它就会启动。

总的来说java虚拟机中默认主要有三个类加载器:BootStrap,ExtClassLoader,AppClassLoader。

当Java虚拟机要加载一个类时,到底该派哪个类加载器去加载呢?

(1) 首先是当前线程的类加载器去加载线程中的第一个类。

(2) 如果类A中引用了类B,Java虚拟机将使用加载类A的类加载器来加载类B。

(3) 还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。

 

二、Java中stack和heap的区别

java中内存分为两类:stack(堆栈)和heap(堆);

stack是程序内存空间,因此所有基本类型和对象的引用都在stack中

heap是java虚拟机存储对象的,它是一个巨大的内存。当创建一个对象的时候,java虚拟机将对象的引用放在stack中,创建的对象

放在heap中。

所以,基本类型和对象的引用在stack中,对象在heap中。

 

三、java中synchronized加在类上和方法上的区别

     java的synchronized可以加载方法上,也可以加在对象上,从而一段代码只会有一个线程在运行,保证线程同步。是 一种“时间换空间”的做法。

  

public class SyncTest {
public static synchronized void testSyncOnStaticMethod() {

System.out.println(“testSyncOnStaticMethod”);

try {

Thread.sleep(10000);

} catch (InterruptedException e) {

}

}

public static void testSyncOnClass() {

synchronized (SyncTest.class) {

System.out.println(“testSyncOnClass”);

try {

Thread.sleep(10000);

} catch (InterruptedException e) {

}

}

}

public synchronized void testSyncOnMethod() {

System.out.println(“testSyncOnMethod”);

try {

Thread.sleep(10000);

} catch (InterruptedException e) {

}

}

public void testSyncOnThis() {

synchronized (this) {

System.out.println(“testSyncOnThis”);

try {

Thread.sleep(10000);

} catch (InterruptedException e) {

}

}

}

public static void case1() {

// case1

// 先输出testSyncOnThis或者testSyncOnMethod

// 然后停顿10秒,再输出另一个

// 这个现象表明了

 

// public synchronized void func() {

// }

 

// 等价于

 

// public void func() {

// synchronized (this) {

// }

// }

final SyncTest t1 = new SyncTest();

new Thread(new Runnable() {

@Override

public void run() {

t1.testSyncOnThis();

}

}).start();

new Thread(new Runnable() {

@Override

public void run() {

t1.testSyncOnMethod();

}

}).start();

}

public static void case2() {

// case2

// 先输出testSyncOnClass或者testSyncOnStaticMethod

// 然后停顿10秒,再输出另一个

// 这个现象表明了

 

// public synchronized static void staticFunc() {

// }

 

// 等价于

 

// public static void staticFunc() {

// synchronized (SyncTest.class) {

// }

// }

new Thread(new Runnable() {

@Override

public void run() {

SyncTest.testSyncOnClass();

}

}).start();

new Thread(new Runnable() {

@Override

public void run() {

SyncTest.testSyncOnStaticMethod();

}

}).start();

}

public static void main(String[] args) {

case1();

case2();

}

}

 从上面的代码我们可以看出synchronized加在方法上本质上还是等价于加在对象上的。

如果synchronized加在一个类的普通方法上,那么相当于synchronized(this)。

如果synchronized加载一个类的静态方法上,那么相当于synchronized(Class对象)。

在使用多线程的时候,知道这个是很关键的,因为synchronized的两种不用用法可能导致两段不相干的代码是互斥的,增加了同步的开销(例如这里的函数testSyncOnThis和testSyncOnMethod,他们在同一个对象this上加了锁),更严重的是可能导致死锁。

你可能感兴趣的:(java基础)