- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个 SwingWorker,它调用 twitter API 并获取一些结果,每次收到结果时,我首先更新成员变量 _latestValidResult
,然后从结果中获取推文列表并将其添加到我的列表中,然后我发布该列表的大小。然后在发布中我需要使用 _latestValidResult
来访问一些 API 计时器限制,以便我可以更新我的 GUI。
我在更新 _latestValidResult
的 doInBackground()
和使用 _latestValidResult 的
获取计时器限制。process()
中有一个同步块(synchronized block)
我询问是否正确使用了同步块(synchronized block),因为我从 IDE 收到警告,提示我正在同步非最终变量。
这是带有一些伪代码的准系统代码,因此不会让您头疼:
public class ProduceStatusWorker extends SwingWorker<Void, Integer> {
private QueryResult _latestValidResult;
@Override
protected Void doInBackground() {
QueryResult result = null;
Set<Status> allStatuses = new HashSet<>();
do {
try {
if (isCancelled()) {
return null;
}
result = making_api_call_here;
// If the call succeeded (no exception) then copy the result to the _latestValidResult
synchronized (_latestValidResult) {
_latestValidResult = result;
}
allStatuses.addAll(result.getTweets());
publish(allStatuses.size());
} catch (TwitterException te) {
// Handle exceptions here
}
} while (more_statuses_can_be_retrieved);
return null;
}
@Override
protected void process(List<Integer> chunks) {
final int apiCallsTotal;
final int apiCallsLeft;
// Get the variables I need from _latestValidResult
synchronized (_latestValidResult) {
apiCallsTotal = _latestValidResult.getRateLimitStatus().getLimit();
apiCallsLeft = _latestValidResult.getRateLimitStatus().getRemaining();
}
// Update GUI according to the variables
jAPICallsLeftLabel.setText(Integer.toString(apiCallsLeft) + "/" + Integer.toString(apiCallsTotal));
// Update GUI according to the publish
jTweetsInMemory.setText(Integer.toString(chunks.get(chunks.size() - 1)));
}
}
以下是 ProduceStatusWorker
类的完整代码:
public class ProduceStatusWorker extends SwingWorker<Void, Integer> {
private final Query _query;
private QueryResult _latestValidResult;
private static final int QUERY_MAX_COUNT = 100;
public ProduceStatusWorker(Query query) {
_query = query;
_query.setLang("en");
_query.setResultType(ResultType.recent);
_query.setCount(QUERY_MAX_COUNT);
}
@Override
protected Void doInBackground() {
QueryResult result = null;
Set<Status> allStatuses = new HashSet<>();
long lastID = Long.MAX_VALUE;
do {
try {
while (timerIsRunning(_countdown) && !isCancelled()) {
try {
synchronized (this) {
wait(1000);
}
} catch (InterruptedException ex) {
}
}
if (isCancelled()) {
return null;
}
result = DbTools.TWITTER_FACTORY.search(_query);
synchronized (_latestValidResult) {
_latestValidResult = result;
}
allStatuses.addAll(result.getTweets());
for (Status status : result.getTweets()) {
if (status.getId() < lastID) {
lastID = status.getId();
}
}
publish(allStatuses.size());
_query.setMaxId(lastID - 1);
} catch (TwitterException te) {
if (te.getErrorCode() == 88) {
if (!timerIsRunning(_countdown)) {
_countdown = new Timer(1000, new ActionListener() {
private int _count = _latestValidResult != null
? _latestValidResult.getRateLimitStatus().getSecondsUntilReset() : 10;
@Override
public void actionPerformed(ActionEvent e) {
if (_count == 0) {
synchronized (ProduceStatusWorker.this) {
ProduceStatusWorker.this.notifyAll();
}
jTimeLeftLabel.setText(DurationFormatUtils.formatDuration(_count-- * 1000, "mm:ss"));
((Timer) e.getSource()).stop();
} else {
jTimeLeftLabel.setText(DurationFormatUtils.formatDuration(_count-- * 1000, "mm:ss"));
}
}
});
_countdown.start();
}
} else {
cancel(true);
Printer.showError(te);
}
}
} while ((_countdown != null && _countdown.isRunning()) || result.getTweets().size() > 0);
return null;
}
@Override
protected void process(List<Integer> chunks) {
final int apiCallsTotal;
final int apiCallsLeft;
synchronized (_latestValidResult) {
apiCallsTotal = _latestValidResult.getRateLimitStatus().getLimit();
apiCallsLeft = _latestValidResult.getRateLimitStatus().getRemaining();
}
jAPICallsLeftLabel.setText(Integer.toString(apiCallsLeft) + "/" + Integer.toString(apiCallsTotal));
jTweetsInMemory.setText(Integer.toString(chunks.get(chunks.size() - 1)));
}
@Override
protected void done() {
jStart.setSelected(false);
JOptionPane.showMessageDialog(null, "Done!");
}
}
最佳答案
你不会得到想要的效果。仅当所有参与者都使用同一把锁时,锁才有用。对于您的代码,您试图控制的一件事情将会有很多锁。相反,使用什么锁并不重要,只要它是同一个锁即可。
一种简单的方法是进行同步(this)
。但是,推荐的方法是:
private final Object lock = new Object();
//
void method(){
synchronized(lock){
// stuff
}
}
这样不相关的参与者就不会意外使用你的锁(不可否认,这段代码看起来很愚蠢,在我看来是 Java 的一个设计缺陷)。
但是,就您而言,您可以简单地执行以下操作:
volatile QueryResult latestValidResult;
然后就完成了。这是因为 QueryResult 实际上是不可变的(据我猜测),并且因为有多个读取器但只有一个写入器,所以您只需使最新的有效结果对其他线程可见。在这种情况下, volatile
就足够了,IMO更容易理解你想要做什么。
关于java - 我是否正确使用了同步块(synchronized block)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25461226/
在 java 中不使用“同步”关键字的情况下,是否有其他方法可以同步类或方法? 谢谢, 马利卡琼·科卡塔努尔 最佳答案 您可能想查看并发包中引入的对 JDK 5 的更改。 http://java.su
第 1 部分: 假设下面这段代码 void method1(){ synchronized (lockObject){ method2(); System.ou
我有一个 REST 服务器和一个在移动设备上运行的客户端应用程序。客户端有一些数据并希望从服务器获取数据更新。如何以 RESTful 方式在单个事务中执行此操作? 假设客户有以下元素: widge
我有一个多线程 Java 应用程序。在一种方法中,需要同步一个 ArrayList。由于 arrayList 不是线程安全的,所以我必须使用同步。问题是 ArrayList 类型的对象不是对象的成员变
我正在阅读 Android 示例中的 BluetoothChatService.java 文件,有一件事特别让我感到困惑。 方法在多个位置访问静态成员,并且定义为同步。 在另一部分中,正在访问同一个静
我知道为了实现线程安全和同步,我们使用同步块(synchronized block)或方法。 但我无法理解声明- “Java 中的同步块(synchronized block)在某些对象上同步 ” 任
在 Scala 中使用 JDBC 的示例中,有以下代码: this.synchronized { if (!driverLoaded) loadDriver() } 为什么this.synchro
abstract class A { protected abstract int isRunning(); public void concreteMethod() { synchr
有谁可以分享一下他们的经验吗?“我们什么时候在同步方法和同步块(synchronized block)之间进行调用”有任何性能问题吗? 最佳答案 When do we make a call to u
这是我之前问题的后续问题,Is this variable being safely accessed by using synchronization? 对于下面的程序, Class SubClas
我目前正在为 N 体问题实现多线程版本的 Barnes-Hut 算法。虽然该算法有效,但它不是很优化,我正在尝试减少我的程序的运行时间。 我已经确保有多个线程可以准确地找到我正在使用的空间的边界,并意
我有这门课: public class MyClass { public MyClass(){} public void actionA(){ synchronized
又是一个关于ArrayList和synchronize的问题。 我只想知道这段代码到底做了什么: ArrayList list = ....; synchronized (list) { if
我可以在另一个同步块(synchronized block)中包含同步块(synchronized block)以同步另一个对象吗? 例子: synchronized(myObjetc1){
public class ObjectCounter { private static long numOfInstances = 0; public ObjectCounter(){
我在某处读到,对于 various reasons 应该避免 synchronized(this) .然而,我遇到的一些值得尊敬的代码在构造函数中使用了以下内容: public SomeClass(C
Java 为同步代码的关键部分提供了一种非常方便的习惯用法: synchronized(someObject) { // do something really important all b
我有一个 WeakReference 的 Collections.synchronizedList,_components; 我写了类似下面的内容,希望编译者会提示: public boolean a
使用下面两个版本的Singleton Classes有什么区别 首先我使用的是synchronized(Singleton.class) 在第二个我使用同步(Obj)//第一种类型 公共(public
我正在查看 DatagramSocket 的源代码,我发现了这个: public void disconnect() { synchronized (this) { if (i
我是一名优秀的程序员,十分优秀!