我试图使用Java从远程系统设置一个linux系统的日期和时间。为了做到这一点,我创建了一个服务器,接受来自远程系统的时间为:。
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket;
public class Set_date_n_time {
public static void main(String[] args) throws IOException, InterruptedException {
// TODO Auto-generated method stub
String date_time = new String();
//@SuppressWarnings("resource")
ServerSocket s1 = new ServerSocket(7105);
System.out.println("server started");
while (true) {
Socket sckt = s1.accept();
InputStream input = sckt.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(input));
date_time = reader.readLine();
String command="sudo date -s "+"\""+date_time+"\"";
Process p;
try {
p = Runtime.getRuntime().exec(command);
p.waitFor();
System.out.println ("date set");
p.destroy();
} catch (Exception e) {}
}
}
}
而远程系统的时间将被复制为:
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintStream;
import java.net.Socket;
import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;
public class Fix_my_Date {
public static void main(String args[]) throws IOException {
String addr_list=args[0];
String[] hostList = readAddressList(addr_list);
for(int i=0; i<hostList.length;i++) {
setDate(hostList[i]);
}
}
//@SuppressWarnings("resource")
private static void setDate(String address) throws IOException {
{
Scanner sc = new Scanner(System.in);
Socket s = null;
String date =new String();
String time = new String();
try {
s = new Socket(address, 7105);
System.out.println("connection to "+address+" done");
Process p, p1;
try {
p = Runtime.getRuntime().exec("date +%Y%m%d");
BufferedReader br = new BufferedReader(
new InputStreamReader(p.getInputStream()));
date = br.readLine();
br.close();
p.waitFor();
p.destroy();
p1 = Runtime.getRuntime().exec("date +%H:%M:%S");
BufferedReader br1 = new BufferedReader(
new InputStreamReader(p1.getInputStream()));
time = br1.readLine();
br1.close();
p1.waitFor();
p1.destroy();
PrintStream pr = new PrintStream(s.getOutputStream());
pr.print(date+" "+time+"");
sc.close();
s.close();
} catch (Exception e) {
System.out.println("Problem Setting date and time");
}
//s.close();
} catch (Exception e) {
System.out.println("Couldn't connect to: "+address+"");
sc.close();
//s.close();
}
}
return;
}
private static String[] readAddressList(String addr_list) throws IOException {
FileReader fileReader = new FileReader(addr_list);
BufferedReader bufferedReader = new BufferedReader(fileReader);
List<String> lines = new ArrayList<String>();
String line = null;
while ((line = bufferedReader.readLine()) != null)
{
lines.add(line);
}
bufferedReader.close();
System.out.println("Loaded the host list");
return lines.toArray(new String[lines.size()]);
}
}
但时间并没有被服务器代码设置。我的错误在哪里?
你是想在虚拟机上设置时间吗?如果是这样的话,它可能被设置为同步到主机上,这将覆盖你的date -s命令。
我无法让它与Runtime.exec()一起工作(还没有),但它与ProcessBuilder一起完美地工作。这就是它。
package set_date_n_time;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket;
public class Set_date_n_time {
public static void main(String[] args) throws IOException, InterruptedException {
// TODO Auto-generated method stub
String date_time = new String();
//@SuppressWarnings("resource")
ServerSocket s1 = new ServerSocket(7105);
System.out.println("server started");
while (true) {
Socket sckt = s1.accept();
InputStream input = sckt.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(input));
date_time = reader.readLine();
ProcessBuilder builder = new ProcessBuilder("date", "--set=" + date_time + "");
final Process p = builder.start();
p.waitFor();
p.destroy();
}
}
}
很明显你必须以root身份运行,并确保你运行的是
sudo systemctl stop systemd-timesyncd.service
或类似的东西,以确保系统不会覆盖你的日期修正。
我,你,还有很多人犯的错误也是(除了这里的其他有用的答案),你不读标准输出和标准错误,如果你的命令产生任何输出或错误,它就会阻塞,因为没有缓冲区,它可以写到,你可以观察与 strace
.
这可以用一个额外的线程来解决,如这里所说。https:/www.javaworld.comjavaworldjw-12-2000jw-1229-traps.html
如果你使用sudo,它可能会失败,这取决于sudo的设置,如果它不允许没有终端的sudo (requiretty),请参阅更多信息,例如。https:/bugzilla.redhat.comshow_bug.cgi?id=1196451。
如果你需要或喜欢使用Runtime.exec(),那么(不要问我为什么)只需将。
String command = "sudo date -s " + "\"" + date_time + "\"";
改成
String[] command = new String[]{"sudo", "date", "-s", date_time};
在类Set_date_n_time中。
替换这个。
} catch (Exception e) {}
用这个来代替
} catch (Exception e) {
throw new IllegalStateException("Unexpected exception", e);
}
就像Ole V. V. 说的那样, 你的代码中的空捕获块几乎可以肯定是抛弃了你所关心的真正的失败. 我猜想,无论是 sudo
拒绝调用,或者命令本身有问题。异常会告诉你到底是哪里出了问题。
如果 你发现有一些异常被抛出,而你真的想忽略它们,请单独处理它们,但你几乎永远不应 catch (Exception e)
并扔掉异常。
也可以使用 ProcessBuilder
而不是 Runtime.exec()
. 这是一个更强大、更灵活的与子进程交互的API。特别是: 从来没有 使用 Runtime.exec(String)
虽然它可以用于简单的命令,但它是 不 shell,并且对于带有特殊字符的命令,如引号或空格,会以令人惊讶的方式失败。
例如
p = new ProcessBuilder("sudo", "date", "-s", date_time).start();