Shell 命令行,写一个自动整理 ~/Downloads/ 文件夹下文件的脚本

Shell 命令行,写一个自动整理 ~/Downloads/ 文件夹下文件的脚本

mac 或者 linux 系统中,我们的浏览器或者其他下载软件下载的文件全部都下载再 ~/Downloads/ 文件夹下面。日积月累,我们的文件会越来越多。手工整理这些文件是比较繁琐的工作,于是,我就思考,我能不能用 shell 来自动整理这些下载的文件。

说干就干。

为了避免破坏我本地的文件,我自己搞了一个虚拟机跑了一个 centos 然后写下了如下的脚本。

第一版代码(处理文件名带空格的文件会出错)

#!/bin/bash
# Name cleardown
# Description move you files in ~/Downloads to ~/Documents/OfficeFiles
# Author FungLeo
# WebSite http://blog.csdn.net/fungleo

# find .  -maxdepth 1 -type f
# Excel  Other  PDF  Photo  PPT  Word  Xmind  Zip

# 设定要整理的文件夹为下载目录
downFinder=~/Downloads/
# 看看下载目录根目录下有哪些文件
dfiles=$(ls -l $downFinder | grep ^- | sed 's/[ ][ ]*/,/g' | awk -F "," '{print $NF}')
# 准备将这些文件处理到哪里去
filesFinder=~/Documents/OfficeFiles/

# 分辨文件类型,并给出放到哪里去的建议。这里大家可以根据自己的需求完善 case 语句
function fileType() {
  case $1 in
    'jpg' | 'png' | 'gif' | 'jpeg' | 'bmp')
      echo 'Photo'
    ;;
    'doc' | 'docx')
      echo 'Word'
    ;;
    'xls' | 'xlsx')
      echo 'Excel'
    ;;
    'ppt' | 'pptx')
      echo 'PPT'
    ;;
    'zip' | '7z' | 'rar')
      echo 'Zip'
    ;;
    'xmind')
      echo 'Xmind'
    ;;
    'pdf')
      echo 'PDF'
    ;;
    *)
      echo 'Other'
    ;;
  esac
}

# 判断目标文件夹中是否包含这个文件
function hasfile() {
  if [ -f $1 ]; then
    echo 'has'
  else
    echo 'nohas'
  fi
}

# 给文件重新命个名字,我这里是在后面加了一个 1
function changeFileName() {
  local filename=$(basename $1)
  echo ${filename%.*}1.${filename##*.}
}

# 开始搬文件的函数
function mvFile() {
  # 这个函数需要传两个参数,一个是原文件名,一个是新文件名。
  local name=$1
  local newname=$2
  # 获取文件的后缀名,并且转化为小写
  local type=$(echo $1 | awk -F "." '{print $NF}' | tr "[:upper:]" "[:lower:]" )
  local classify=$(fileType $type)
  local file=$filesFinder$classify'/'$newname
  # 判断新文件名在目标地址是否有同名文件
  local hasf=$(echo $(hasfile $file))
  if [ $hasf = 'has' ]; then
    mvFile $name $(changeFileName $newname)
  else
    mv $downFinder$name $file
  fi
}

# 循环这些文件,并且进行处理
for i in $dfiles; do
  mvFile $i $i
done

小结

其中还是使用到了很多的知识点的。

  1. case 语句。一开始用 if 判断,越写越丑。查了下 case 语句,果然清爽很多了。
  2. 获取文件后缀名。本例中用了两种方法。
    1. awk 方法。awk -F "." '{print $NF}'. 分割取最后一个。
    2. ${filename##*.} 取后缀名。${filename%.*} 取文件名
  3. 函数的写法。其实不写 function 也是可以的。
  4. 函数自己调自己,和 js 也没太大区别嘛。
  5. 字符串大小写转换 tr "[:upper:]" "[:lower:]"

2017年08月08日补充,解决文件名中包含空格的问题

#!/bin/bash
# Name cleardown
# Description move you files in ~/Downloads to ~/Documents/Office Files
# Author FungLeo
# WebSite http://blog.csdn.net/fungleo

# find .  -maxdepth 1 -type f
# Excel  Other  PDF  Photo  PPT  Word  Xmind  Zip

# 设定要整理的文件夹为下载目录
downFinder="${HOME}/Downloads/"
# 准备将这些文件处理到哪里去
filesFinder="${HOME}/Documents/Office Files/"

# 分辨文件类型,并给出放到哪里去的建议。这里大家可以根据自己的需求完善 case 语句
function fileType() {
  case $1 in
    'jpg' | 'png' | 'gif' | 'jpeg' | 'bmp')
      echo 'Photo'
    ;;
    'doc' | 'docx')
      echo 'Word'
    ;;
    'xls' | 'xlsx')
      echo 'Excel'
    ;;
    'ppt' | 'pptx')
      echo 'PPT'
    ;;
    'zip' | '7z' | 'rar')
      echo 'Zip'
    ;;
    'xmind')
      echo 'Xmind'
    ;;
    'pdf')
      echo 'PDF'
    ;;
    *)
      echo 'Other'
    ;;
  esac
}

# 判断目标文件夹中是否包含这个文件
function hasfile() {
  if [ -f $1 ]; then
    echo 'has'
  else
    echo 'nohas'
  fi
}

# 给文件重新命个名字,我这里是在后面加了一个 1
function changeFileName() {
  local filename=$(basename $1)
  echo ${filename%.*}1.${filename##*.}
}

# 开始搬文件的函数
function mvFile() {
  # 这个函数需要传两个参数,一个是原文件名,一个是新文件名。
  local name=$1
  local newname=$2
  # 获取文件的后缀名,并且转化为小写
  local type=$(echo $1 | awk -F "." '{print $NF}' | tr "[:upper:]" "[:lower:]" )
  local classify=$(fileType $type)
  local file="$filesFinder$classify/$newname"
  # 判断新文件名在目标地址是否有同名文件
  local hasf=$(echo $(hasfile $file))
  if [ $hasf = 'has' ]; then
    mvFile $name $(changeFileName $newname)
  else
    mv "$downFinder$name" "$file"
  fi
}

# 设置分隔符为换行
OLD_IFS=$IFS
IFS=$'\n'
# 循环这些文件,并且进行处理
for i in $(find "$downFinder" -maxdepth 1 -type f -not -name ".*" | awk -F "/" '{print $NF}'); do
  mvFile $i $i
done

# 将分隔符设置为默认,以免影响后面的程序
IFS=$OLD_IFS

补充小结

  1. 我先前没有考虑文件夹或者文件包含空格的情况,导致很多问题。
    1. 当把路径用引号 " 引起来的时候,不会解析 ~ 所以要用 ${HOME} 代替
    2. 在引用变量的时候,变量也要用引号引起来。注意,不能是单引号。
  2. 默认分隔符为空白,包含:空格、制表符、回车符,用 IFS 表示。
  3. echo 最终命令,和实际执行的结果不一定是一致的。
  4. find . -not -name ".*" 表示不找隐藏文件

以上脚本均在 centos 和 MAC 下测试通过,在其他 linux 下可能会有稍许不同。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java进阶架构师

解密Dubbo:自己动手编写一个较为完善的RPC框架(两万字干货)

现在很多企业都在使用Dubbo或者Spring Cloud做企业的微服务架构,其实对于Dubbo最核心的技术就是RPC调用,现在我们就来动手自己编写一个RPC框...

25650
来自专栏技术墨客

Hazelcast集群服务(2)——Hazelcast基本配置

    在入门及使用案例一文介绍了什么是Hazelcast,并展示了一个简单的使用例子。原理大家都懂了,后面的篇章会给兄弟们更多干货。

52330
来自专栏chenssy

【死磕Sharding-jdbc】---基于ssm

本篇文章讲解如何在ssm(spring、springmvc、mybatis)结构的程序上集成sharding-jdbc(版本为1.5.4.1)进行分库分表; 假...

20820
来自专栏C/C++基础

Linux命令(29)——ls命令

ls命令用来显示目录内容,在Linux中是使用率较高的命令。ls命令的输出信息可以进行彩色加亮显示,以分区不同类型的文件。

7910
来自专栏java 成神之路

基于ReentrantLock发生死锁的解决方案

33560
来自专栏Danny的专栏

System.AccessViolationException”类型的未经处理的异常在 System.Data.dll 中发生。其他信息:尝试读取或写入受保护的内存。这通常指示其他内存已损坏。

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/huyuyang6688/article/...

17020
来自专栏人工智能LeadAI

ElasticSearch优化系列二:机器设置(内存)

预留一半内存给Lucence使用 一个常见的问题是配置堆太大。你有一个64 GB的机器,觉得JVM内存越大越好,想给Elasticsearch所有64 GB的内...

47340
来自专栏知识分享

关于原子哥ENC28J60网络通信模块接收数据代码的一点疑惑

---恢复内容开始--- 这几天做STM32的ENC28J60网络通信模块,自己在原子哥的代码上进行修改测试,,发现一个问题,电脑和板子进行通信的时候总隔一段时...

37580
来自专栏逢魔安全实验室

微软公式编辑器系列漏洞分析(一):CVE-2017-11882

? 0x00 简介 CVE-2017-11882为Office内存破坏漏洞。攻击者可以利用漏洞以当前登录的用户的身份执行任意命令。所影响的组件是Office...

35150
来自专栏JackieZheng

Spring实战——缓存

缓存 提到缓存,你能想到什么?一级缓存,二级缓存,web缓存,redis…… 你所能想到的各种包罗万象存在的打着缓存旗号存在的各种技术或者实现,无非都是宣扬缓...

204100

扫码关注云+社区

领取腾讯云代金券