linux 远程主机后台运行任务 挂起脚本

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了linux 远程主机后台运行任务 挂起脚本相关的知识,希望对你有一定的参考价值。

参考技术A

背景 :工作需要,需要远程控制主机使其断开Xshell后也能一直运行。
关键字 :& ,nohup,脚本挂起。

在应用Unix/Linux时,我们一般想让某个程序在后台运行,于是我们将常会用 & 在程序结尾来让程序自动运行。

键入Ctrl + C,发出SIGINT信号,程序会继续运行

方法一、输入命令:jobs

方法二:输入命令:ps

fg、bg、jobs、&、nohup、ctrl+z、ctrl+c 命令

加在一个命令的最后,可以把这个命令放到后台执行,如

可以将一个正在前台执行的命令放到后台,并且处于暂停状态。

查看当前有多少在后台运行的命令
jobs -l选项可显示所有任务的PID,jobs的状态可以是running, stopped, Terminated。但是如果任务被终止了(kill),shell 从当前的shell环境已知的列表中删除任务的进程标识。

将后台中的命令调至前台继续运行。如果后台中有多个命令,可以用fg %jobnumber(是命令编号,不是进程号)将选中的命令调出。

将一个在后台暂停的命令,变成在后台继续执行。如果后台中有多个命令,可以用bg %jobnumber将选中的命令调出。

法子1:通过jobs命令查看job号(假设为num),然后执行kill %num
法子2:通过ps命令查看job的进程号(PID,假设为pid),然后执行kill pid
前台进程的终止:Ctrl+c

如果让程序始终在后台执行,即使关闭当前的终端也执行(之前的&做不到),这时候需要nohup。该命令可以在你退出帐户/关闭终端之后继续运行相应的进程。
关闭中断后,在另一个终端jobs已经无法看到后台跑得程序了,此时利用ps(进程查看命令)

如上所示:
nohup 和 & 组合了后台运行程序。
可以输出到 out.log保存程序输出,也可以输出到/dev/null 即空设备,不保存输出。

上面三种方式无论在nohup.out 或test.out都看不到数据结果,这是因为python执行有缓存输出
解决

新的接口接入查看程序

使用&后台运行程序:

结果会输出到终端
使用Ctrl + C发送SIGINT信号, 程序免疫
关闭session发送SIGHUP信号, 程序关闭

使用nohup运行程序:

结果默认会输出到nohup.out
使用Ctrl + C发送SIGINT信号, 程序关闭
关闭session发送SIGHUP信号, 程序免疫
平日线上经常使用nohup和&配合来启动程序:
同时免疫SIGINT和SIGHUP信号

同时,还有一个最佳实践:
不要将信息输出到终端标准输出,标准错误输出,而要用日志组件将信息记录到日志里
nohup和&究竟有啥区别

远程通知的后台任务在片​​刻后暂停

【中文标题】远程通知的后台任务在片​​刻后暂停【英文标题】:Background task on remote notification suspended after a short while 【发布时间】:2016-05-19 11:12:25 【问题描述】:

我正在使用带有 content-available: true 标志的远程通知来在后台以静默推送通知启动应用程序,并从远程 API 处理或获取更新。当应用程序处于前台或上次运行后处于挂起状态时,代码执行良好。

在后台测试期间,当系统根据传入的静默推送通知启动应用程序时,代码仅被部分处理,应用程序在大约 150 毫秒后快速挂起。我预计应用程序将有 30 秒的时间来处理传入的通知及其有效负载。如果我需要更多时间来处理和/或获取新数据,是否需要调整应用功能或请求后台任务?

部署目标 iOS 8,在 iOS 9 上测试。Xcode 7.3.1,Swift 2.2.1。

功能:开启后台模式,模式:启用远程通知

AppDelegate

func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject], fetchCompletionHandler completionHandler: (UIBackgroundFetchResult) -> Void) 

    if let userInfo = userInfo as? [String: AnyObject] 

        // Processing of the payload is done in separate Operation class (using Operations framework)
        // The completion handler is called on the end of the processing/fetching in that operation
        // But in case of launching the app in the background it never reaches the call of the completion handler

        let parseNotificationOperation = ParseNotificationOperation(userInfo: userInfo, fetchCompletionHandler: completionHandler)
        MainService.shared.enqueueApnsOperation(parseNotificationOperation)
    

【问题讨论】:

【参考方案1】:

马丁·科尔斯,

您可以使用 expirationHandlers 来延长后台执行时间。虽然 iOS 会为您的应用分配多少时间取决于我们无法控制的各种因素,但我们注意到它主要为我们的应用在后台执行提供了 3 分钟。

以下是实现它的方法:)

在你的 AppDelegate 中声明,

UIBackgroundTaskIdentifier backgroundTask;

当你在 didReceiveRemoteNotification 中收到 APNS 时,写下这个,

if (!backgroundTask || backgroundTask == UIBackgroundTaskInvalid) 
        backgroundTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^
                //do all clean up job here gets called few seconds before expiration
                [[UIApplication sharedApplication] endBackgroundTask:backgroundTask];
                backgroundTask = UIBackgroundTaskInvalid;
        ];
    

编辑

刚刚意识到您正在使用 swift,所以这里是 swift 中的代码 :)

在AppDelegate中声明一个叫backgroundTask的变量,

var backgroundTask : UIBackgroundTaskIdentifier = UIBackgroundTaskInvalid

并在您的 didRecieveRemoteNotification 中使用它,如下所示,

func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject], fetchCompletionHandler completionHandler: (UIBackgroundFetchResult) -> Void) 

    if let userInfo = userInfo as? [String: AnyObject] 
        let parseNotificationOperation = ParseNotificationOperation(userInfo: userInfo, fetchCompletionHandler: completionHandler)
        MainService.shared.enqueueApnsOperation(parseNotificationOperation)

        if (backgroundTask == UIBackgroundTaskInvalid) 
            backgroundTask = UIApplication.sharedApplication().beginBackgroundTaskWithExpirationHandler  () -> Void in
                self.endTask()
            
        
    

最后编写一个方法来使你的过期处理程序在你完成后失效:)

func endTask()
   //do all your clean up here, this is called few seconds before the task expires.
   UIApplication.sharedApplication().endBackgroundTask(backgroundTask)
   backgroundTask = UIBackgroundTaskInvalid;

就是这样 :) 快乐的编码伙伴 :)

【讨论】:

@SandeepBhandari 我想在收到静默推送后触发本地通知,但是一旦我将应用程序置于后台,我就没有收到任何本地通知。你能帮我解决这个问题吗? 苹果这次是否允许后台任务静默通知作弊? @digs :对不起,我还没有用 iOS 13 的变化更新我自己,所以现在还不知道:|如果我在这方面有所了解,会及时通知你

以上是关于linux 远程主机后台运行任务 挂起脚本的主要内容,如果未能解决你的问题,请参考以下文章

远程通知的后台任务在片​​刻后暂停

怎么让 Linux 进程在后台运行?

linux系统下如何从后台启动进程?

linux 把任务放到后台

Linux下使Shell 命令脱离终端在后台运行

让Linux进程在后台运行