- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在开始 Chủ đề
từ ServletContextListener
当上下文被初始化并试图在上下文被销毁时停止它。该类是:
public enum BlinkLedTask {
VÍ DỤ;
private Logger logger = RpiLogger.getLogger(getClass());
private Task task;
private ExecutorService service;
private BlinkLedTask() {
}
public void run(String[] frequency) {
dừng lại();
task = new Task(frequency);
service = Executors.newSingleThreadExecutor(RpiThreadFactory.INSTANCE);
service.execute(task);
}
public void stop() {
if(Objects.isNull(task) || Objects.isNull(service)) {
trở lại;
}
thử {
task.terminate();
service.shutdownNow();
} catch (Exception cause) {
logger.error(cause.getMessage(), cause);
}
}
private static class Task implements Runnable {
private volatile boolean running = true;
private String[] frequency;
private volatile Logger logger = RpiLogger.getLogger(getClass());
private Task(String[] frequency) {
this.frequency = frequency;
}
@Ghi đè
công khai void run() {
while(running && !Thread.interrupted()) {
thử {
resetLed();
blinkLed();
} catch (Throwable cause) {
logger.error(cause.getMessage(), cause);
running = false;
thử {
resetLed();
} catch (Throwable ignore) {
}
}
}
}
private void resetLed() throws Exception {
executeScript(Script.BLINK_LED_RESET);
}
private void blinkLed() throws Exception {
executeScript(Script.BLINK_LED, new String[]{frequency[0], frequency[1], frequency[2]});
}
private void executeScript(Script script, String... args) {
ScriptExecutor scriptExecutor = new ScriptExecutor(ScriptExecutor.BASH, script);
scriptExecutor.execute(true, args);
}
private void terminate() {
logger.info("Stopping - " + Thread.currentThread().getName());
running = false;
}
}
}
đây là mộtSingleton
并且它运行一个 shell 脚本直到它被停止。这个类可以从任何地方调用,所以我需要停止线程,如果有任何当前正在执行 shell 脚本,然后再创建一个新的 Chủ đề
。 .
出于测试目的,我执行了 run()
初始化上下文并调用 stop()
时此类的方法在毁灭的时候。
删除代码后我重新部署了 war 文件 run()
, 我期待 stop()
将终止 task
,但它没有。
Tôi cũng đã thử run()
的不同实现方式和 stop()
:
public void run(String[] frequency) {
dừng lại();
task = new Task(frequency);
Thread thread = RpiThreadFactory.INSTANCE.newThread(task);
tasks.add(ImmutablePair.of(thread, task));
thread.bắt đầu();
}
public void stop() {
for(ImmutablePair pair : tasks) {
thử {
pair.right.terminate();
pair.left.join();
} catch (Ngoại lệ ex) {
}
}
}
这里是 tasks
Đúngprivate ArrayList<>
. ImmutablePair
属于commons-lang3。但是我收到了java.util.ConcurrentModificationException
关于增强的 for 循环的迭代。原因我不知道。
当服务器关闭时 stop()
正在按预期工作。我正在使用 Jetty。
RpiThreadFactory
:
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;
import org.apache.log4j.Logger;
import com.edfx.rpi.app.utils.logger.RpiLogger;
public enum RpiThreadFactory implements ThreadFactory {
VÍ DỤ;
private final AtomicInteger poolNumber = new AtomicInteger(1);
private final Logger logger = RpiLogger.getLogger(getClass());
private final ThreadGroup threadGroup;
private final AtomicInteger threadNumber = new AtomicInteger(1);
private final String namePrefix;
private RpiThreadFactory() {
SecurityManager securityManager = System.getSecurityManager();
threadGroup = (securityManager != null) ? securityManager.getThreadGroup() : Thread.currentThread().getThreadGroup();
namePrefix = "RpiPool-" + poolNumber.getAndIncrement() + "-Thread-";
}
public Thread newThread(Runnable runnable) {
Thread thread = new Thread(threadGroup, runnable, namePrefix + threadNumber.getAndIncrement(), 0);
thread.setPriority(Thread.NORM_PRIORITY);
thread.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
public void uncaughtException(Thread thread, Throwable cause) {
logger.error(cause.getMessage(), cause);
}
});
return thread;
}
}
ScriptExecutor
:
import java.io.BufferedReader;
import java.io.InputStream;
nhập java.io.InputStreamReader;
nhập java.util.ArrayList;
nhập java.util.Arrays;
nhập java.util.List;
import java.util.Objects;
import org.apache.commons.lang3.StringUtils;
import org.apache.log4j.Logger;
import com.edfx.rpi.app.utils.logger.RpiLogger;
public class ScriptExecutor {
private static final Logger LOGGER = RpiLogger.getLogger(ScriptExecutor.class);
public static final String BASH = "/bin/bash";
private Script script;
private Process process;
private String output;
private int exitValue;
public ScriptExecutor(Script script) {
this.script = script;
}
public void execute(boolean destroyProcess, String... args) throws ScriptNotExistException {
if(!script.exists()) {
throw new ScriptNotExistException(script.getScriptName() + " does not exists.");
}
thử {
List commands = new ArrayList<>();
commands.add(BASH);
commands.add(script.getAbsoultePath());
if(Objects.nonNull(args)) {
commands.addAll(Arrays.asList(args));
}
StringBuilder builder = new StringBuilder("Executing script: ");
builder.append(script.getScriptName());
if(Objects.nonNull(args) && args.length > 0) {
builder.append(" with parameters: ");
builder.append(StringUtils.join(args, " "));
}
LOGGER.info(builder.toString());
ProcessBuilder processBuilder = new ProcessBuilder(commands.toArray(new String[commands.size()]));
process = processBuilder.start();
StringBuilder outputBuilder = new StringBuilder();
InputStream inputStream = process.getInputStream();
InputStreamReader inputStreamReader = new InputStreamReader(inputStream);
BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
String line = StringUtils.EMPTY;
while ((line = bufferedReader.readLine()) != null) {
outputBuilder.append(line);
outputBuilder.append("\n");
}
process.waitFor();
exitValue = process.exitValue();
LOGGER.info("Process for: " + script.getScriptName() + " is executed. Exit value: " + exitValue);
if(destroyProcess) {
destroyProcess();
}
output = outputBuilder.toString();
} catch (Exception cause) {
throw new ScriptExecutionException(cause);
}
}
public String getOutput() {
return output;
}
public int getExitValue() {
return exitValue;
}
public void destroyProcess() {
if(Objects.nonNull(process)) {
LOGGER.info("Process for: " + script.getScriptName() + " is destroyed.");
process.destroy();
}
}
}
这是一个在 Jetty 网络容器中运行的网络应用程序。服务器安装在启用了 java 的嵌入式硬件中。该硬件如何连接 LED。该应用程序接受外部请求,可以是 REST 并启动和停止 LED。因此 LED 可以针对任何请求开始闪烁;但它一次只处理一个请求。
这就是为什么我有 dừng lại
如果有的话,它会停止以前运行的进程。 dừng lại
适用于正常条件。
但是我看到当 LED 闪烁并且我在没有停止服务器的情况下进行部署时,正在运行的线程并没有停止。如果我停止服务器并进行部署并再次启动,此时正在运行的线程将终止。
trong khi
中的线程循环并执行 Process
给本地人。这Process
是一次性工作,所以这 Process
不会让线程被杀死。
为了重现我所做的问题,我在上下文初始化时创建了线程,并在它被销毁时试图杀死它。现在,如果我在 contextDestroyed
中写点什么我可以看到他们被处决。
我不明白为什么停止服务器不会在我重新部署时杀死线程。
câu trả lời hay nhất
您应该在 processBuilder.start() 返回的 Process 实例上调用 process.destroy()。实际上你在调用 BlinkLedTask.terminate() 时所做的只是设置一些标志。此时您应该调用 process.destroy()。
下面我将展示一个如何重写它的例子。它不涉及你的类 ScriptExecutor(当然你可以将你的逻辑移到那里并在调用 blinkLed() 时将进程实例返回给 BlinkLedTask)。
这里的主要区别在于,我在字段 blinkLedProcess 中保留对 Process 实例的引用,当调用 terminate() 时,我直接调用 process.destroy() 来销毁进程。
您写道“当服务器关闭时,stop() 会按预期工作。我正在使用 Jetty。”确实是的。这是因为通过调用 processBuilder.start();
您可以创建主 jetty 进程的子进程。当你杀死 jetty 时,它的所有子进程也会被杀死。如果您不终止 jetty,则需要通过调用 destroy() 方法手动终止子进程。
Nó sẽ trông như thế này:
public enum BlinkLedTask {
(...)
private Process resetLedProcess;
private Process blinkLedProcess;
(...)
private void blinkLed() throws Exception {
String[] args = new String[] { frequency[0], frequency[1], frequency[2] };
List commands = new ArrayList<>();
//commands.add(BASH);
commands.add(script.getAbsoultePath());
if (Objects.nonNull(args)) {
commands.addAll(Arrays.asList(args));
}
StringBuilder builder = new StringBuilder("Executing script: ");
builder.append(script.getAbsoultePath());
if (Objects.nonNull(args) && (args.length > 0)) {
builder.append(" with parameters: ");
builder.append(StringUtils.join(args, " "));
}
ProcessBuilder processBuilder = new ProcessBuilder(commands.toArray(new String[commands.size()]));
blinkLedProcess = processBuilder.start();
StringBuilder outputBuilder = new StringBuilder();
InputStream inputStream = blinkLedProcess.getInputStream();
InputStreamReader inputStreamReader = new InputStreamReader(inputStream);
BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
String line = StringUtils.EMPTY;
while ((line = bufferedReader.readLine()) != null) {
outputBuilder.append(line);
outputBuilder.append("\n");
}
blinkLedProcess.waitFor();
int exitValue = blinkLedProcess.exitValue();
Hệ thống.out.println(
"Process for: " + Script.BLINK_LED.getAbsoultePath() + " is executed. Exit value: " + exitValue);
}
(...)
private void terminate() {
System.out.println("Stopping - " + Thread.currentThread().getName());
running = false;
if (resetLedProcess != null) {
resetLedProcess.destroy();
System.out.println("Destroyed reset process");
}
if (blinkLedProcess != null) {
blinkLedProcess.destroy();
System.out.println("Destroyed blink process");
}
}
(...)
}
关于java - ExecutorService - 上下文被销毁时无法从 ServletContextListener 停止线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25933310/
Tôi đã viết một số ứng dụng Rest thông qua ioc mùa xuân. Nhưng tôi không thể giải quyết vấn đề này. Đây là ngoại lệ của tôi: org.springframework.beans.factory.BeanC
Tôi hoàn toàn mới đối với TestNG, Spring framework, v.v. và tôi đang cố gắng truy cập các chú thích tệp cấu hình thông qua @Configuration bằng chú thích @Value. Điều tôi đang cố gắng đạt được ở đây là yêu cầu bảng điều khiển ghi "xin chào" từ tệp cấu hình, thông qua
Đã làm việc này trong nhiều giờ. Tôi hoàn toàn bối rối. Đây là phòng thí nghiệm cho CS113. Nếu người dùng chọn tiếp tục ở cuối chương trình (máy tính nhị phân), chúng ta cần sử dụng câu lệnh goto để lên đầu chương trình. Tuy nhiên, chúng ta cũng cần giải phóng tất cả bộ nhớ được phân bổ.
Tôi đang cố gắng xây dựng một chương trình C nhỏ bằng thư viện ffmpeg. Nhưng tôi thậm chí không thể mở tệp âm thanh bằng avformat_open_input() Sau khi thiết lập chức năng kiểm tra mã lỗi, tôi nhận được kết quả đầu ra sau: Mã lỗi:
Sử dụng Spring Khởi tạo để tạo một Spring boot đơn giản. Tôi vừa chọn DevTools trong các tùy chọn có sẵn. Sau khi tạo dự án, bạn không cần thực hiện bất kỳ thay đổi nào đối với dự án để chạy chương trình bình thường. Bây giờ khi tôi thử trong dự án
Vì vậy, tôi vừa cài đặt qt qua brew trong Mac OS X. Nhưng nó không thể liên kết nó. Khi tôi cố chạy brew link qt hoặc brew link --overwrite qt tôi nhận được thông tin sau: ton
Tôi gặp vấn đề với cam kết và kéo: trong IDE cam kết tôi thấy: cảnh báo không phải tất cả các thay đổi cục bộ đều có thể được hiển thị do lỗi: không thể
Tôi đã chạy man gcc | grep "-L" và tôi đã hiểu cách sử dụng: grep [OPTION]... PATTERN [FILE]... Hãy thử `grep --help' để biết thêm thông tin
Tôi có một đoạn mã được thiết kế để lấy bất kỳ URL nào và trích xuất nó khỏi trang web. Cho đến nay nó vẫn hoạt động tốt cho đến khi ai đó cung cấp cho nó URL này: http://www.aspensurgical.com/static/images/a
Tôi đã cố gắng thiết lập WireGuard trên máy chủ của mình trong 5 giờ qua nhưng sau khi thiết lập xong, tôi không thể ping IP hoặc phân giải miền. Dưới đây là cấu hình server [Giao diện] Địa chỉ = 10.
Tôi đang cố gắng phân nhánh một trong các dự án riêng tư của mình trong GitLab, nhưng khi nhấn nút phân nhánh, tôi nhận được thông báo sau: Không có không gian tên khả dụng để phân nhánh
Tôi đang gặp một số vấn đề ở đây. Tôi mới làm quen với node.js và Rest API nhưng tôi đang cố gắng tự học. Tôi đã tạo một API REST sử dụng MongoDB để giao tiếp với cơ sở dữ liệu của mình và tôi sử dụng Postman để kiểm tra cách của mình
Mã bên dưới cung cấp thông báo sau trong bảng điều khiển: Uncaught DOMException: Không thể thực thi 'appendChild' trên 'Node': El con mới
Tôi đang cố gắng gọi một điểm cuối mới để hiển thị dữ liệu và tôi nhận ra rằng trong tập dữ liệu hợp lệ cuối cùng, nó được điều khiển bằng một cặp dấu ngoặc "[]" bổ sung xung quanh dữ liệu, tôi nghĩ đó chính là vấn đề và Điểm cuối mới không tạo ra dữ liệu theo cách tôi sử dụng. Đây là lý do NgFor thất bại!
Tôi đang cố gắng triển khai ứng dụng Symfony2 lên Ứng dụng web Azure của mình và tôi đang gặp một số sự cố. Khi đẩy tới điều khiển từ xa, tôi nhận được thông báo sau trong điều khiển từ xa: Đang cập nhật nhánh 'mas
Minikube đang hoạt động mà không gặp bất kỳ lỗi nào, nhưng tôi không thể cuộn IP. Tôi đã theo dõi ở đây: https://docs.traefik.io/user-guide/kubernetes/ và dường như không có đề cập đến việc tắt máy
Bất cứ khi nào tôi cố gắng soạn thảo bất kỳ dự án nào, tôi đều gặp lỗi sau. Tôi đã thử có và không có sudo. Tôi chỉ gặp vấn đề này trên máy này. Tôi có thể chạy cùng một vùng chứa trên máy Mac và Amazon WorkSpace của mình. (myslabs)
Tôi đang cố gắng cài đặt khổ thơ và nhận được thông báo này: LỖI: Không tìm thấy phân phối phù hợp cho đèn pin>=1.3.0 (từ khổ thơ
Độ phân giải DNS có vẻ ổn nhưng tôi không thể ping dịch vụ của mình. Lý do có thể là gì? Từ một Pod khác trong cụm: $ ping backend PING backend.default.svc.cluster.l
Tôi đang sử dụng Hibernate 4 + Spring MVC 4 và khi khởi động Apache Tomcat Server 8, tôi gặp lỗi này: Lỗi tạo đậu có tên 'wel
Tôi là một lập trình viên xuất sắc, rất giỏi!