web-dev-qa-db-fra.com

Comment utiliser Head and Tail pour imprimer des lignes spécifiques d'un fichier

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?

firstline = $2 et lastline = $3 et filename = $1.

L'exécution devrait ressembler à ceci:

./lines.sh filename firstline lastline
14
user2232423

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
17
sfstewman
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 ))
28
fedorqui

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
6
Kent

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)

4
Dado