The
nom
(no match) script takes filenames (usually, expanded by the shell) from
its command line.
It outputs all filenames in the current directory that
don't
match.
As article
15.2
explains,
ksh
has a
!
operator
that works like
nom
, and
tcsh
has
^
pattern
,
but other shells don't.
Here are some examples of
nom
:
-
To get the names of all files that
don't
end with
.ms
:
%
nom *.ms
-
To edit all files whose names don't have any lowercase letters, use
command substitution (
9.16
)
:
%
vi `nom *[a-z]*`
-
To copy all files to a directory named
Backup
(except
Backup
itself):
%
cp `nom Backup` Backup
Here's the script:
trap
case
$*
comm
-
|
#! /bin/sh
temp=/tmp/NOM$$
stat=1 # ERROR EXIT STATUS (SET TO 0 BEFORE NORMAL EXIT)
trap 'rm -f $temp; exit $stat' 0 1 2 15
# MUST HAVE AT LEAST ONE ARGUMENT. ALL MUST BE IN CURRENT DIRECTORY:
case "$*" in
"") echo Usage: `basename $0` pattern 1>&2; exit ;;
*/*) echo "`basename $0` quitting: I can't handle '/'s." 1>&2; exit ;;
esac
# GET NAMES WE DON'T WANT TO MATCH; REPLACE BLANKS WITH NEWLINES:
echo "$*" | tr ' ' '\012' | sort > $temp
# COMPARE TO CURRENT DIRECTORY (-1 = ONE NAME PER LINE); OUTPUT NAMES WE WANT:
ls -1 | comm -23 - $temp
stat=0
|
You can remove the
-1
option on the script's
ls
command
line if your version of
ls
lists one filename per line by default;
almost all versions of
ls
do that when they're writing into a pipe.
Note that
nom
doesn't know about files whose names begin with a
dot (
.
); you can change that if you'd like by adding
the
ls -A
option (uppercase letter "A", which isn't on all
versions of
ls
).
The script line with
tr
(
35.11
)
will split filenames containing space characters.
You can replace that line with
the following three lines;
they run more
slowly on some shells but will fix this (unlikely) problem:
for
done |
|
for file
do echo "$file"
done | sort > $temp
|