烟台市铁路建设管理局网站,wordpress简约下载站模板下载,域名空间商,ipv6改造wordpress主要学习那些场景需要额外注意线程安全问题#xff0c;在这里总结了四中场景。
访问共享变量或资源
第一种场景是访问共享变量或共享资源的时候#xff0c;典型的场景有访问共享对象的属性#xff0c;访问static静态变量#xff0c;访问共享的缓存#xff0c;等等。因为…主要学习那些场景需要额外注意线程安全问题在这里总结了四中场景。
访问共享变量或资源
第一种场景是访问共享变量或共享资源的时候典型的场景有访问共享对象的属性访问static静态变量访问共享的缓存等等。因为这些信息不仅会被一个线程访问到。还有可能被多个线程同时访问那么就有可能在并发读写的情况下发生线程安全问题。比如我们上一课讲过的多线程同时i的例子 如代码所示两个线程同时对 i 进行 i 操作最后的输出可能是 15875 等小于20000的数而不是我们期待的20000这便是非常典型的共享变量带来的线程安全问题。
依赖时序的操作
第二个需要我们注意的场景是依赖时序的操作如果哦我们的操作的正确性是依赖时序的而在多线程的情况下又不能保障执行的顺序和我们预想的一致这个时候就会发生线程安全的问题如下面的代码所示 代码中首先检查 map 中有没有 key 对应的元素如果有则继续执行 remove 操作。此时这个组合操作就是危险的因为它是先检查后操作而执行过程中可能会被打断。如果此时有两个线程同时进入 if() 语句然后它们都检查到存在 key 对应的元素于是都希望执行下面的 remove 操作随后一个线程率先把 obj 给删除了而另外一个线程它刚已经检查过存在 key 对应的元素if 条件成立所以它也会继续执行删除 obj 的操作但实际上集合中的 obj 已经被前面的线程删除了这种情况下就可能导致线程安全问题。
类似的情况还有很多比如我们先检查x1如果x1就修改x的值代码如下所示
这样类似的场景都是同样的道理“检查与执行”并非原子性操作在中间可能被打断而检查之后的结果也可能在执行时已经过期、无效换句话说获得正确结果取决于幸运的时序。这种情况下我们就需要对它进行加锁等保护措施来保障操作的原子性。
不同数据之间存在绑定关系
第三种需要我们注意的线程安全场景是不同数据之间存在相互绑定关系的情况。有时候我们的不同数据之间是成组出现的存在着相互对应或绑定的关系最典型的就是 IP 和端口号。有时候我们更换了 IP往往需要同时更换端口号如果没有把这两个操作绑定在一起就有可能出现单独更换了 IP 或端口号的情况而此时信息如果已经对外发布信息获取方就有可能获取一个错误的 IP 与端口绑定情况这时就发生了线程安全问题。在这种情况下我们也同样需要保障操作的原子性。
对方没有声明自己是线程安全的
第四种值得注意的场景是在我们使用其他类时如果对方没有声明自己是线程安全的那么这种情况下对其他类进行多线程的并发操作就有可能会发生线程安全问题。举个例子比如说我们定义了 ArrayList它本身并不是线程安全的如果此时多个线程同时对 ArrayList 进行并发读/写那么就有可能会产生线程安全问题造成数据出错而这个责任并不在 ArrayList因为它本身并不是并发安全的正如源码注释所写的 Note that this implementation is not synchronized. If multiple threads access an ArrayList instance concurrently, and at least one of the threads modifies the list structurally, it must be synchronized externally. 这段话的意思是说如果我们把 ArrayList 用在了多线程的场景需要在外部手动用 synchronized 等方式保证并发安全。
所以 ArrayList 默认不适合并发读写是我们错误地使用了它导致了线程安全问题。所以我们在使用其他类时如果会涉及并发场景那么一定要首先确认清楚对方是否支持并发操作以上就是四种需要我们额外注意线程安全问题的场景分别是访问共享变量或资源依赖时序的操作不同数据之间存在绑定关系以及对方没有声明自己是线程安全的。