Synchronized总结

来源:互联网 发布:热聊营销软件 编辑:程序博客网 时间:2024/06/14 02:57

线程安全概念:

多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。

synchronized:可以在任意对象及方法上加锁,而加锁的这段代码称为"互斥区"或"临界区"


执行步骤:

当多线程访问一个对象的run方法时,会以排队的方式进行处理(这里的排队是按照CPU分配的先后顺序而定的,不是代码执行顺序)
一个线程想要执行synchronized修饰的方法里的代码:
1、尝试获得锁
2、如果拿到锁,执行synchronized代码体内容,拿不到锁,这个线程会不断的尝试获取锁,直到拿到为止,而且是多个线程同时去
竞争这把锁(锁竞争问题)


synchronized修饰方法:

被关键字synchronized修饰的方法采用的是对象锁,而不是把一段代码(方法)当做锁,
所以代码中哪个线程先执行synchronized关键字修饰的方法,哪个线程就持有该方法所属对象的锁Lock(不同线程调用同一对象的synchronized方法)

同一个对象调用这两个方法的时候能同步,如果是不同的对象,将不能同步

如果想要不同对象同步,那就要加上static,变成类锁


对象锁和类锁:

静态方法static上加synchronized关键字,表示锁定.class类,类一级别的锁(独占.class类),
所以如果要实现多个对象公共一个锁,得用类锁,在方法和变量前加上static,类变量,类方法

也可以采用代码块的方式:

Synchronized(TicketManager.class){ 业务逻辑 }     -->类锁

Sysochronized(this) { 业务逻辑 }       -->对象锁


同步和异步:

synchronized同步:
同步的概念就是共享,我们要牢牢记住"共享"这俩个字,如果不是共享的资源,就没有必要进行同步。
同步的目的就是为了线程安全,其实对于线程安全来说,需要满足俩个特性:
原子性(同步)
可见性

异步:asynchronized
异步的概念就是独立,相互之间不受到任何制约。就好像我们学习http的时候,在页面发起的Ajax请求,我们还可以继续浏览或操作页面的内容,二者之间没有任何关系。

public class MyObject {public synchronized void method1(){try {System.out.println(Thread.currentThread().getName());Thread.sleep(4000);} catch (InterruptedException e) {e.printStackTrace();}}/** 先不加synchronized */public void method2(){System.out.println(Thread.currentThread().getName());}public static void main(String[] args) {final MyObject mo = new MyObject();/** * 分析: * t1线程先持有object对象的Lock锁,t2线程可以以异步的方式调用对象中的非synchronized修饰的方法 * t1线程先持有object对象的Lock锁,t2线程如果在这个时候调用对象中的同步(synchronized)方法则需等待,也就是同步 */Thread t1 = new Thread(new Runnable() {@Overridepublic void run() {mo.method1();}},"t1");Thread t2 = new Thread(new Runnable() {@Overridepublic void run() {mo.method2(); //method2不加synchronized,立刻执行,加上,就要等method1释放对象锁}},"t2");//如果是对象锁(仅仅一个),那么两个线程调用同一个对象的两个synchronized方法,需要等第一个方法调用完,返回锁对象,才能继续下一个方法的调用(同步)t1.start();t2.start();}}

synchronized总结:

看到类里面有synchronized方法修饰,要想到:

1、既然有synchronized,说明要同步,多个线程访问这个方法要排队,这个方法里面必定牵扯到了公共资源(成员变量)的访问

2、判断是对象锁还是类锁,就看有没有static修饰,static修饰说明是静态资源,也就是类成员。

也可以用代码块synchronized(this){ 业务逻辑 }的方式写.

(代码块的方式可以做到优化,如果一个方法前面比较耗时,但不牵扯公共资源访问,这部分代码就没必要加同步,只用synchronized把需要同步的成员变量操作括起来就可以了):


脏读:

setValue()如果是synchronizedgetValue()就也要加synchronized,也就是我在setValue的时候,不允许别的线程进来getValue这样数据才不会脏读

如果是对象锁,如下:

public synchronized void setValue(String username, String password){//xxxx}public synchronized void getValue(String username, String password){//syso();}


注意事项:

不能用字符串常量当成锁

public class ChangeLock {private String lock = "lock";private void method(){synchronized (lock) {try {System.out.println("当前线程 : "  + Thread.currentThread().getName() + "开始");lock = "change lock";Thread.sleep(2000);System.out.println("当前线程 : "  + Thread.currentThread().getName() + "结束");} catch (InterruptedException e) {e.printStackTrace();}}}}


关系型数据库事务的四个特性:

数据库事务必须具备ACID特性。ACID是

Atomic原子性,Consistency一致性:Isolation隔离性,Durability持久性。

原子性:整个事务中的所有操作,要么全部完成,要么全部不完成,不可能停滞在中间某个环节。事务在执行过程中发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。

一致性:在事务开始之前和事务结束以后,数据库的完整性约束没有被破坏。

隔离性:(4种隔离级别)两个事务的执行是互不干扰的,一个事务不可能看到其他事务运行时,中间某一时刻的数据。1.READ_UNCOMMITTED 2.READ_COMMITTED 3.REPEATABLE_READ 4.SERIALIZABLE

持久性:在事务完成以后,该事务所对数据库所作的更改便持久的保存在数据库之中,并不会被回滚。


一致性案例:

9点读取数据,10分钟才获取,一定是获取9点那一时刻的数据,snaptshot(即使中间被dml操作了,数据库会有undo区域,用于回滚),这是会读取undo的数据,而不是dml后的新值


0 0
原创粉丝点击