Làm thế nào để tích hợp lệnh mv sau khi tìm lệnh?


61

Tôi đang tìm kiếm các tệp có tên AAAtrong đường dẫn của chúng bằng lệnh sau:

find path_A -name "*AAA*"

Với đầu ra được hiển thị bởi lệnh trên, tôi muốn di chuyển các tệp đó sang một đường dẫn khác path_B. Thay vì di chuyển từng tệp một, tôi có thể tối ưu hóa lệnh bằng cách di chuyển các tệp đó ngay sau lệnh find không?

Câu trả lời:


87

Với GNU mv :

find path_A -name '*AAA*' -exec mv -t path_B {} +

Điều đó sẽ sử dụng -exectùy chọn find thay thế {}lần lượt với từng kết quả tìm kiếm và chạy lệnh bạn đưa ra. Như đã giải thích trong 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.  

Trong trường hợp này, chúng tôi đang sử dụng +phiên bản -execđể chúng tôi chạy càng ít mvthao tác càng tốt:

   -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.

Đừng quên "\;" vào cuối của người thực hiện!
Charles Roth

@Charles Roth: đó là +công việc, bạn có thể đọc trích dẫn của tôi ở trên hoặc man findthay vào đó
cuonglm

Cảm ơn vì điều đó! Tôi đã cố gắng sử dụng -exec mv {} path_b +và nó đã bị lỗi với quyền. TBH, tôi vẫn không hiểu tại sao, nhưng -exec mv -t path_b {} +làm việc một điều trị!
Jeremy Davis

6
@JeremyDavis Khi sử dụng -exec ... {} +, {}phải là điều cuối cùng trước +. Đây là lý do tại sao anh ta sử dụng mv -t destdir {} +và không mv {} destdir +. Một cảm lạnh đã được sử dụng -exec mv {} destdir ';'thay thế, nhưng điều đó sẽ thực hiện mvmột lần cho mỗi tệp.
Kusalananda

23

Bạn có thể làm một cái gì đó như dưới đây là tốt.

find path_A -name "*AAA*" -print0 | xargs -0 -I {} mv {} path_B

Ở đâu,

  1. -0Nếu có khoảng trắng hoặc ký tự (bao gồm cả dòng mới), nhiều lệnh sẽ không hoạt động. Tùy chọn này quan tâm đến tên tệp với không gian trống.
  2. -IThay thế các lần xuất hiện của str-str trong các đối số ban đầu bằng các tên được đọc từ đầu vào tiêu chuẩn. Ngoài ra, khoảng trống không trích dẫn không chấm dứt các mục đầu vào; thay vào đó dấu phân cách là ký tự dòng mới.

Kiểm tra

Tôi đã tạo hai thư mục là sourcedirdestdir. Bây giờ, tôi đã tạo ra một loạt các tập tin bên trong sourcedirnhư file1.bak, file2.bakfile3 with spaces.bak

Bây giờ, tôi đã thực thi lệnh như,

find . -name "*.bak" -print0 | xargs -0 -I {} mv {} /destdir/

Bây giờ, bên trong destdir, khi tôi làm ls, tôi có thể thấy rằng các tệp đã được chuyển từ sourcedirsang destdir.

Người giới thiệu

http://www.cyberciti.biz/faq/linux-unix-bsd-xargs-construct-argument-lists-utility/


22

Vì lợi ích của người dùng OS X khi gặp câu hỏi này, cú pháp trong OS X hơi khác một chút. Giả sử bạn không muốn tìm kiếm đệ quy trong các thư mục con của path_A:

find path_A -maxdepth 1 -name "*AAA*" -exec mv {} path_B \;

Nếu bạn muốn tìm kiếm tất cả các tệp đệ quy trong path_A:

find path_A -name "*AAA*" -exec mv {} path_B \;

Đây không phải là một cú pháp cụ thể của OS X, nó giống với bất kỳ findtôi đã sử dụng. Điểm hay: -maxdepth(đặc biệt nếu path_B là thư mục con - tránh mvcố gắng di chuyển các tệp đã có ở đó!) Và sử dụng \; (vì vậy {} không phải là tham số cuối cùng và mvcó thể sử dụng cú pháp bình thường )
drevicko

6

Đây -execlà cách tốt nhất để làm điều này. Nếu vì lý do nào, đây không phải là một tùy chọn, bạn cũng có thể đọc kết quả trong một vòng lặp:

find path_A -name "*AAA*" -print0 | 
    while IFS= read -r -d $'\0' file; do mv "$file" path_B; done

Đó là cách an toàn, nó có thể xử lý các tên tệp có chứa khoảng trắng, dòng mới hoặc các ký tự lạ khác. Một cách đơn giản hơn, nhưng một cách không thành công trừ khi tên tệp của bạn chỉ bao gồm các ký tự chữ và số đơn giản , là

mv $(find path_A -name "*AAA*") path_B

Nhưng sử dụng vòng lặp while.


Cách đơn giản hơn vẫn có thể thất bại nếu your file names consist only of simple alphanumeric characters, ví dụ: nếu đạtARG_MAX


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.