Làm thế nào tôi có thể chạy một lệnh cụ thể cho từng kết quả tìm thấy?


49

Làm thế nào một người có thể chạy một lệnh cụ thể cho mỗi tệp được tìm thấy bằng cách sử dụng findlệnh? Đối với mục đích của câu hỏi, hãy nói rằng tôi chỉ muốn xóa từng tệp được tìm thấy bởi find.

Câu trả lời:


60

Chỉnh sửa: Mặc dù câu trả lời sau giải thích trường hợp sử dụng chung, tôi nên lưu ý rằng việc xóa các tệp và thư mục là trường hợp đặc biệt. Thay vì sử dụng -execdir rm {} \;cấu trúc, chỉ cần sử dụng -delete, như trong:

find -iname '*.txt' -delete

Điều này xử lý một loạt các trường hợp cạnh mà bạn có thể không nghĩ đến bao gồm các tập tin và thư mục thứ tự cần phải được xóa để không gặp lỗi. Đối với các trường hợp sử dụng khác ...

Cách tốt nhất để xử lý các lệnh chạy kết quả tìm kiếm thường là sử dụng các -exectùy chọn khác nhau cho findlệnh. Cụ thể, bạn nên cố gắng sử dụng -execdirbất cứ khi nào có thể vì nó chạy trong thư mục của tệp được tìm thấy và thường an toàn hơn (theo nghĩa là ngăn chặn những sai lầm ngu ngốc là thảm họa) so với các tùy chọn khác.

Các -exectùy chọn được theo sau bởi lệnh bạn muốn chạy với {}biểu thị vị trí nơi tìm thấy tệp tìm thấy và được chấm dứt bằng cách \;chạy lệnh một lần cho mỗi tệp hoặc +thay thế {}bằng danh sách các đối số của tất cả các kết quả khớp . Lưu ý rằng dấu chấm phẩy được thoát ra để vỏ không được hiểu là dấu phân cách dẫn đến lệnh mới.

Hãy nói rằng bạn đã tìm thấy tất cả các tệp văn bản:

find -iname '*.txt' -execdir rm {} \;

Đây là bit có liên quan từ hướng dẫn tìm ( man find):

   -exec command ;
          Execute  command;  true  if 0 status is returned.  All following
          arguments to find are taken to be arguments to the command until
          an  argument  consisting of ‘;’ is encountered.  The string ‘{}’
          is replaced by the current file name being processed  everywhere
          it occurs in the arguments to the command, not just in arguments
          where it is alone, as in some versions of find.  Both  of  these
          constructions might need to be escaped (with a ‘\’) or quoted to
          protect them from expansion by the shell.  See the EXAMPLES sec-
          tion for examples of the use of the -exec option.  The specified
          command is run once for each matched file.  The command is  exe-
          cuted  in  the starting directory.   There are unavoidable secu-
          rity problems surrounding use of the -exec  action;  you  should
          use the -execdir option instead.


   -exec command {} +
          This  variant  of the -exec action runs the specified command on
          the selected files, but the command line is built  by  appending
          each  selected file name at the end; the total number of invoca-
          tions of the command will  be  much  less  than  the  number  of
          matched  files.   The command line is built in much the same way
          that xargs builds its command lines.  Only one instance of  ‘{}’
          is  allowed  within the command.  The command is executed in the
          starting directory.


   -execdir command ;

   -execdir command {} +
          Like -exec, but the specified command is run from the  subdirec-
          tory  containing  the  matched  file,  which is not normally the
          directory in which you started find.  This a  much  more  secure
          method  for invoking commands, as it avoids race conditions dur-
          ing resolution of the paths to the matched files.  As  with  the
          -exec action, the ‘+’ form of -execdir will build a command line
          to process more than one matched file, but any given  invocation
          of command will only list files that exist in the same subdirec-
          tory.  If you use this option, you must ensure that  your  $PATH
          environment  variable  does  not  reference  ‘.’;  otherwise, an
          attacker can run any commands they like by leaving an  appropri-
          ately-named  file in a directory in which you will run -execdir.
          The same applies to having entries in $PATH which are  empty  or
          which are not absolute directory names.

Lệnh "find" do Busybox cung cấp không hỗ trợ tùy chọn -execdir, do đó có thể cần phải sử dụng một trong các phương thức pipe / xargs được đề cập dưới đây.
MikeW

9

Một cách khác là dẫn đầu ra và phân tích cú pháp bằng các lệnh tiếp theo. Cách an toàn duy nhất để làm như vậy là sử dụng -print0tùy chọn, cho biết findsử dụng ký tự null làm dấu phân cách kết quả. Các lệnh nhận phải có khả năng nhận ra đầu vào được phân tách bằng null. Thí dụ:

find /home/phunehehe -iregex '.*\.png$' -print0 | xargs -0 file

Lưu ý rằng -0tùy chọn yêu xargscầu xử lý đầu vào là không giới hạn.


Bạn có thể -execvới nhiều tệp hơn nếu bạn kết thúc nó +thay vì ;. Xem câu trả lời của caleb.
Kevin

@Kevin bạn nói đúng, tôi cập nhật câu trả lời.
phunehehe

3

Find có một lệnh xóa tích hợp nếu đó là tất cả những gì bạn cần làm.

find . -name "*.txt" -delete

Bất kỳ tệp .txt nào được tìm thấy sẽ bị xóa bằng lệnh trên.


2

Tôi đã tìm kiếm một câu trả lời cho điều này và tôi tình cờ thấy chủ đề này. Các câu trả lời đã cho tôi và ý tưởng về cách tôi có thể đạt được nó. Giả sử bạn muốn tìm mediainfotất cả các tệp JPEG

Điều này sẽ xuất hiện mediainfo "ở đầu và "cuối của mỗi tệp phù hợp (Để thoát các ký tự đặc biệt càng nhiều càng tốt), đặt nó vào một tập lệnh và chạy tập lệnh:

find . -name *.jpg | sed -e 's/^/mediainfo "/g;' | sed -e 's/$/"/g;' > foo.sh && sh foo.sh

Trong trường hợp bạn lo lắng điều gì đó có thể sai, bạn có thể bỏ qua việc chuyển hướng đầu ra sang một tệp và chỉ xem kết quả trong thiết bị đầu cuối trước khi chạy tập lệnh.


1

Bạn có thể thực hiện điều này bằng cách sử dụng xargslệnh. xargsvề cơ bản chạy một lệnh một lần cho mỗi lệnh của đầu vào tiêu chuẩn của nó. Vì vậy, nếu bạn cần xóa tất cả .jpgcác tệp trong một thư mục chẳng hạn, một cách nhanh chóng trên dòng lệnh là:

$ find ./ -name "*.jpg" | xargs rm 

Bạn cũng có thể sử dụng backtick (phía trên nút Tab) để làm điều này (lưu ý rằng đây là ký tự backquote, không phải ký tự trích dẫn đơn):

$ rm `find ./ -name "*.jpg"`

Lưu ý rằng do cách xargsvà shell xử lý đầu vào của chúng, phương thức xargs chỉ hoạt động nếu không có tên tệp và tên thư mục nào liên quan có chứa khoảng trắng hoặc bất kỳ \"'; phương thức backquote chỉ hoạt động nếu không có tên tệp và tên thư mục nào liên quan có chứa khoảng trắng hoặc bất kỳ \[?*.


3
Cả hai phương pháp này đều có khả năng rất nguy hiểm, đặc biệt là phương pháp backtick. Có rất nhiều vấn đề tiềm ẩn với các ký tự chưa thoát trong tên tệp có thể khiến các phương thức này bị hỏng.
Caleb

1
Tôi thấy quan điểm của bạn, nhưng các lệnh này cũng có thể được sử dụng cùng với các công cụ khác ngoài tìm kiếm, vì vậy tôi nghĩ rằng chúng đáng được đề cập ở đây.
IG83

Chúng có thể đáng được đề cập, nhưng khi không sử dụng chúng là điều quan trọng để chỉ định. Câu hỏi của OP ở đây được yêu cầu cụ thể để xử lý đầu ra find, -execnói chung là một giải pháp tốt hơn. Nếu bạn muốn chỉ định những cái này là thay thế, ít nhất hãy giải thích cách sử dụng find -print0 | xargs -0để xử lý ngắt tên tệp an toàn và giải thích khi nào nên cẩn thận về backticks.
Caleb

1
Chào mừng bạn đến trang web bằng cách này, tôi thấy đây là câu trả lời đầu tiên của bạn. Xin lỗi để nhảy qua nó. Điều quan trọng là dạy mọi người trước những vấn đề là để họ không mắc phải những lỗi khó mắc phải sau này, nhưng tôi vẫn nhớ những ngày tôi không hiểu tại sao đây lại là một vấn đề lớn như vậy, vì vậy xin đừng 'T lấy sự điều chỉnh là cá nhân.
Caleb

3
Cảm ơn bạn đã chào đón! Tất nhiên không có cảm giác khó khăn, bạn chắc chắn đúng trong đó -exec là cách thích hợp để xử lý việc này. Tôi thực sự bắt đầu thấy nền tảng này tuyệt vời như thế nào, bạn đang học những thứ mới ngay cả từ các bình luận :)
IG83
Khi sử dụng trang web của chúng tôi, bạn xác nhận rằng bạn đã đọc và hiểu Chính sách cookieChính sách bảo mật của chúng tôi.
Licensed under cc by-sa 3.0 with attribution required.