web-dev-qa-db-ja.com

同じファイル名で異なるコンテンツを見つける方法

同じ名前であるが、2つの異なるディレクトリから異なるコンテンツを含むテキストファイルを見つけようとしています。

以下は私のコードですが、ここでクラッシュし続けます

cati=`ls $1 | cat $i`

catj=`ls $2 | cat $j` 

等...

どうすれば修正できますか?

#!/bin/bash

if [ $# -ne 2 ];then
    echo "Usage ./search.sh dir1 dir2"
    exit 1
Elif [ ! -d  $1 ];then
    echo "Error cannot find dir1"
    exit 2
Elif [ ! -d  $2 ];then
        echo "Error cannot find dir2"
        exit 2
Elif [ $1 == $2 ];then
    echo "Error both are same directories"

else
    listing1=`ls $1`
    listing2=`ls $2`
    for i in $listing1; do
            for j in $listing2; do
                if [ $i == $j ];then
                     cati=`ls $1 | cat $i`
                     catj=`ls $2 | cat $j`
                     if [ $cati != $catj ];then
                           echo $j
                     fi
                fi

            done
    done
fi
2
newuserr

findwhile read loopを使用する別のオプションがあります

#!/usr/bin/env bash

##: Set a trap to clean up temp files.
trap 'rm -rf "$tmpdir"' EXIT

##: Create temp directory using mktemp.
tmpdir=$(mktemp -d) || exit

##: If arguments less than 2
if (( $# < 2 )); then
  printf >&2 "Usage %s dir1 dir2 \n" "${BASH_SOURCE##*/}"
  exit 1
fi

dirA=$1
dirB=$2

##: Loop through the directories
for d in "$dirA" "$dirB"; do
  if  [[ ! -e $d ]]; then  ##: If directory does not exists
    printf >&2 "%s No such file or directory!\n" "$d"
    exit 1
  Elif [[ ! -d $d ]]; then  ##: If it is not a directory.
    printf >&2 "%s is not a directory!\n" "$d"
    exit 1
  fi
done

##: If dir1 and dir2 has the same name.
if [[ $dirA = $dirB ]]; then
  printf >&2 "Dir %s and %s are the same directory!\n" "$dirA" "$dirB"
  exit 1
fi

##: Save the list of files in a file using find.
find "$dirA" -type f -print0 > "$tmpdir/$dirA.txt"
find "$dirB" -type f -print0 > "$tmpdir/$dirB.txt"


##: Although awk is best for comparing 2 files I'll leave that to the awk masters.

while IFS= read -ru "$fd" -d '' file1; do  ##: Loop through files of dirB
  while IFS= read -r -d '' file2; do ##: Loop through files of dirA
    if [[ ${file1##*/} = ${file2##*/} ]]; then  ##: If they have the same name
      if ! cmp -s "$file1" "$file2"; then ##: If content are not the same.
        printf 'file %s and %s has the same name but different content\n' "$file1" "$file2"
      else
        printf 'file %s and %s has the same name and same content\n' "$file1" "$file2"
      fi
    fi
  done < "$tmpdir/$dirA.txt"
done {fd}< "$tmpdir/$dirB.txt"
1
Jetchisel

「newuserr」へようこそ!ここでの主な問題は、ファイル全体をシェル変数にロードしようとしていることだと思います。まず、それはうまくスケーリングしません。次に、ファイルがシェルに読み込まれるときにバックティックによって変更が加えられるため、比較はうまく機能しません。そして最後に、ファイル内のすべての空白がコードを壊します。しかし、これを回避する方法があります! :)

diffからの戻り値を使用して、ファイルをシェルに読み込まないようにすることができます。そのようです:

if diff $i $j; then
     echo $j
fi

他のいくつかの提案:

  • シェルで条件文を実行するには、[[ではなく[を使用することをお勧めします。 [[は、混乱したアイテムをより適切に処理します。
  • どんな欠品? $iまたは$jが空だった場合はどうなりますか?次に、シェルはその行でエラーになります。 "$i"または"$j"を実行することにより、シェルに欠落している変数の場所を保持させることができます。
  • shellcheck コードを読み取り、ベストプラクティスの提案を行います。
  • グーグルのシェルスタイルガイド 他にもたくさんのヒントがあります。
0
chicks