В следующем сценарии показан способ сделать это с помощью функций. Вы можете поставить bgxupdate
и bgxlimit
функции в скрипте или иметь их в отдельный файл, который поступает из вашего сценария с:
. /path/to/bgx.sh
Это имеет то преимущество, что вы можете поддерживать несколько групп процессов независимо друг от друга (вы можете например, одна группа с пределом 10 и другая полностью отдельная группа с пределом 3).
Он использовал bash
встроенный, jobs
, чтобы получить список подпроцессов, но поддерживает их в отдельных переменных. В петле внизу вы можете увидеть, как вызвать функцию bgxlimit
:
- установить пустую переменную группы.
- перевод это
bgxgrp
.
- позвонить
bgxlimit
с лимитом и командой, которую вы хотите запустить.
- переведите новую группу обратно в свою групповую переменную.
Конечно, если у вас есть только одна группа, просто используйте bgxgrp
, а не передавайте и выходите.
#!/bin/bash
# bgxupdate - update active processes in a group.
# Works by transferring each process to new group
# if it is still active.
# in: bgxgrp - current group of processes.
# out: bgxgrp - new group of processes.
# out: bgxcount - number of processes in new group.
bgxupdate() {
bgxoldgrp=${bgxgrp}
bgxgrp=""
((bgxcount = 0))
bgxjobs=" $(jobs -pr | tr '\n' ' ')"
for bgxpid in ${bgxoldgrp} ; do
echo "${bgxjobs}" | grep " ${bgxpid} " >/dev/null 2>&1
if [[ $? -eq 0 ]] ; then
bgxgrp="${bgxgrp} ${bgxpid}"
((bgxcount = bgxcount + 1))
fi
done
}
# bgxlimit - start a sub-process with a limit.
# Loops, calling bgxupdate until there is a free
# slot to run another sub-process. Then runs it
# an updates the process group.
# in: $1 - the limit on processes.
# in: $2+ - the command to run for new process.
# in: bgxgrp - the current group of processes.
# out: bgxgrp - new group of processes
bgxlimit() {
bgxmax=$1 ; shift
bgxupdate
while [[ ${bgxcount} -ge ${bgxmax} ]] ; do
sleep 1
bgxupdate
done
if [[ "$1" != "-" ]] ; then
$* &
bgxgrp="${bgxgrp} $!"
fi
}
# Test program, create group and run 6 sleeps with
# limit of 3.
group1=""
echo 0 $(date | awk '{print $4}') '[' ${group1} ']'
echo
for i in 1 2 3 4 5 6 ; do
bgxgrp=${group1} ; bgxlimit 3 sleep ${i}0 ; group1=${bgxgrp}
echo ${i} $(date | awk '{print $4}') '[' ${group1} ']'
done
# Wait until all others are finished.
echo
bgxgrp=${group1} ; bgxupdate ; group1=${bgxgrp}
while [[ ${bgxcount} -ne 0 ]] ; do
oldcount=${bgxcount}
while [[ ${oldcount} -eq ${bgxcount} ]] ; do
sleep 1
bgxgrp=${group1} ; bgxupdate ; group1=${bgxgrp}
done
echo 9 $(date | awk '{print $4}') '[' ${group1} ']'
done
Вот пример работы:
0 12:38:00 [ ]
1 12:38:00 [ 3368 ]
2 12:38:00 [ 3368 5880 ]
3 12:38:00 [ 3368 5880 2524 ]
4 12:38:10 [ 5880 2524 1560 ]
5 12:38:20 [ 2524 1560 5032 ]
6 12:38:30 [ 1560 5032 5212 ]
9 12:38:50 [ 5032 5212 ]
9 12:39:10 [ 5212 ]
9 12:39:30 [ ]
- Все это начинается в 12:38:00 и, как вы можете видеть, первые три процесса сразу же запустить.
- Каждый процесс спит для
n*10
секунд, поэтому четвертый процесс не начинается до первого выхода (в момент времени t = 10 или 12:38:10). Вы можете видеть, что процесс 3368 исчез из списка до добавления 1560.
- Аналогично, пятый процесс (5032) начинается, когда второй (5880) выходит в момент времени t = 20.
- И, наконец, шестой процесс (5212) начинается, когда третий (2524) выходит в момент времени t = 30.
- Затем начинается отсчет, четвертый процесс выходит при t = 50 (начался с 10, длительность 40), пятый при t = 70 (начался с 20, длительность 50) и шестой при t = 90 (начался с 30, продолжительность 60).
Или, в виде временной линии:
Process: 1 2 3 4 5 6
-------- - - - - - -
12:38:00^^^
12:38:10 v | |^
12:38:20 v | |^
12:38:30 v | |^
12:38:40 | | |
12:38:50 v | |
12:39:00 | |
12:39:10 v |
12:39:20 |
12:39:30 v
Я думаю, что этот вопрос может помочь вам: http://stackoverflow.com/questions/38160/parallelize-bash-script –
Итак, много запутанных ответов, но никак не сказать bash "максимум десять одновременных заданий!". Наверное, тогда нет никого. Жаль, что это действительно хорошая функция. –