java - Linux - 通过shell脚本运行jar文件时出现IO异常

标签 java linux shell

我正在使用 Debain 8.0 - jessie(64 位)。

当我尝试通过 shell 脚本从其目录外部运行 jar 文件时,出现 IO 异常。

目录:“/home/rscedit”

文件:“data(目录)”“run.sh(文件)”“Webserver.jar(文件)”

当我尝试从“/home/rscedit”之外的任何地方运行“run.sh”时,我的 jar 文件中遇到 IO 异常。

但是如果我尝试从“/home/rscedit”运行“run.sh”,它运行得很好。

我想在启动时运行我的 shell 脚本,所以我应该能够从“/home/rscedit”外部运行我的 shell 脚本,对吗?

Shell 脚本

#!/bin/sh
java -jar -Xmx20480m /home/rscedit/Webserver.jar
read –n1

执行 shell 脚本时遇到的错误

java.nio.file.NoSuchFileException: ./data/log/ipn.log.lck
        at sun.nio.fs.UnixException.translateToIOException(UnixException.java:86)
        at sun.nio.fs.UnixException.rethrowAsIOException(UnixException.java:102)
        at sun.nio.fs.UnixException.rethrowAsIOException(UnixException.java:107)
        at sun.nio.fs.UnixFileSystemProvider.newFileChannel(UnixFileSystemProvider.java:177)
        at java.nio.channels.FileChannel.open(FileChannel.java:287)
        at java.nio.channels.FileChannel.open(FileChannel.java:335)
        at java.util.logging.FileHandler.openFiles(FileHandler.java:459)
        at java.util.logging.FileHandler.<init>(FileHandler.java:326)
        at org.displee.utilities.logging.LogFactory.loadFileLogger(LogFactory.java:44)
        at org.displee.utilities.logging.LogFactory.<clinit>(LogFactory.java:19)

LogFactory.java

package org.displee.utilities.logging;

import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.*;

public class LogFactory {

    private static final String FORMAT = "%1$td-%1$tm-%1$tY %1$tH:%1$tM:%1$tS %4$s %2$s - %5$s%6$s%n";

    private static final Map<String, Logger> MAP = new HashMap<>();

    static {
        try {
            loadFileLogger("ipn", "./data/log/ipn.log");
            loadFileLogger("error", "./data/log/error.log");

            Logger logger = Logger.getLogger("console");
            logger.setUseParentHandlers(false);
            ConsoleHandler ch = new ConsoleHandler();
            ch.setFormatter(new ConsoleFormatter());
            ch.setLevel(Level.ALL);
            logger.addHandler(ch);
            register(logger.getName(), logger);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void register(String name, Logger logger) {
        MAP.putIfAbsent(name, logger);
    }

    public static Logger get(String name) {
        return MAP.get(name);
    }

    private static Logger loadFileLogger(String name, String path) throws IOException {
        Logger logger = Logger.getLogger(name);
        FileHandler fh = new FileHandler(path, true);
        fh.setFormatter(new ConsoleFormatter());
        logger.addHandler(fh);
        logger.setUseParentHandlers(false);
        register(name, logger);
        return logger;
    }

    private static class ConsoleFormatter extends Formatter {
        @Override
        public synchronized String format(LogRecord record) {
            Date date = new Date();
            date.setTime(record.getMillis());
            String source;
            if (record.getSourceClassName() != null) {
                source = record.getSourceClassName();
                if (record.getSourceMethodName() != null) {
                    source += " " + record.getSourceMethodName();
                }
            } else {
                source = record.getLoggerName();
            }
            String message = formatMessage(record);
            String throwable = "";
            if (record.getThrown() != null) {
                StringWriter sw = new StringWriter();
                PrintWriter pw = new PrintWriter(sw);
                pw.println();
                record.getThrown().printStackTrace(pw);
                pw.close();
                throwable = sw.toString();
            }
            return String.format(FORMAT,
                    date,
                    source,
                    record.getLoggerName(),
                    "LOG",
                    message,
                    throwable);
        }
    }

}

我用来执行 shell 脚本的命令

exec /home/rscedit/run.sh

编辑:我的数据 map 不仅包含日志文件,还包含其他内容,例如网站文件。

最佳答案

您的java程序似乎对工作目录敏感。我知道的最简单的解决方案,改变这个

java -jar -Xmx20480m /home/rscedit/Webserver.jar

(cd /home/rscedit ; java -jar -Xmx20480m Webserver.jar)

或者,改变Java,这个(以及你有模式的任何地方)

loadFileLogger("ipn", "./data/log/ipn.log");
loadFileLogger("error", "./data/log/error.log");

类似于(使其相对于$HOME),

loadFileLogger("ipn", new File(System.getProperty("user.home"), 
        "data/log/ipn.log").getPath());
loadFileLogger("error", new File(System.getProperty("user.home"), 
        "data/log/error.log").getPath());

关于java - Linux - 通过shell脚本运行jar文件时出现IO异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41430614/

相关文章:

java - 用户关闭(Xs out)JFrame 后立即执行操作

java - Android:java.lang.IllegalStateException:适配器的内容已更改但ListView未收到通知

java - Java 使用 split 时出现 ArrayIndexOutOfBoundsException

shell - 如何列出不包含两个不同字符串的所有文件

java - 如何为类加载器加载任意选择的 JAR?

linux - 无法在我的命令中重定向错误消息

c++ - 在系统级别监听关键事件

linux - 在 shell 脚本中使用\c

security - 这个bash脚本可以更简洁吗?

linux - 多核Linux系统上的实时音频