Je veux dire les lignes de sortie 5 à 10 d'un fichier, comme arguments passés.
Comment pourrais-je utiliser head
et tail
pour ce faire?
où firstline = $2
et lastline = $3
et filename = $1
.
L'exécution devrait ressembler à ceci:
./lines.sh filename firstline lastline
Outre les réponses données par fedorqui et Kent , vous pouvez également utiliser une seule commande sed
:
#! /bin/sh
filename=$1
firstline=$2
lastline=$3
# Basics of sed:
# 1. sed commands have a matching part and a command part.
# 2. The matching part matches lines, generally by number or regular expression.
# 3. The command part executes a command on that line, possibly changing its text.
#
# By default, sed will print everything in its buffer to standard output.
# The -n option turns this off, so it only prints what you tell it to.
#
# The -e option gives sed a command or set of commands (separated by semicolons).
# Below, we use two commands:
#
# ${firstline},${lastline}p
# This matches lines firstline to lastline, inclusive
# The command 'p' tells sed to print the line to standard output
#
# ${lastline}q
# This matches line ${lastline}. It tells sed to quit. This command
# is run after the print command, so sed quits after printing the last line.
#
sed -ne "${firstline},${lastline}p;${lastline}q" < ${filename}
Ou, pour éviter tout utilitaire externe, si vous utilisez une version récente de bash (ou zsh):
#! /bin/sh
filename=$1
firstline=$2
lastline=$3
i=0
exec <${filename} # redirect file into our stdin
while read ; do # read each line into REPLY variable
i=$(( $i + 1 )) # maintain line count
if [ "$i" -ge "${firstline}" ] ; then
if [ "$i" -gt "${lastline}" ] ; then
break
else
echo "${REPLY}"
fi
fi
done
head -n XX # <-- print first XX lines
tail -n YY # <-- print last YY lines
Si vous voulez des lignes de 20 à 30, cela signifie que vous voulez 11 lignes à partir de 20 et se terminant à 30:
head -n 30 file | tail -n 11
#
# first 30 lines
# last 11 lines from those previous 30
Autrement dit, vous obtenez d'abord 30
lignes, puis vous sélectionnez le dernier 11
(C'est, 30-20+1
).
Donc dans votre code ce serait:
head -n $3 $1 | tail -n $(( $3-$2 + 1 ))
Basé sur firstline = $2
, lastline = $3
, filename = $1
head -n $lastline $filename | tail -n $(( $lastline -$firstline + 1 ))
essayez ce one-liner:
awk -vs="$begin" -ve="$end" 'NR>=s&&NR<=e' "$f"
dans la ligne ci-dessus:
$begin is your $2
$end is your $3
$f is your $1
Enregistrez ceci sous "script.sh":
#!/bin/sh
filename="$1"
firstline=$2
lastline=$3
linestoprint=$(($lastline-$firstline+1))
tail -n +$firstline "$filename" | head -n $linestoprint
Il n'y a AUCUNE GESTION DES ERREURS (pour plus de simplicité), vous devez donc appeler votre script comme suit:
./ script.sh yourfile.txt firstline lastline
$ ./script.sh yourfile.txt 5 10
Si vous n'avez besoin que de la ligne "10" de votrefichier.txt:
$ ./script.sh yourfile.txt 10 10
Veuillez vous assurer que: (première ligne> 0) ET (dernière ligne> 0) ET (première ligne <= dernière ligne)