如何制作启动 Python 脚本的 linux 后台进程(在 c 中)
How can I make a linux background process (in c) that launches a Python script
我做了一个Linux后台进程(用c ++(,它监视一个目录,并在某个文件出现在该目录中时尝试启动Python脚本。我的问题是负责启动 Python 脚本的子进程在调用 execvp 函数后立即退出,我不明白为什么。所有必要的文件都在root的所有权下。如果有帮助,这是我的代码。提前感谢您的任何指示!我已经在代码中标记了错误发生错误的位置。我还包含了要调用的 Python 脚本
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
using namespace std;
char* arguments[3];
FILE* fd;
const char* logFilePath = "/home/BluetoothProject/Logs/fileMonitorLogs.txt";
char* rfcommPath = (char*)"/home/BluetoothProject/RFCOMMOut.py";
void logToFile(const char*);
void doWork();
void logToFile(const char* str) {
fd = fopen(logFilePath, "a");
fprintf(fd, "%sn", str);
fclose(fd);
}
int main() {
arguments[0] = (char*)"python";
arguments[1] = rfcommPath;
arguments[2] = NULL;
pid_t pid = fork();
if(pid < 0) {
printf("Fork failed");
exit(1);
} else if(pid > 0) {
exit(EXIT_SUCCESS);
}
umask(0);
pid_t sid = setsid();
if(sid < 0) {
logToFile("setsid() didn't work.");
exit(1);
}
if ((chdir("/")) < 0) {
logToFile("chdir() didn't work.");
exit(EXIT_FAILURE);
}
close(STDIN_FILENO);
close(STDOUT_FILENO);
close(STDERR_FILENO);
doWork();
}
void doWork() {
pid_t pid = fork();
if(pid < 0) {
logToFile("doWork() fork didn't work.");
} else if(pid > 0) {
int status = 0;
waitpid(pid, &status, 0);
if(WEXITSTATUS(status) == 1) {
logToFile("Child process exited with an error.");
}
} else {
int error = execvp(arguments[0], arguments); //Here is where the error is
if(error == -1) {
logToFile("execvp() failed.");
}
exit(1);
}
}
Python 脚本(又名 RFCOMMOut.py(
import RPi.GPIO as gpio
import serial
led_state = 0
led_pin = 11
gpio.setmode(gpio.BOARD)
gpio.setwarnings(False)
gpio.setup(led_pin, gpio.OUT)
try:
ser = serial.Serial(port = '/dev/rfcomm0',
baudrate = 9600,
parity = serial.PARITY_NONE,
stopbits = serial.STOPBITS_ONE,
bytesize = serial.EIGHTBITS)
except IOException as e:
logFile = open("/home/BluetoothProject/Logs/fileMonitorLogs.txt", "a")
logFile.write("(First error handler) There was an exception:n")
logFile.write(str(e))
logFile.write("n")
logFile.close()
#gpio.output
def process_input(input):
global led_state
if input == "I have been sent.n":
if led_state == 1:
led_state = 0
gpio.output(led_pin, led_state)
else:
led_state = 1
gpio.output(led_pin, led_state)
while True:
try:
transmission = ser.readline()
process_input(transmission)
except IOError as e:
logFile = open("/home/BluetoothProject/Logs/fileMonitorLogs.txt", "a")
logFile.write("(second error handler) There was an exception:n")
logFile.write(str(e))
logFile.write("n")
logFile.close()
break
led_state = 0
gpio.output(led_pin, led_state)
gpio.cleanup()
print("End of programn")
这个问题有点不清楚,所以我会尝试对问题是什么进行一些不同的有根据的猜测,并单独解决每一个。
TL;DR:删除close(STDOUT_FILENO)
并close(STDERR_FILENO)
以获取更多调试信息,希望这些信息能为您指明正确的方向。
execvp(3( 返回 -1
根据 execvp(3( 文档, execvp(3( 在失败时设置 errno。为了理解它失败的原因,你的程序需要在某处输出 errno 的值;也许是 stdout、stderr 或您的日志文件。一个方便的方法是使用 perror(3(。例如:
#include <stdio.h>
...
void doWork() {
...
} else {
int error = execvp(arguments[0], arguments);
if(error == -1) {
perror("execvp() failed");
}
}
...
}
如果不知道 errno 值是什么,就很难确定 execvp(3( 失败的原因。
execvp(3( 成功了,但我的 Python 程序似乎没有运行
execvp(3( 成功意味着 Python 解释器已成功调用 (假设您的 PATH 中没有名为 "python" 的程序, 但实际上不是 Python 解释器(。如果您的程序似乎没有运行,这意味着 Python 在加载程序时遇到困难。据我所知,在这种情况下,Python 总是会将相关的错误消息输出到 stderr;例如,如果 Python 找不到你的程序,它会向 stderr 输出"没有这样的文件或目录"。
但是,看起来您的 C 程序在调用doWork()
之前正在调用close(STDERR_FILENO)
。根据 fork(2(, 子进程继承其父进程的一组打开文件描述符的副本。这意味着在分叉之前调用close(STDERR_FILENO)
将导致子进程没有打开的 stderr 文件描述符。如果 Python 在执行程序时出现任何错误,你永远不会知道,因为 Python 试图通过不存在的文件描述符来通知你。如果 execvp(3( 成功了,而 Python 程序似乎根本无法运行, 那么我建议您从 C 程序中删除close(STDERR_FILENO)
并重新运行所有内容。如果不看到 Python 输出的错误消息,将很难确定它无法运行 Python 程序的原因。
顺便说一句,我建议不要明确关闭stdin,stdout和stderr。根据 stdin(3(, 标准流通过调用 exit(3( 和正常程序终止来关闭。
execvp(3( 成功了, 我的 Python 程序正在运行, 但我的 Python 程序在做任何有用的工作之前就退出了
在这种情况下,我不确定问题可能是什么,因为我对树莓派不是很熟悉。但我认为,如果在运行 Python 程序之前不关闭标准流,调试会更容易。
希望这有帮助。
- 有没有一种方法可以捕获进程中的堆栈溢出?C++Linux
- 如何制作启动 Python 脚本的 linux 后台进程(在 c 中)
- 如何在 Linux 下使用 c++ 知道文件是否被其他进程使用?
- linux mint进程上的代码块已终止,状态为-1
- 如何使用 std::in 和 getline() 检测后台进程中从 bash 注入的输入
- 在 Linux 平台上以 C/C++ 打印进程的所有线程堆栈跟踪
- 使用C 获取Linux进程列表和状态
- Fedora Linux 中的进程
- 当多个进程尝试同时替换同一文件时,Linux 如何处理这种情况?
- 使用 POCO 启动后台进程
- 以编程方式对子进程 (Linux) 设置各种限制
- 如何仅在C中向后台进程发送SIGINT
- 识别后台进程并在 C++ WIN32 中置于前台
- 如何在后台进程上处理来自Windows任务管理器的"End Task"?
- Fork进程:在子进程终止时恢复父进程(Linux)
- 将后台进程发送到前台
- 在arduino上运行后台进程
- Pidof从一个后台脚本到另一个后台进程
- 运行后台进程Linux简单shell.c++
- 使用system()执行后台进程