16

So, I have some jobs like this:

sleep 30 | sleep 30 &

The natural way to think would be:

kill `jobs -p`

But that kills only the first sleep but not the second.

Doing this kills both processes:

kill %1

But that only kills at most one job if there are a lot of such jobs running.

It shouldn't kill processes with the same name but not run in this shell.

user23013
  • 1,087
  • 1
  • 9
  • 18

5 Answers5

12

A somewhat shorter version of xhienne's answer, but not pure-bash:

jobs -p | xargs -I{} kill -- -{}
sduthil
  • 359
7

Use this:

pids=( $(jobs -p) )
[ -n "$pids" ] && kill -- "${pids[@]/#/-}"

jobs -p prints the PID of process group leaders. By providing a negative PID to kill, we kill all the processes belonging to that process group (man 2 kill). "${pids[@]/#/-}" just negates each PID stored in array pids.

xhienne
  • 17,793
  • 2
  • 53
  • 69
  • I couldn't get this to work in Ubu 20.04. (kill complained about negitave PID numnbers.). However, this works: while [ -n "$(jobs -p)" ]; do kill %%; done – jpaugh Mar 01 '21 at 05:39
  • @jpaugh I don't have U20.4. What you describe would be a bug to report to Bash, since negative PIDs are a property of the kill(2) system call and are part of the POSIX standard. Are you sure you copied-pasted my code correctly? What is your version of Bash? You can adapt my code to /bin/kill if Bash's kill is buggy (but IMO this is very unlikely). – xhienne Mar 01 '21 at 10:23
  • I tried both kill and /usr/bin/kill; While I may have gotten the code wrong, I did verify the format of the PIDs before passing them to kill. I'm not going to look into this further, but hopefully this convo will help someone else. – jpaugh Mar 02 '21 at 17:31
  • For the reference, can you please add here the error messages produced by both Bash's builtin kill and /bin/kill when given a (valid) negative PID number? (/usr/bin/kill is surprising, you probably mean /bin/kill) – xhienne Mar 02 '21 at 18:06
  • No, I used /usr/bin/kill. It's probably a symlink to /bin/kill (Ubuntu). I'll try to remember to re-run it when I'm in front of a linux box. – jpaugh Mar 02 '21 at 20:15
3

Use kill $( jobs -p )... I love it. Use this if you’ve jost got too many jobs for kill’s argv (it can happen I guess):

killalljobs() { for pid in $( jobs -p ); do kill -9 $pid ; done ; }

user2497
  • 747
  • 1
  • 5
  • 8
2

You can use pkill and pgrep to kill a list of process names.

From the man page:

pgrep looks through the currently running processes and lists the process IDs which matches the selection criteria to stdout. All the criteria have to match. For example, pgrep -u root sshd will only list the processes called sshd AND owned by root. On the other hand, pgrep -u root,daemon will list the processes owned by root OR daemon. pkill will send the specified signal (by default SIGTERM) to each process instead of listing them on stdout.

An example using pgrep, pkill,

$ pgrep -l script.sh
12406 script.sh
12425 script.sh

$ pkill $(pgrep script.sh)

$ cat signal-log
Name: ./script.sh Pid: 12406 Signal Received: SIGTERM
Name: ./script.sh Pid: 12425 Signal Received: SIGTERM
Chen A.
  • 1,323
2

This will kill all jobs, most recently created first:

while kill %%; do :; done

This stops when there are no more jobs to kill, and prints:

bash: kill: %%: no such job

Works in zsh too.

Tom Hale
  • 30,455