Trova file con lo stesso nome ma con contenuti diversi?


9

Voglio generare un elenco di file che hanno:

  • Stesso nome
  • Contenuto diverso

in una directory (compresi tutti gli elenchi e i contenuti secondari).

Come fare? Bash, perl, tutto va bene.

Pertanto, non dovrebbero essere visualizzati due file con lo stesso nome e lo stesso contenuto.


Supponiamo che ci siano tre file con lo stesso nome e due dei tre siano identici?
Kyle Jones

@KyleJones: "tre file con lo stesso nome e due dei tre sono identici"> Quindi questo nome file deve essere aggiunto all'elenco
Nicolas Raoul

Risposte:


8

Aggiornamento: corretto un refuso nello script: modificato print $NFin print $3; ha anche riordinato le cose e aggiunto alcuni commenti.

Supponendo che i nomi dei file non contengano \n, il seguente stampa un elenco ordinato che si interrompe (come in: interruzioni di controllo di sezione ) in univoco file name, univoco md5sume mostra il gruppo corrispondente di percorsi di file.

#!/bin/bash

# Choose which script to use for the final awk step 
out_script=out_all

# Print all duplicated file names, even when md5sum is the same 
out_all='{ if( p1 != $1 ) { print nl $1; print I $2 }
      else if( p2 != $2 ) { print I $2 }
      print I I $3; p1=$1; p2=$2; nl="\n" }
   END { printf nl}'

# Print only duplicated file names which have multiple md5sums.
out_only='{ if( p1 != $1 ) { if( multi ) { print pend }
                             multi=0; pend=$1 "\n" I $2 "\n" }
       else if( p2 != $2 ) { multi++; pend=pend I $2 "\n" } 
       pend=pend I I $3 "\n"; p1=$1; p2=$2 } 
   END { if( multi ) print pend }'

# The main pipeline 
find "${1:-.}" -type f -name '*' |  # awk for duplicate names
awk -F/ '{ if( name[$NF] ) { dname[$NF]++ }
           name[$NF]=name[$NF] $0 "\n" } 
     END { for( d in dname ) { printf name[d] } 
   }' |                             # standard md5sum output 
xargs -d'\n' md5sum |               # " "==text, "*"==binary
sed 's/ [ *]/\x00/' |               # prefix with file name  
awk -F/ '{ print $3 "\x00" $0 }' |  # sort by name. md5sum, path 
sort |                              # awk to print result
awk -F"\x00" -v"I=   " "${!out_script}"

Output che mostra solo i nomi di file con più md5 s

afile.html
   53232474d80cf50b606069a821374a0a
      ./test/afile.html
      ./test/dir.svn/afile.html
   6b1b4b5b7aa12cdbcc72a16215990417
      ./test/dir.svn/dir.show/afile.html

Output che mostra tutti i file con lo stesso nome.

afile.html
   53232474d80cf50b606069a821374a0a
      ./test/afile.html
      ./test/dir.svn/afile.html
   6b1b4b5b7aa12cdbcc72a16215990417
      ./test/dir.svn/dir.show/afile.html

fi    le.html
   53232474d80cf50b606069a821374a0a
      ./test/dir.svn/dir.show/fi    le.html
      ./test/dir.svn/dir.svn/fi    le.html

file.html
   53232474d80cf50b606069a821374a0a
      ./test/dir.show/dir.show/file.html
      ./test/dir.show/dir.svn/file.html

file.svn
   53232474d80cf50b606069a821374a0a
      ./test/dir.show/dir.show/file.svn
      ./test/dir.show/dir.svn/file.svn
      ./test/dir.svn/dir.show/file.svn
      ./test/dir.svn/dir.svn/file.svn

file.txt
   53232474d80cf50b606069a821374a0a
      ./test/dir.show/dir.show/file.txt
      ./test/dir.show/dir.svn/file.txt
      ./test/dir.svn/dir.show/file.txt
      ./test/dir.svn/dir.svn/file.txt

1

Ecco uno script Perl. Eseguirlo nella directory nella parte superiore dell'albero che si desidera cercare. Lo script dipende finde md5, ma quest'ultimo può essere sostituito con sha1, sumo qualsiasi altro programma di file hashing che accetta l'input su stdin ed emette un hash su stdout.

use strict;

my %files;
my %nfiles;
my $HASHER = 'md5';

sub
print_array
{
    for my $x (@_) {
        print "$x\n";
    }
}

open FINDOUTPUT, "find . -type f -print|" or die "find";

while (defined (my $line = <FINDOUTPUT>)) {
    chomp $line;
    my @segments = split /\//, $line;
    my $shortname = pop @segments;
    push @{ $files{$shortname} }, $line;
    $nfiles{$shortname}++;
}

for my $shortname (keys %files) {
    if ($nfiles{$shortname} < 2) {
        print_array @{ $files{$shortname} };
        next;
    }
    my %nhashes;
    my %revhashes;
    for my $file (@{ $files{$shortname} }) {
        my $hash = `$HASHER < $file`;
        $revhashes{$hash} = $file;
        $nhashes{$hash}++;
    }
    for my $hash (keys %nhashes) {
        if ($nhashes{$hash} < 2) {
            my $file = $revhashes{$hash};
            print "$file\n";
        }
    }
}

1

finddup questo strumento può anche aiutarti a elencare i file con gli stessi nomi o contenuti.


1

Per coloro che vogliono vedere solo un elenco di nomi di file, qui è la parte rilevante della Peter.O 's risposta :

find "${1:-.}" -type f -name '*' | 
awk -F/ '{ if( name[$NF] ) { dname[$NF]++ }
       name[$NF]=name[$NF] $0 "\n" } 
 END { for( d in dname ) { printf name[d] "\n" } 

}'

Non ho bisogno di md5sums perché uso fslint-guiprima dello script per cancellare tutti i duplicati.


sul mio mac questo mostra i file duplicati con lo stesso nome e lo stesso contenuto
nightograph
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.