हर 15 सेकंड में एक यूनिक्स स्क्रिप्ट कैसे प्राप्त करें?


84

मैंने कुछ समाधान देखे हैं, जिसमें घड़ी भी शामिल है और बस बैकग्राउंड में एक लूपिंग (और स्लीपिंग) स्क्रिप्ट चल रही है, लेकिन कुछ भी आदर्श नहीं है।

मेरे पास एक स्क्रिप्ट है जिसे हर 15 सेकंड में चलाने की आवश्यकता है, और चूंकि क्रोन सेकंड का समर्थन नहीं करेगा, इसलिए मुझे कुछ और पता लगाना बाकी है।

यूनिक्स पर हर 15 सेकंड में स्क्रिप्ट चलाने का सबसे मजबूत और कुशल तरीका क्या है? रिबूट के बाद स्क्रिप्ट को भी चलाने की आवश्यकता है।


1
इसे चलाने में कितना समय लगता है?
एडम मटन

जवाबों:


77

मैं हर मिनट एक स्क्रिप्ट चलाने के लिए क्रोन का उपयोग करता हूं, और उस स्क्रिप्ट को रन के बीच 15 सेकंड की नींद के साथ आपकी स्क्रिप्ट को चार बार चलाता हूं।

(ऐसा माना जाता है कि आपकी स्क्रिप्ट चलने में तेज है - आप नींद के समय को समायोजित कर सकते हैं यदि नहीं।)

इस तरह, आपको cron15 सेकंड की अवधि के साथ-साथ सभी लाभ मिलते हैं ।

संपादित करें: नीचे @ bmb की टिप्पणी भी देखें।


@ माता-पिता: हा! मेरी दासता, हम फिर से मिलते हैं!
रिचीहिंडले

56
यदि स्क्रिप्ट चलने में संगत नहीं है, तो उसे चलाने में कितनी देर लगती है, स्क्रिप्ट की चार प्रतियां बनाएं। एक शुरू होने से 15 सेकंड पहले, दूसरा 30, 45 और दूसरा शून्य पर सोता है। फिर हर चार मिनट में चलाएं।
जून'09 bmb

@ रीचीहिंडल - कोई डर नहीं है, मुझे मिनटों को सेकंड में दाने नहीं देने के लिए हत्या कर दी गई। लेकिन मैं तुम्हें देख रहा हूं: P
Aiden Bell

इस क्रोन को हर 1 मिनट में कैसे ट्रिगर किया जा सकता है
DevZer0

वास्तव में आउट-स्क्रिप्ट को आंतरिक स्क्रिप्ट को तीन बार चलाना चाहिए, चार नहीं। अन्यथा अंतिम मिनट का अंतिम रन अगले मिनट के फ्रिस्ट रन को ओवरलैप कर देगा। यह आंतरिक स्क्रिप्ट को चार के बजाय हर मिनट 5 बार चल रहा होगा।
तुलसी कोर्डोवा

290

यदि आप क्रोन से अपनी स्क्रिप्ट चलाने के लिए जोर देते हैं:

* * * * * /foo/bar/your_script
* * * * * sleep 15; /foo/bar/your_script
* * * * * sleep 30; /foo/bar/your_script
* * * * * sleep 45; /foo/bar/your_script

और अपना स्क्रिप्ट नाम और पथ / foo / bar / your_script बदलें


18
यह बहुत चालाक है। +1
सिंगलनेगलाइजेशन क्लेम

4
इसने मेरे लिए पूरी तरह से काम किया। एक पृष्ठभूमि कार्य का उपयोग करने पर इस के ऊपर की सॉलक्शन कई बच्चे की प्रक्रियाओं को पैदा कर रहा था और मेरे अंत में स्मृति मुद्दों का कारण बना।
हैकनाइटली

2
अगर php स्क्रिप्ट चल रही है तो ऐसा करें:* * * * * sleep 15; php /foo/bar/your_script
ImaginedDesign

1
अगर php स्क्रिप्ट चल रही है, तो आप #!/usr/bin/phpअपनी php स्क्रिप्ट के शीर्ष पर लाइन को प्रीपेंड कर सकते हैं और इसे निष्पादन योग्य बना सकते हैं
हारून ब्लेंकश

18
मैं शर्मिंदा महसूस करता हूं कि मुझे इस समाधान के लिए गूगल करना पड़ा। हो सकता है कि stackoverflow मुझे कम लगता है।
क्रिस फिन

15

उपरोक्त का संशोधित संस्करण:

mkdir /etc/cron.15sec
mkdir /etc/cron.minute
mkdir /etc/cron.5minute

/ etc / crontab में जोड़ें:

* * * * * root run-parts /etc/cron.15sec > /dev/null 2> /dev/null
* * * * * root sleep 15; run-parts /etc/cron.15sec > /dev/null 2> /dev/null
* * * * * root sleep 30; run-parts /etc/cron.15sec > /dev/null 2> /dev/null
* * * * * root sleep 45; run-parts /etc/cron.15sec > /dev/null 2> /dev/null

* * * * * root run-parts /etc/cron.minute > /dev/null 2> /dev/null
*/5 * * * * root run-parts /etc/cron.5minute > /dev/null 2> /dev/null

13

पृष्ठभूमि में यह नहीं चल रहा है?

#!/bin/sh
while [ 1 ]; do
    echo "Hell yeah!" &
    sleep 15
done

यह लगभग उतना ही कुशल है जितना इसे मिलता है। महत्वपूर्ण भाग केवल हर 15 सेकंड में निष्पादित हो जाता है और स्क्रिप्ट बाकी समय (इस प्रकार चक्र बर्बाद नहीं कर) सोता है।


8
संपादन कम से कम 8 वर्ण (जो कि मुहावरेदार है, IMHO) होना चाहिए, इसलिए मैं &पंक्ति 3 के अंत में नहीं जोड़ सका । किसी भी मामले में, जैसा कि यह है, यह हर 15 सेकंड में नहीं चलता है। यह हर "15 सेकंड + हालांकि लंबे समय echo helloतक चलने के लिए चलता है"। जो 0.01 सेकंड हो सकता है; 19 घंटे हो सकते हैं।
पार्थियन शॉट

1

मैंने क्रोन की तुलना में एक अनुसूचक को तेजी से लिखा। मैंने एक ओवरलैपिंग गार्ड भी लागू किया है। यदि पिछली प्रक्रिया अभी भी चल रही है तो आप नई प्रक्रिया शुरू नहीं करने के लिए अनुसूचक को कॉन्फ़िगर कर सकते हैं। Https://github.com/sioux1977/scheduler/wiki पर एक नज़र डालें


0

नैनोसेलेप (2) का प्रयोग करें । यह संरचना का उपयोग करता है timespecजो नैनोसेकंड परिशुद्धता के साथ समय के अंतराल को निर्दिष्ट करने के लिए उपयोग किया जाता है।

struct timespec {
           time_t tv_sec;        /* seconds */
           long   tv_nsec;       /* nanoseconds */
       };

1
मैं आगे बढ़ने जा रहा हूं और अनुमान लगा रहा हूं कि उन्हें नैनोसेकंड प्रिसेंस की जरूरत नहीं है, क्योंकि वे हर 15 सेकंड में जिस चीज को खोल रहे हैं वह एक शेल स्क्रिप्ट है, न कि कर्नेल थ्रेड।
पार्थियन ने 23

@ParthianShot शायद लेकिन आप कभी नहीं जानते।
अलेक्जेंडर सुरफेल

0
#! /bin/sh

# Run all programs in a directory in parallel
# Usage: run-parallel directory delay
# Copyright 2013 by Marc Perkel
# docs at http://wiki.junkemailfilter.com/index.php/How_to_run_a_Linux_script_every_few_seconds_under_cron"
# Free to use with attribution

if [ $# -eq 0 ]
then
   echo
   echo "run-parallel by Marc Perkel"
   echo
   echo "This program is used to run all programs in a directory in parallel" 
   echo "or to rerun them every X seconds for one minute."
   echo "Think of this program as cron with seconds resolution."
   echo
   echo "Usage: run-parallel [directory] [delay]"
   echo
   echo "Examples:"
   echo "   run-parallel /etc/cron.20sec 20"
   echo "   run-parallel 20"
   echo "   # Runs all executable files in /etc/cron.20sec every 20 seconds or 3 times a minute."
   echo 
   echo "If delay parameter is missing it runs everything once and exits."
   echo "If only delay is passed then the directory /etc/cron.[delay]sec is assumed."
   echo
   echo 'if "cronsec" is passed then it runs all of these delays 2 3 4 5 6 10 12 15 20 30'
   echo "resulting in 30 20 15 12 10 6 5 4 3 2 executions per minute." 
   echo
   exit
fi

# If "cronsec" is passed as a parameter then run all the delays in parallel

if [ $1 = cronsec ]
then
   $0 2 &
   $0 3 &
   $0 4 &
   $0 5 &
   $0 6 &
   $0 10 &
   $0 12 &
   $0 15 &
   $0 20 &
   $0 30 &
   exit
fi

# Set the directory to first prameter and delay to second parameter

dir=$1
delay=$2

# If only parameter is 2,3,4,5,6,10,12,15,20,30 then automatically calculate 
# the standard directory name /etc/cron.[delay]sec

if [[ "$1" =~ ^(2|3|4|5|6|10|12|15|20|30)$ ]]
then
   dir="/etc/cron.$1sec"
   delay=$1
fi

# Exit if directory doesn't exist or has no files

if [ ! "$(ls -A $dir/)" ]
then
   exit
fi

# Sleep if both $delay and $counter are set

if [ ! -z $delay ] && [ ! -z $counter ]
then
   sleep $delay
fi

# Set counter to 0 if not set

if [ -z $counter ]
then
   counter=0
fi

# Run all the programs in the directory in parallel
# Use of timeout ensures that the processes are killed if they run too long

for program in $dir/* ; do
   if [ -x $program ] 
   then
      if [ "0$delay" -gt 1 ] 
      then
         timeout $delay $program &> /dev/null &
      else
         $program &> /dev/null &
      fi
   fi
done

# If delay not set then we're done

if [ -z $delay ]
then
   exit
fi

# Add delay to counter

counter=$(( $counter + $delay ))

# If minute is not up - call self recursively

if [ $counter -lt 60 ]
then
   . $0 $dir $delay &
fi

# Otherwise we're done

0

अपने पिछले उत्तर के बाद से मैं एक और समाधान के साथ आया हूं जो अलग है और शायद बेहतर है। यह कोड माइक्रोसेकंड परिशुद्धता के साथ प्रक्रियाओं को एक मिनट में 60 से अधिक बार चलाने की अनुमति देता है। इस काम को करने के लिए आपको usleep प्रोग्राम की आवश्यकता है। एक सेकंड में 50 गुना तक अच्छा होना चाहिए।

#! /bin/sh

# Microsecond Cron
# Usage: cron-ms start
# Copyright 2014 by Marc Perkel
# docs at http://wiki.junkemailfilter.com/index.php/How_to_run_a_Linux_script_every_few_seconds_under_cron"
# Free to use with attribution

basedir=/etc/cron-ms

if [ $# -eq 0 ]
then
   echo
   echo "cron-ms by Marc Perkel"
   echo
   echo "This program is used to run all programs in a directory in parallel every X times per minute."
   echo "Think of this program as cron with microseconds resolution."
   echo
   echo "Usage: cron-ms start"
   echo
   echo "The scheduling is done by creating directories with the number of"
   echo "executions per minute as part of the directory name."
   echo
   echo "Examples:"
   echo "  /etc/cron-ms/7      # Executes everything in that directory  7 times a minute"
   echo "  /etc/cron-ms/30     # Executes everything in that directory 30 times a minute"
   echo "  /etc/cron-ms/600    # Executes everything in that directory 10 times a second"
   echo "  /etc/cron-ms/2400   # Executes everything in that directory 40 times a second"
   echo
   exit
fi

# If "start" is passed as a parameter then run all the loops in parallel
# The number of the directory is the number of executions per minute
# Since cron isn't accurate we need to start at top of next minute

if [ $1 = start ]
then
   for dir in $basedir/* ; do
      $0 ${dir##*/} 60000000 &
   done
   exit
fi

# Loops per minute and the next interval are passed on the command line with each loop

loops=$1
next_interval=$2

# Sleeps until a specific part of a minute with microsecond resolution. 60000000 is full minute

usleep $(( $next_interval - 10#$(date +%S%N) / 1000 ))

# Run all the programs in the directory in parallel

for program in $basedir/$loops/* ; do
   if [ -x $program ] 
   then
      $program &> /dev/null &
   fi
done

# Calculate next_interval

next_interval=$(($next_interval % 60000000 + (60000000 / $loops) ))

# If minute is not up - call self recursively

if [ $next_interval -lt $(( 60000000 / $loops * $loops)) ]
then
   . $0 $loops $next_interval &
fi

# Otherwise we're done

1
फिर से पोस्ट करने के बजाय मूल को संपादित करें!
वोगोन जेल्ट्ज़

-1

निष्पादन के संभावित अतिव्यापी से बचने के लिए, उस थ्रेड में वर्णित के रूप में एक लॉकिंग तंत्र का उपयोग करें ।


2
-1 ओपी ने यह नहीं कहा कि उसे अतिव्यापी निष्पादन से बचने की आवश्यकता है; बात फिर से हो सकती है। साथ ही, यह सवाल का जवाब नहीं देता है।
पार्थियन शॉट
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.