org.redisson.misc.ReclosableLatch Maven / Gradle / Ivy
Show all versions of redisson Show documentation
/**
* Copyright (c) 2013-2024 Nikita Koksharov
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* Copyright 2009 - 2017, Red Hat Inc. and/or its affiliates.
*
* All files in this repository or distribution are licensed under the
* Apache License, Version 2.0 (the "License");
* you may not use any files in this repository or distribution except
* in compliance with the License.
*
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.redisson.misc;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;
/**
* A thread gate, that uses an {@link java.util.concurrent.locks.AbstractQueuedSynchronizer}.
*
* This implementation allows you to create a latch with a default state (open or closed), and repeatedly open or close
* the latch.
*
* @author Manik Surtani ([email protected])
* @since 4.0
*/
@SuppressWarnings({"MultipleVariableDeclarations", "AvoidInlineConditionals", "UpperEll"})
public class ReclosableLatch extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = 1744280161777661090l;
// the following states are used in the AQS.
private static final int OPEN_STATE = 0, CLOSED_STATE = 1;
public ReclosableLatch() {
setState(CLOSED_STATE);
}
public ReclosableLatch(boolean defaultOpen) {
setState(defaultOpen ? OPEN_STATE : CLOSED_STATE);
}
@Override
public final int tryAcquireShared(int ignored) {
// return 1 if we allow the requestor to proceed, -1 if we want the requestor to block.
return getState() == OPEN_STATE ? 1 : -1;
}
@Override
public final boolean tryReleaseShared(int state) {
// used as a mechanism to set the state of the Sync.
setState(state);
return true;
}
public final void open() {
// do not use setState() directly since this won't notify parked threads.
releaseShared(OPEN_STATE);
}
public final void close() {
// do not use setState() directly since this won't notify parked threads.
releaseShared(CLOSED_STATE);
}
public boolean isOpened() {
return getState() == OPEN_STATE;
}
// waiting for an open state
public final void await() throws InterruptedException {
acquireSharedInterruptibly(1); // the 1 is a dummy value that is not used.
}
public final void awaitUninterruptibly() {
try {
await();
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}
public final boolean await(long time, TimeUnit unit) throws InterruptedException {
return tryAcquireSharedNanos(1, unit.toNanos(time)); // the 1 is a dummy value that is not used.
}
@Override
public String toString() {
int s = getState();
String q = hasQueuedThreads() ? "non" : "";
return "ReclosableLatch [State = " + s + ", " + q + "empty queue]";
}
}