Вопрос

Мне нужно найти PID текущего запущенного процесса на платформе Linux (это может быть системно-зависимое решение).Java не поддерживает получение идентификатора процесса, а в JRuby в настоящее время имеется ошибка в методе Ruby Process.pid.

Есть ли другой способ получить PID?

Это было полезно?

Решение

Если у вас есть прокфс установлен, вы можете найти идентификатор процесса через символическую ссылку /proc/self, которая указывает на каталог с именем pid (здесь также есть файлы с другой соответствующей информацией, включая PID, но каталог — это все, что вам нужно в этом случай).

Таким образом, с помощью Java вы можете:

String pid = new File("/proc/self").getCanonicalFile().getName();

В JRuby вы можете использовать то же решение:

pid = java.io.File.new("/proc/self").canonical_file.name

Особая благодарность каналу #stackoverflow на бесплатном узле за помощь в решении этой проблемы!(конкретно, Джеруб, Грег, и Верхняя палуба)

Другие советы

Протестировано только в Linux с использованием Sun JVM.Может не работать с другими реализациями JMX.

String pid = ManagementFactory.getRuntimeMXBean().getName().split("@")[0];

Вы можете использовать интерфейс JNI для вызова функции POSIX. получитьпид().Это довольно просто.Вы начинаете с класса необходимых вам функций POSIX.Я называю это POSIX.java:

import java.util.*;

class POSIX
{
    static { System.loadLibrary ("POSIX"); }
    native static int getpid ();
}

Скомпилируйте его с помощью

$ javac POSIX.java

После этого вы создаете заголовочный файл POSIX.h с

$ javah -jni POSIX

Заголовочный файл содержит прототип функции C с оболочкой функции getpid.Теперь вам нужно реализовать функцию, что довольно просто.я сделал это в POSIX.c:

#include "POSIX.h"

#include <sys/types.h>
#include <unistd.h>

JNIEXPORT jint JNICALL Java_POSIX_getpid (JNIEnv *env, jclass cls)
{
    return getpid ();
}

Теперь вы можете скомпилировать его с помощью gcc:

$ gcc -Wall -I/usr/lib/jvm/java-1.6.0-sun-1.6.0.21/include -I/usr/lib/jvm/java-1.6.0-sun-1.6.0.21/include/linux -o libPOSIX.so -shared -Wl,-soname,libPOSIX.so POSIX.c -static -lc

Вам необходимо указать место, где установлена ​​ваша Java.Вот и все.Теперь вы можете использовать его.Создайте простую программу getpid:

public class getpid
{
    public static void main (String argv[])
    {
        System.out.println (POSIX.getpid ());
    }
}

Скомпилируйте его с помощью javac getpid.java и запустите его:

$ java getpid &
[1] 21983
$ 21983

Первый pid записывается оболочкой, а второй — программой Java после возврата приглашения оболочки.∎

Создайте процесс оболочки, который будет читать pid своего родителя.Должно быть, это наш pid.Вот работающий код без исключений и обработки ошибок.

import java.io.*;
import java.util.Scanner;

public class Pid2
{
  public static void main(String sArgs[])
    throws java.io.IOException, InterruptedException
  {
    Process p = Runtime.getRuntime().exec(
      new String[] { "sh", "-c", "ps h -o ppid $$" });
    p.waitFor();
    Scanner sc = new Scanner(p.getInputStream());
    System.out.println("My pid: " + sc.nextInt()); 
    Thread.sleep(5000);
  }
}

Это решение кажется лучшим, если PID необходимо получить только для выполнения другой команды оболочки.Достаточно заключить команду в обратные кавычки, чтобы передать ее в качестве аргумента другой команде, например:

nice `ps h -o ppid $$`

Это может заменить последнюю строку в массиве, переданном exec вызов.

Вы можете попробовать getpid() в JNR-Posix.

Он также имеет оболочку Windows POSIX, которая вызывает getpid() из libc.JNI не нужен.

Java 9 наконец предлагает официальный способ сделать это с помощью дескриптор процесса:

ProcessHandle.current().pid();

Этот:

  • Сначала получает ProcessHandle ссылка на текущий процесс.

  • Чтобы получить доступ к его pid.

Импорт не требуется, так как ProcessHandle часть java.lang.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top