1:java并发的分类,在实际开发中我一般讲java并发分为2个大类
1)业务并发,也叫程序并发,比如同时打开两个相同的网页.然后同时点了2个相同的按钮.(也就是java中的多线程)
2)数据并发,数据库中某个表中的某条记录,被多个程序同时使用.其中给一个程序对数据进行了修改,会影响另外一个程序对该数据进行操作,这个一般涉及到的数据库隔离级别,
数据库事务的隔离级别有4个,由低到高依次为Read uncommitted、Read committed、Repeatable read、Serializable,这四个级别可以逐个解决脏读、不可重复读、幻读这几类问题。
√: 可能出现 ×: 不会出现
脏读 | 不可重复读 | 幻读 | |
Read uncommitted | √ | √ | √ |
Read committed | × | √ | √ |
Repeatable read | × | × | √ |
Serializable | × | × | × |
2:java并发的解决办法.
1)对于我上面的分类,第一种java并发问题,我一般是通过锁来解决,最喜欢在java的方法上面加上synchronized关键字,在方法内部做条件判断,如果不满足某一条件,抛出异常.
比如,创建订单的时候,一般不允许重复创建订单,那么在创建订单的方法中加上synchronized关键字后,每次只有一个线程进入该方法,然后,在该方法中加上查询订单的方法,如果订单可以查询到,抛出异常,不再允许创建订单了.反之,如果订单查询不到,那么可以允许创建订单.
所以通过加锁来控制并发问题,一般都要在锁中做条件判断.不然加了锁也没什么用,程序只是顺序执行罢了,没有抛异常的操作,所有的锁都是没用,相反只会让程序变得更慢,毕竟现在都支持多线程,而电脑都是多核的处理器.
2)对于上面的第二种分类,涉及到数据的隔离级别,一般在开发中将数据的隔离级别设置为
Read committed,这样可以防止脏读,然后通过数据库的时间戳来,或者程序时间戳来解决不可重复读问题.