- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在研究避免死锁的措施,其中一种可能的方法是通过强制线程放弃它在访问另一个锁但无法访问该锁时已经持有的锁来打破循环等待。
以最简单的银行账户转账为例:
class Account {
private int balance;
void transfer(Account target, int amt){
//lock the from account
synchronized(this) {
//lock the to account
synchronized(target) {
if (this.balance > amt) {
this.balance -= amt;
target.balance += amt;
}
}
}
}
}
使用嵌套同步块(synchronized block)时,当一个线程持有当前账户的监视器时,当它无法访问目标账户的锁时,它会被阻塞,直到锁可用,并且无法响应任何中断。它没有机会先尝试第二个锁是否可用,这样就没有机会放弃它已经持有的锁。
tryLock()
可以无缝支持场景的方法
lockInterruptibly()
这意味着当它由于无法访问第二个锁而被阻塞时,它仍然可以响应中断。但我想知道的是,当被中断时,是什么确保线程会丢弃它已经拥有的所有锁?因为我没有找到像
Object.wait()
这样的文档其中有文档明确表示它将放弃它所拥有的所有同步。除了
Thread.interrupt()
只是调用一个我看不到任何实现的 native 方法。
Object.wait()
时犯了一个错误。在无法访问监视器时执行此操作。我已经尝试为这种情况制定一个锁定解决方案,但我不确定它是否有一些谬误。如果代码中有错误,希望有人能指出。
class Account {
private int balance;
Lock lock = new ReentrantLock();
void transfer(Account target, int amt) {
//lock the from account
try {
if (lock.tryLock(50, TimeUnit.MILLISECONDS)) {
if (target.lock.tryLock(50, TimeUnit.MILLISECONDS)) {
this.balance -= amt;
target.balance += amt;
target.lock.unlock();
}
lock.unlock();
}
} catch (InterruptedException e) {
System.out.println("Interrupted while attempting the lock.");
}
}
}
如果有人可以使用
lockInterruptibly()
提供有关如何处理这种情况的示例,我将不胜感激。 .
最佳答案
您使用 tryLock
的方法正朝着无死锁解决方案的方向发展。但是您必须确保在特殊情况下正确关闭锁。此外,由于 tryLock
可能会失败,导致 Action 没有被执行,你需要返回一个状态:
class Account {
private int balance;
Lock lock = new ReentrantLock();
boolean transfer(Account target, int amt) {
boolean success = false;
//lock the from account
try {
if(lock.tryLock(50, TimeUnit.MILLISECONDS)) try {
if(target.lock.tryLock(50, TimeUnit.MILLISECONDS)) try {
this.balance -= amt;
target.balance += amt;
success = true;
}
finally {
target.lock.unlock();
}
}
finally {
lock.unlock();
}
} catch(InterruptedException ex) {
// success still false
}
return success;
}
}
然后,您必须考虑如何处理失败,重试操作或放弃。这导致了另一个问题,即所谓的活锁问题。当一个线程尝试从 A 传输到 B 而另一个线程尝试从 B 传输到 A 时,两者都可能成功锁定其源,然后又无法锁定另一个。重复尝试可能会反复导致失败。涉及更多线程的类似场景是可能的。在这些情况下,不会发生死锁,但线程仍然无法取得进展。
class Account {
final BigInteger accountNumber;
private int balance;
Lock lock = new ReentrantLock();
Account(BigInteger accountNumber) {
this.accountNumber = accountNumber;
}
void transfer(Account target, int amt) {
if(accountNumber.compareTo(target.accountNumber) < 0)
transfer(this, target, amt);
else
transfer(target, this, -amt);
}
static void transfer(Account from, Account to, int amt) {
from.lock.lock();
try {
to.lock.lock();
try {
from.balance -= amt;
to.balance += amt;
}
finally {
to.lock.unlock();
}
}
finally {
from.lock.unlock();
}
}
}
通过总是先锁定具有较小数字的帐户,我们永远不会遇到拥有较大数字锁的另一个线程试图获取我们已经拥有的锁的情况。
关于java - 使用 java.util.concurrent.locks.Lock 而不是同步的 : can my code avoid dead-lock in a bank transfer scenario?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62668648/
我正在尝试将网页内容打印到一页纸上。但是,它将内容分成 2 页,所以我在这里做了一些研究,看到有人推荐: #my_print_div{ width:940px; height:770px; page
我目前正在打印一些东西。我有一个动态页面,其中包含可变数量的 block 级元素。有些可能是 1 行,有些可能是 100 多行。 1text 1 line.... 2text 10 lines....
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
我正在训练一个 randomForest 模型,目的是保存它以进行预测(它将被下载并在外部上下文中使用)。我希望这个模型尽可能最小。 我读到有很多options和 packages减少模型的内存大小。
这个问题在这里已经有了答案: MySQL connection timeout (3 个答案) 关闭 9 年前。 我一直在尝试使用 Tomcat 的 native 连接池功能来避免我的 Java W
我正在使用 Phonegap/Cordova 开发 Android 应用程序。我已经按照这样的百分比安排了我的布局(在 CSS 中): 标题 - 50px; Content_row1 - 30%(剩下
我正在编写一个插件,它将表情符号转换为特定站点文本 block 中的图像。简单的答案是使用正则表达式检测 innerHTML 上的触发文本并插入 img 标签,然后将字符串通过管道返回到 innerH
如何避免在我的 Drupal View 上重复? 我应该添加一个过滤器,指定特定字段(即用户 ID)不应出现两次吗?我找不到这样的选项 看法 http://dl.dropbox.com/u/72686
感谢您查看我的 typescript 问题。 为简单起见,我对 typescript “过度属性检查”行为有疑问。我想确保 TypeScript 不接受具有额外属性的对象。 在我的简单界面示例中,OF
我发现对于某些图表,我从 Prometheus 获得了 doubles 值,其中应该只是一个: 我使用的查询: increase(signups_count[4m]) 抓取间隔设置为 recommen
假设我正在运行N个线程。 每个线程都需要与下一个和上一个同步。 for (i = 0 ; i < NITER; i++){ do_something (); sync_
如今,服务器虚拟化是一件大事,所以我的任务是在虚拟化服务器上安装我们的一些软件,看看会发生什么。长话短说:rsync 传输会立即使虚拟化服务器崩溃。虚拟化主机是一台强大的机器,没有其他负载;我认为
以下正则表达式在应用于大型 html 页面时会创建 StackOverflowError: (.|\s)*? 我的假设是,这是由于逻辑“OR”运算符(|)在匹配器中创建了递归调用,并且由于需要解析的
我在运行时使用表达式树构建委托(delegate): Type type = GetType(); ParameterExpression parameterType = Expression.Par
我正在使用 scikit-learn TfidfVectorizer 找出两个文档中最重要的单词。每个文档大小为 1.9GB(约 9000 万字),并且已采用小写、词干化(使用 nltk.stem.p
我进行了一个中间件调用来获取 String 数组,如下所示: String[] freqwords = MViewer.getWordNames(); 问题是可能没有可用数据,因此任何进一步的操作(如
在 JavaFx 中,我使用以下代码创建一个 StackedBarChart: String[] ACTIVITIES = new String[10]{ ... };// there
我正在尝试制作一个使用类 AnimationTimer 来处理它的游戏。我的代码摘要如下所示: 主类 object Game extends JFXApp{ def showMenu{
我正在用不同的步骤创建一个小的 javascript/jQuery 应用程序。为此,我使用了一个具有不同功能的 js 文件。 在文件的顶部我调用了我的第一个函数。在我的第一个函数中,我在单击按钮时调用
我正在使用表格 View 来显示从服务器加载的文本字段数组,所以我有一个表格 View 字段列表,当我填充这些数据字段并向下滚动以填充其他字段时,当我再次向上滚动时,我发现值发生变化并且存在重复值 -
我是一名优秀的程序员,十分优秀!