多线程的小知识点

来源:互联网 发布:网络延长器是一对吗 编辑:程序博客网 时间:2024/05/29 07:50

1、++i不是原子性的,它是由三个步骤组成的,获得当前值,加1,写回新值。假设当前i的值为9,如果两个线程都同时读取i值,读到的都是9,然后同时加1,那么这时i的值变为10,但其实是需要加两次,值是11的。

2、重进入:当一个线程请求其他线程已经占有的锁时,请求线程将被阻塞,然后内部锁是可重进入的,因此线程在试图获得它自己占有的锁时,请求会成功。重进入的实现是通过为每个锁关联一个请求计数和一个占有它的线程,。当计数为0,认为它的锁是未被占有的,线程请求一个未被占有的锁时,jvm将记录锁的占有者,并将请求计数为1.如果同一线程再次请求这个锁,计数将递增,每次占有线程退出同步块,计数器值将递减,直到计数器达到0时,锁被释放

public class Widget {public synchronized void doSomething(){System.out.println("widget:"+this);}}
public class LoggingWidget extends Widget{public synchronized void doSomething(){System.out.println("LoggingWidget:"+this);/** * 这样子调用的话父类方法锁还是子类的对象 * 也就是子类调用了父类的方法 * 如果没有重入锁,代码将死锁 */super.doSomething();}public static void main(String[] args) {new LoggingWidget().doSomething();/** * LoggingWidget:com.test.LoggingWidget@c17164 * widget:com.test.LoggingWidget@c17164 */}}
3、线程锁的一个作用是同步,另一个作用是可见性。

4、ThreadLocal你可以将它看作map<Thread,T>它存储了于线程相关的值,不过事实上它并非是这样实现的,与线程相关的值存储在线程对象自身中,线程终止后,这些值会被垃圾回收。下面是jdk源码:

    public T get() {        Thread t = Thread.currentThread();        ThreadLocalMap map = getMap(t);        if (map != null) {            ThreadLocalMap.Entry e = map.getEntry(this);            if (e != null)                return (T)e.value;        }        return setInitialValue();    }    ThreadLocalMap getMap(Thread t) {        return t.threadLocals;    }
它有一种简单的实现,它利用静态ThreadLocal持有事务上下文,当框架代码需要获知当前正在运行的是哪个事务时,只要从ThreadLocal中获得事务的上下文即可。





原创粉丝点击