【Java】资源共享有冲突

2024-08-26 02:18
文章标签 java 冲突 资源共享

本文主要是介绍【Java】资源共享有冲突,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      开发过程中经常会遇到并发处理某共享数据时,产生不一致的情况,如何解决呢?方案是-----加锁。

一、对线程加锁

      对线程加锁,就是利用Java提供的synchronized关键字。

【修饰一个代码块】

import java.lang.*;
/*** 线程同步Demo* @author 郑艳霞**/
public class Test implements Runnable {public void run() {synchronized (this) {for (int i = 0; i < 4; i++) {try {System.out.println("当前执行的是:"+Thread.currentThread().getName());Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}}// 测试类public static void main(String[] args) {Test test = new Test();Thread thread1 = new Thread(test, "线程一");Thread thread2 = new Thread(test, "线程二");thread1.start();thread2.start();}
}
    运行结果为:


      访问synchronized同步代码块时,其他线程将被阻塞,因为执行synchronized代码块时,会锁定当前对象,只有释放对该对象的锁,下一个线程才能执行。
      补充:synchronized只锁定对象,每个对象都只对应一个锁。如果在测试类中重新再声明一个对象,会证明相同的结论。(不要被单纯的运行结果欺骗哦)

【修饰一个方法】
      其实在上面例子中就等同于是修饰一个方法。简单说,修饰代码块,是指将synchronized(this){//代码块},但是当这段代码块就是整个方法的时候,那就是在修饰一个方法了呗。这个时候就有了另一种写法,即:直接将synchronized关键字放在方法名前面。如下:
public synchronized void run() {//synchronized (this) {for (int i = 0; i < 4; i++) {try {System.out.println("当前执行的是:"+Thread.currentThread().getName());Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}
//}
      运行结果当然也是一样的,就不在赘述。
【修饰一个类】
       跟上面一样,synchronized(this)锁住的其实就是this的范围,当this只是一个类中的一个方法中的一小段代码时,那锁的范围就是这段代码,这段代码中的对象在同一时刻只能被一个线程调用。如果this可以等同于一个方法,那锁的范围就是一个方法,这段代码中的对象在同一时刻只能被一个线程调用。如果范围是一个类,如下写法:当synchronized(test.class),那锁住的就是一个类,类中所有的对象同用一把锁。

二、对资源加锁

      对资源进行加锁,就是从数据库方面进行控制。悲观锁是基于数据库锁机制使用的。

【悲观锁(Pessimistic Lock)

      悲观锁认为每次拿数据的时候都会有人来修改,认为肯定会出现影响数据完整性一致性的问题产生,所以每次拿数据的时候都会加锁。即,在整个事务处理过程中,数据都处于锁定状态。

这条sql语句锁定了t_table_id表中所有符合检索条件的记录,在这次查询事务提交之前,这些记录都无法被其他人修改。直至本次事务提交,锁释放。
【乐观锁(Optimistic Lock)

      乐观锁认为每次去拿数据都不会有人来修改,所以不加锁。但是在更新时会判断一下再此期间是否有人更新这些数据,所以用到了数据库的版本记录机制,通过version字段来判断是否是过期数据。(但是乐观锁不能解决读脏数据问题)

【比较】
      相对于悲观锁,乐观锁机制采取了更加宽松的加锁机制。悲观锁大多数情况下依靠数据库的锁机制实现,来保证操作最大程度的独占性。但是如果某次事务时间特别长,还使用悲观锁的话,将会导致严重后果。乐观锁在一定程度上解决了这个问题。

【适用】
    乐观锁适用于写比较少的情况,即很少发生冲突的情况,这样就可以省去锁开销,加大了系统的整个吞吐量。
    悲观锁适用于更改频繁的数据表,一开始查询就加锁,直至更新操作结束才释放,但是性能有所下降。
【悲观锁的实现】

package com.bjpowernode.drp.util;import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;/***  Id生成器* @author happy**/
public class IdGenerator {public static int generate(String tableName) {int value = 0;Connection conn = DbUtil.getConnection();PreparedStatement pstmt = null;ResultSet rs = null;//悲观锁String sql = "select value from t_table_id where table_name=? for update";// 开始事务DbUtil.beginTransaction(conn);try {pstmt = conn.prepareStatement(sql);pstmt.setString(1, tableName);rs = pstmt.executeQuery();if (!rs.next()) {throw new RuntimeException();}value = rs.getInt("value");value++;modifyValueField(conn, tableName, value);// 提交事务--释放锁DbUtil.commitTransaction(conn);} catch (SQLException e) {e.printStackTrace();// 回滚事务--释放锁DbUtil.rollbackTransaction(conn);throw new RuntimeException();} finally {DbUtil.close(rs);DbUtil.close(pstmt);DbUtil.resetConnection(conn);// 重置Connection的状态DbUtil.close(conn);}return value;}/*** 根据表名更新序列字段的值* * @param conn* @param tableName* @param value* @throws SQLException*/private static void modifyValueField(Connection conn, String tableName,int value) throws SQLException {String sql = "update t_table_id set value=? where table_name=?";PreparedStatement pstmt = null;try {pstmt = conn.prepareStatement(sql);pstmt.setInt(1, value);pstmt.setString(2, tableName);pstmt.executeUpdate();} finally {DbUtil.close(pstmt);}}public static void main(String[] args) {int retValue = IdGenerator.generate("t_client");System.out.println(retValue);}
}


这篇关于【Java】资源共享有冲突的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1107270

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定