多线程的小知识点

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中获得事务的上下文即可。





  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值