python开发目录合并小工具 PathMerge
前言
这个程序陆陆续续开发了几天,正好我在学Python,就一边做一边学,倒是学到不少东西。
不得不说python是快速开发的好工具。
程序做了一些改进,这两天又忙着毕设,现在才想起来发到博客上。想想回头github也要启用起来,不能继续荒废了……
如果嫌运行python程序不方便的话,可以直接使用exe可执行文件版本。
exe发布器在这里:使用pyInstaller发布PathMerge的exe版本(py转换成exe)
废话不多说,我介绍一下这个工具。
应用场景
在日常使用中,我们往往需要把一个文件夹拷到另一个文件夹下面,做一些修改之后,再将改动的文件夹合并回去。
例如,从优盘中把一个要用的文件夹拷到电脑上,在pc机上工作,之后你需要将修改后的文件夹再拷回优盘中带走。
这个时候问题来了。
如果你做的修改不多,我们可以简单的把修改的文件复制回去,这个时候你清楚的记得你到底修改了哪些文件。
但是如果你对很多文件做了改动,并且还创建了几个文件或者文件夹,导致目录结构变得十分复杂,最后你自己都记不起来到底哪些文件做了哪些改动,目录做了哪些更新,这个时候你要把文件夹合并回去就有些凌乱了……
So,你需要一个工具去帮你记忆你做的改动,自动的帮你合并文件夹,并且不会丢失任何文件。
这个时候这个小工具就可以派上用场了~
下面演示一下怎么用。
使用演示
现在有两个工作空间

工作空间1是你之前做好的一个工程或者文件夹,里面放满了文件,目录结构复杂。


详细目录:

工作空间2是你现在要工作的目录,现在是空的。

我们把要用的文件夹拷过来

并在里面做一些改动。
这里不放图了,总之我创建了一个文件夹,创建了两个文件,修改了三个文件……现在目录结构变成这样:

大家看到,1/下多了一个文件夹33,里面多了两个文件111.txt和222.txt,我还修改了一个txt文件,一个ppt文件和一个word文件。
下面运行PathMerge:

输入A目录(修改的文件夹)和B目录(原文件夹),要将A==>B,A合并到B。

回车开始合并:

合并成功!
合并的记录都列在上面了,如果名称相同的文件会将旧文件创建一个副本,修改后的文件直接复制过去,程序中没有删除文件的代码,可放心食用。
上图可知复制了三个文件,创建了三个旧文件副本,另外复制了创建的目录。
将工作空间2中1文件做的所有改动都包含了进去。
以上,就是PathMerge的使用演示。
代码
# coding=gbk # PathMerge.py
# 作者:freecode
# 创建时间:2016.4.9 20:15
# 作用:
# 合并并两个目录/文件夹。
# 将目录A合并到目录B,同级目录下,
# 将A中有,B中没有的目录完全复制到B中;
# 将A中没有,B中有的目录不做改动;
# 将A中做了修改的文件,在B的同级目录下创建一个副本。(注意不是覆盖)
#
# 适应场景:
# 一般A是从B复制过来的文件夹,做了修改后,想合并回B。
# 这样在A中做的一些改动我们就不知道了,这个程序的作用是,
# 将A中修改的部分在B中更新。
#
# 版本:
# 将修改后的文件完全复制过去
# 旧的文件创建副本
# 基于文件的MD5值判断是否修改过
# import os
import shutil
import time
import hashlib
import sys def Help(): # 输出帮助文档
print """
PathMerge.py
作者:freecode
创建时间:2016.4.9 20:15
作用:
合并并两个目录/文件夹。
将目录A合并到目录B,同级目录下,
将A中有,B中没有的目录完全复制到B中;
将A中没有,B中有的目录不做改动;
将A中做了修改的文件,在B的同级目录下创建一个副本。(注意不是覆盖) 适应场景:
一般A是从B复制过来的文件夹,做了修改后,想合并回B。
这样在A中做的一些改动我们就不知道了,这个程序的作用是,
将A中修改的部分在B中更新。 版本:
将修改后的文件完全复制过去
旧的文件创建副本
基于文件的MD5值判断是否修改过
""" def GetFileMd5(filename): # 计算文件的md5值
if not os.path.isfile(filename):
return
myhash = hashlib.md5()
f = file(filename,'rb')
while True:
b = f.read(8096)
if not b :
break
myhash.update(b)
f.close()
return myhash.hexdigest() def isModify(A_file,B_file): # 判断两个文件是否相同,如果不同,表示修改过
# 参数需是绝对路径
return GetFileMd5(A_file) != GetFileMd5(B_file) def Stamp2Time(Stamp): # 将时间戳转换成时间显示格式
timeArray = time.localtime(Stamp)
Time = time.strftime("%Y年%m月%d日 %H时%M分%S秒 旧文件副本", timeArray)
return Time def Merge(A_path,B_path): # 合并两个目录
B_paths = os.listdir(B_path) # 获取当前B中的目录结构
for fp in os.listdir(A_path): # 遍历当前A目录中的文件或文件夹
A_new_path = os.path.join(A_path,fp) # A中的文件或目录
B_new_path = os.path.join(B_path,fp) # B中对应的文件或路径,不一定存在 if os.path.isdir(A_new_path): # A中的目录
if os.path.exists(B_new_path): # 如果在B中存在
Merge(A_new_path,B_new_path) # 继续合并下一级目录
else: # 如果在B中不存在
print '[目录]\t%s ===> %s' %(A_new_path,B_new_path)
shutil.copytree(A_new_path,B_new_path) # 完全复制目录到B elif os.path.isfile(A_new_path): # A中的文件
if os.path.exists(B_new_path): # 如果在B中存在
s = os.stat(B_new_path)
if isModify(A_new_path,B_new_path) == True: # 如果该文件修改过
# 创建副本
suffix = B_new_path.split('.')[-1] # 得到文件的后缀名
# 将B中原文件创建副本
B_copy_path = B_new_path[:-len(suffix)-1]+"(%s)."%(Stamp2Time(s.st_mtime))+suffix
print '[副本]\t%s ===> %s' %(A_new_path,B_copy_path)
shutil.copy2(B_new_path,B_copy_path)
# 将A中修改后文件复制过来
print '[文件]\t%s ===> %s' %(A_new_path,B_new_path)
shutil.copy2(A_new_path,B_new_path)
else: # 如果该文件没有修改过
pass # 不复制 else: # 如果在B中不存在
# 将该文件复制过去
print '[文件]\t%s ===> %s' %(A_new_path,B_new_path)
shutil.copy2(A_new_path,B_new_path) # 运行模式
if __name__=='__main__':
print """
欢迎使用PathMerge!
本程序将会把目录A合并到目录B,即 A ===> B
将A目录中修改的内容在B目录中更新
合并规则具体见 PathMerge.Help()
"""
if len(sys.argv) == 1:
path1 = raw_input('请输入A目录:').strip()
path2 = raw_input('请输入B目录:').strip()
elif len(sys.argv) == 2:
path1 = sys.argv[1].strip()
print 'A目录为:%s\n' % (path1)
path2 = raw_input('请输入B目录:').strip()
elif len(sys.argv) == 3:
path1 = sys.argv[1].strip()
print 'A目录为:%s\n' % (path1)
path2 = sys.argv[2].strip()
print 'B目录为:%s\n' % (path2)
else:
print 'ERROR:参数错误!\n参数最多有三个!\n'
raw_input('\n请按回车键(Enter)退出……')
sys.exit(0)
# 去除目录的引号
if path1[0]=='\"':
path1 = path1[1:-1]
if path2[0]=='\"':
path2 = path2[1:-1] print """
开始合并目录 %s
到目录 %s
%s ===> %s
""" %(path1,path2,path1,path2) try:
print '合并中……'
Merge(path1,path2)
print ''
except Exception,e:
print '合并失败!'
print '失败原因:\n',e
else:
print '合并成功!' raw_input('\n请按回车键(Enter)退出……')
工具特性
1、可识别加“”的目录和不加“”的目录。
2、使用MD5识别文件是否修改。
MD5识别文件的内容见上一篇博文:python计算文件的md5值
3、可传递参数,用来写脚本。例:


4、原文件夹内的旧文件不会被删除,而是创建一个带有时间戳的副本。修改过的文件直接复制过去。
这样不用担心合并之后文件莫名其妙的丢失啦。使用该工具不会有文件删除操作。
注意
Sorry,目前仅支持windows环境……
参考资料
原创声明
Freecode# : www.cnblogs.com/yym2013
python开发目录合并小工具 PathMerge的更多相关文章
- Python开发目录
Python开发目录 Python之三目运算符 Python之文件的基本操作
- Python 使用 PyQt5 开发的关机小工具
前两天简单认识了一下PyQt5,通过练习开发了一款在Window下自定义关机的小工具,代码如下 import os,sys,time from PyQt5 import QtCore,QtWidget ...
- supervisor python开发的进程管理工具
Supervisor (http://supervisord.org) 是一个用 Python 写的进程管理工具,可以很方便的用来启动.重启.关闭进程(不仅仅是 Python 进程).除了对单个进程的 ...
- python 3.6 MJ小工具
2017.07.14 update 做了个界面,不需要使用cmd命令行+文件路径的方式来使用了: 链接如下: http://www.cnblogs.com/chenyuebai/p/7150382.h ...
- 几款Android开发人员必备小工具
在这里我介绍一下我常常在Android Studio里面使用的小工具吧,这些工具都能够在plugin里面搜索到. (当然了哈.我也是从网上找的.用着挺方便的,在这里总结一下) Gsonformat: ...
- WPF开发查询加班小工具
先说一下,我们公司是六点下班,超过7点开始算加班,但是加班的时间是从六点开始计算,以0.5个小时为计数,就是你到了六点半,不算加班半小时,但是加班到七点半,就是加班了一个半小时. 一.打卡记录 首先, ...
- Python开发——目录
Python基础 Python开发——解释器安装 Python开发——基础 Python开发——变量 Python开发——[选择]语句 Python开发——[循环]语句 Python开发——数据类型[ ...
- 【实战小项目】python开发自动化运维工具--批量操作主机
有很多开源自动化运维工具都很好用如ansible/salt stack等,完全不用重复造轮子.只不过,很多运维同学学习Python之后,苦于没小项目训练.本篇就演示用Python写一个批量操作主机的工 ...
- Python开发接水果小游戏
我研发的Python游戏引擎Pylash已经更新到1.4了.如今我们就来使用它完毕一个极其简单的小游戏:接水果. 下面是游戏截图: 游戏操作说明:点击屏幕左右两边或者使用键盘方向键控制人物移动.使人物 ...
随机推荐
- android 图像处理系列合集
为了便于大家对滤镜算法的学习,以后发布的图像处理滤镜系列帖子会在这里汇总,本人第一次写合集,写得不好的地方大家请见谅,手头上虽然有一些滤镜的算法,但是大多不是android版的,教程里的代码大多是我借 ...
- MySQL安装步骤
MySQL安装步骤 1. 下载MySQL Community Server 5.6.21,注意选择系统类型(32位/64位) 2. 解压MySQL压缩包 将以下载的MySQL压缩包解压到自定义目录下. ...
- 解决:WPS for Linux提示“系统缺失字体symbol、wingdings、wingdings 2、wingdings 3、webding”
WPS for Linux提示“系统缺失字体symbol.wingdings.wingdings 2.wingdings 3.webding” 出现提示的原因是因为WPS for Linux没有自带以 ...
- JAVA之IO流(字节流)
输入和输出 JAVA的流分为输入流和输出流两部分, 输入流:InputStream或者Reader:从文件中读到程序中: 输出流:OutputStream或者Writer:从程序中输出到文件中: Re ...
- Alpha阶段第一次Scrum Meeting
Scrum Meeting Alpha 情况简述 Alpha阶段第一次Scrum Meeting 敏捷开发起始时间 2016/10/18 00:00 敏捷开发终止时间 2016/10/20 00: ...
- js获取url信息
设置或获取对象指定的文件名或路径. alert(window.location.pathname) 设置或获取整个 URL 为字符串. alert(window.location.href); 设置或 ...
- Android接入支付宝和银联
支付宝接入参考链接:https://software.intel.com/zh-cn/node/542608 银联接入参考链接:http://blog.csdn.net/gf771115/articl ...
- ThinkPhp 验证码不显示图片
验证码不显示原因分析: 1.先确保是否开启了 gd库 2.页面是否存在bom头 3.导入的ORG.Util.Image是否存在 4.入口文件中是否有define('APP_DEBUG', TRUE); ...
- thinkphp一句话疑难解决笔记 2
php中的_ _call()方法? 它是php5后为对象 类 新增的一个自动方法. 它会监视类的其他方法的调用, 当调用类的不存在的方法时, 会自动调用类的__call方法. tp的 "命名 ...
- DP专辑
今天练了一波DP.时间紧迫我就只贴代码了. 20141120 fzu2129 http://acm.fzu.edu.cn/problem.php?pid=2129 不同的子序列个数 //#pragma ...