高级的 文件、文件夹、压缩包 处理模块

shutil.copyfileobj(fsrc, fdst[, length])
复制文件内容(不包含元数据)从类文件对象src到类文件对dst。可选参数length指定缓冲区的大小,负数表示一次性读入。默认会把数据切分成小块拷贝,以免占用太多内存。注意:拷贝是从fsrc的当前文件开始,必须先打开文件

1 def copyfileobj(fsrc, fdst, length=16*1024):2     """copy data from file-like object fsrc to file-like object fdst"""
3     while 1:4         buf =fsrc.read(length)5         if notbuf:6             break
7         fdst.write(buf)

View Code

1 f1 = open("a.txt","r",encoding="utf-8")2 f2 = open("b.txt","w",encoding="utf-8")3 shutil.copyfileobj(f1,f2)

shutil.copyfile(src, dst)
拷贝文件,调用了底函数层copyfileobj(),不用再打开文件,如果是同一个文件,会报错

1 defcopyfile(src, dst):2     """Copy data from src to dst"""
3     if_samefile(src, dst):4         raise Error("`%s` and `%s` are the same file" %(src, dst))5
6     for fn in[src, dst]:7         try:8             st =os.stat(fn)9         exceptOSError:10             #File most likely does not exist
11             pass
12         else:13             #XXX What about other special files? (sockets, devices...)
14             ifstat.S_ISFIFO(st.st_mode):15                 raise SpecialFileError("`%s` is a named pipe" %fn)16
17     with open(src, 'rb') as fsrc:18         with open(dst, 'wb') as fdst:19             copyfileobj(fsrc, fdst)

View Code

shutil.copyfile("a.txt","b.txt")

shutil.copymode(src, dst)
仅拷贝权限。内容、组、用户均不变

1 defcopymode(src, dst):2     """Copy mode bits from src to dst"""
3     if hasattr(os, 'chmod'):4         st =os.stat(src)5         mode =stat.S_IMODE(st.st_mode)6         os.chmod(dst, mode)

View Code

shutil.copystat(src, dst)
拷贝状态的信息,包括:mode bits, atime, mtime, flags

1 defcopystat(src, dst):2     """Copy all stat info (mode bits, atime, mtime, flags) from src to dst"""
3     st =os.stat(src)4     mode =stat.S_IMODE(st.st_mode)5     if hasattr(os, 'utime'):6 os.utime(dst, (st.st_atime, st.st_mtime))7     if hasattr(os, 'chmod'):8 os.chmod(dst, mode)9     if hasattr(os, 'chflags') and hasattr(st, 'st_flags'):10         try:11 os.chflags(dst, st.st_flags)12         exceptOSError, why:13             for err in 'EOPNOTSUPP', 'ENOTSUP':14                 if hasattr(errno, err) and why.errno ==getattr(errno, err):15                     break
16             else:17                 raise

View Code

shutil.copy(src, dst)
拷贝文件和权限

1 defcopy(src, dst):2     """Copy data and mode bits ("cp src dst").3
4 The destination may be a directory.5
6     """
7     ifos.path.isdir(dst):8         dst =os.path.join(dst, os.path.basename(src))9 copyfile(src, dst)10     copymode(src, dst)

View Code

shutil.copy2(src, dst)
拷贝文件和状态信息,实际上先调用shutil.copy,然后使用copystat

1 defcopy2(src, dst):2     """Copy data and all stat info ("cp -p src dst").3
4 The destination may be a directory.5
6     """
7     ifos.path.isdir(dst):8         dst =os.path.join(dst, os.path.basename(src))9 copyfile(src, dst)10     copystat(src, dst)

View Code

shutil.ignore_patterns(*patterns)
shutil.copytree(src, dst, symlinks=False, ignore=None)
递归的去拷贝文件

例如:copytree(source, destination, ignore=ignore_patterns('*.pyc', 'tmp*'))

1 def ignore_patterns(*patterns):2     """Function that can be used as copytree() ignore parameter.3
4 Patterns is a sequence of glob-style patterns5 that are used to exclude files"""
6     def_ignore_patterns(path, names):7         ignored_names =[]8         for pattern inpatterns:9 ignored_names.extend(fnmatch.filter(names, pattern))10         returnset(ignored_names)11     return_ignore_patterns12
13 def copytree(src, dst, symlinks=False, ignore=None):14     """Recursively copy a directory tree using copy2().15
16 The destination directory must not already exist.17 If exception(s) occur, an Error is raised with a list of reasons.18
19 If the optional symlinks flag is true, symbolic links in the20 source tree result in symbolic links in the destination tree; if21 it is false, the contents of the files pointed to by symbolic22 links are copied.23
24 The optional ignore argument is a callable. If given, it25 is called with the `src` parameter, which is the directory26 being visited by copytree(), and `names` which is the list of27 `src` contents, as returned by os.listdir():28
29 callable(src, names) -> ignored_names30
31 Since copytree() is called recursively, the callable will be32 called once for each directory that is copied. It returns a33 list of names relative to the `src` directory that should34 not be copied.35
36 XXX Consider this example code rather than the ultimate tool.37
38     """
39     names =os.listdir(src)40     if ignore is notNone:41         ignored_names =ignore(src, names)42     else:43         ignored_names =set()44
45 os.makedirs(dst)46     errors =[]47     for name innames:48         if name inignored_names:49             continue
50         srcname =os.path.join(src, name)51         dstname =os.path.join(dst, name)52         try:53             if symlinks andos.path.islink(srcname):54                 linkto =os.readlink(srcname)55 os.symlink(linkto, dstname)56             elifos.path.isdir(srcname):57 copytree(srcname, dstname, symlinks, ignore)58             else:59                 #Will raise a SpecialFileError for unsupported file types
60 copy2(srcname, dstname)61         #catch the Error from the recursive copytree so that we can
62         #continue with other files
63         exceptError, err:64 errors.extend(err.args[0])65         exceptEnvironmentError, why:66 errors.append((srcname, dstname, str(why)))67     try:68 copystat(src, dst)69     exceptOSError, why:70         if WindowsError is not None andisinstance(why, WindowsError):71             #Copying file access times may fail on Windows
72             pass
73         else:74 errors.append((src, dst, str(why)))75     iferrors:76         raise Error, errors

View Code

1 shutil.copytree("proj","test2")#把文件夹proj中的目录全部复制到test2中
2 shutil.copytree("proj","test3",ignore=shutil.ignore_patterns("__init__.py","urls.py"))#除ignore中的两个文件外,把文件夹proj中的目录全部复制到test2中

shutil.rmtree(path[, ignore_errors[, onerror]])
递归的去删除文件

1 def rmtree(path, ignore_errors=False, οnerrοr=None):2     """Recursively delete a directory tree.3
4 If ignore_errors is set, errors are ignored; otherwise, if onerror5 is set, it is called to handle the error with arguments (func,6 path, exc_info) where func is os.listdir, os.remove, or os.rmdir;7 path is the argument to that function that caused it to fail; and8 exc_info is a tuple returned by sys.exc_info().  If ignore_errors9 is false and onerror is None, an exception is raised.10
11     """
12     ifignore_errors:13         def onerror(*args):14             pass
15     elif onerror isNone:16         def onerror(*args):17             raise
18     try:19         ifos.path.islink(path):20             #symlinks to directories are forbidden, see bug #1669
21             raise OSError("Cannot call rmtree on a symbolic link")22     exceptOSError:23 onerror(os.path.islink, path, sys.exc_info())24         #can't continue even if onerror hook returns
25         return
26     names =[]27     try:28         names =os.listdir(path)29     exceptos.error, err:30 onerror(os.listdir, path, sys.exc_info())31     for name innames:32         fullname =os.path.join(path, name)33         try:34             mode =os.lstat(fullname).st_mode35         exceptos.error:36             mode =037         ifstat.S_ISDIR(mode):38 rmtree(fullname, ignore_errors, onerror)39         else:40             try:41 os.remove(fullname)42             exceptos.error, err:43 onerror(os.remove, fullname, sys.exc_info())44     try:45 os.rmdir(path)46     exceptos.error:47         onerror(os.rmdir, path, sys.exc_info())

View Code

shutil.rmtree("test3")

shutil.move(src, dst)
递归的去移动文件,相当于剪切

1 defmove(src, dst):2     """Recursively move a file or directory to another location. This is3 similar to the Unix "mv" command.4
5 If the destination is a directory or a symlink to a directory, the source6 is moved inside the directory. The destination path must not already7 exist.8
9 If the destination already exists but is not a directory, it may be10 overwritten depending on os.rename() semantics.11
12 If the destination is on our current filesystem, then rename() is used.13 Otherwise, src is copied to the destination and then removed.14 A lot more could be done here...  A look at a mv.c shows a lot of15 the issues this implementation glosses over.16
17     """
18     real_dst =dst19     ifos.path.isdir(dst):20         if_samefile(src, dst):21             #We might be on a case insensitive filesystem,
22             #perform the rename anyway.
23 os.rename(src, dst)24             return
25
26         real_dst =os.path.join(dst, _basename(src))27         ifos.path.exists(real_dst):28             raise Error, "Destination path '%s' already exists" %real_dst29     try:30 os.rename(src, real_dst)31     exceptOSError:32         ifos.path.isdir(src):33             if_destinsrc(src, dst):34                 raise Error, "Cannot move a directory '%s' into itself '%s'." %(src, dst)35             copytree(src, real_dst, symlinks=True)36 rmtree(src)37         else:38 copy2(src, real_dst)39             os.unlink(src)

View Code

shutil.make_archive(base_name, format,...)

创建压缩包并返回文件路径,例如:zip、tar

  • base_name: 压缩包的文件名,也可以是压缩包的路径。只是文件名时,则保存至当前目录,否则保存至指定路径,
    如:www                        =>保存至当前路径
    如:/Users/wupeiqi/www =>保存至/Users/wupeiqi/
  • format: 压缩包种类,“zip”, “tar”, “bztar”,“gztar”
  • root_dir: 要压缩的文件夹路径(默认当前目录)
  • owner: 用户,默认当前用户
  • group: 组,默认当前组
  • logger: 用于记录日志,通常是logging.Logger对象
1
2
3
4
5
6
7
8
9
#将 /Users/wupeiqi/Downloads/test 下的文件打包放置当前程序目录
import shutil
ret = shutil.make_archive("wwwwwwwwww"'gztar', root_dir='/Users/wupeiqi/Downloads/test')
#将 /Users/wupeiqi/Downloads/test 下的文件打包放置 /Users/wupeiqi/目录
import shutil
ret = shutil.make_archive("/Users/wupeiqi/wwwwwwwwww"'gztar', root_dir='/Users/wupeiqi/Downloads/test')

1 def make_archive(base_name, format, root_dir=None, base_dir=None, verbose=0,2                  dry_run=0, owner=None, group=None, logger=None):3     """Create an archive file (eg. zip or tar).4
5 'base_name' is the name of the file to create, minus any format-specific6 extension; 'format' is the archive format: one of "zip", "tar", "bztar"7 or "gztar".8
9 'root_dir' is a directory that will be the root directory of the10 archive; ie. we typically chdir into 'root_dir' before creating the11 archive.  'base_dir' is the directory where we start archiving from;12 ie. 'base_dir' will be the common prefix of all files and13 directories in the archive.  'root_dir' and 'base_dir' both default14 to the current directory.  Returns the name of the archive file.15
16 'owner' and 'group' are used when creating a tar archive. By default,17 uses the current owner and group.18     """
19     save_cwd =os.getcwd()20     if root_dir is notNone:21         if logger is notNone:22             logger.debug("changing into '%s'", root_dir)23         base_name =os.path.abspath(base_name)24         if notdry_run:25 os.chdir(root_dir)26
27     if base_dir isNone:28         base_dir =os.curdir29
30     kwargs = {'dry_run': dry_run, 'logger': logger}31
32     try:33         format_info =_ARCHIVE_FORMATS[format]34     exceptKeyError:35         raise ValueError, "unknown archive format '%s'" %format36
37     func =format_info[0]38     for arg, val in format_info[1]:39         kwargs[arg] =val40
41     if format != 'zip':42         kwargs['owner'] =owner43         kwargs['group'] =group44
45     try:46         filename = func(base_name, base_dir, **kwargs)47     finally:48         if root_dir is notNone:49             if logger is notNone:50                 logger.debug("changing back to '%s'", save_cwd)51 os.chdir(save_cwd)52
53     return filename

View Code

shutil.make_archive("crm/test2","zip","test2")

shutil 对压缩包的处理是调用 ZipFile 和 TarFile 两个模块来进行的,详细:

TarFile :只打包不压缩

1 importzipfile2
3 #压缩
4 z = zipfile.ZipFile('laxi.zip', 'w')5 z.write('a.log')6 z.write('data.data')7 z.close()8
9 #解压
10 z = zipfile.ZipFile('laxi.zip', 'r')11 z.extractall()12 z.close()13
14 zipfile 压缩解压15
16 zipfile 压缩解压

zipfile 压缩解压

1 importtarfile2
3 #压缩
4 tar = tarfile.open('your.tar','w')5 tar.add('/Users/wupeiqi/PycharmProjects/bbs2.zip', arcname='bbs2.zip')6 tar.add('/Users/wupeiqi/PycharmProjects/cmdb.zip', arcname='cmdb.zip')7 tar.close()8
9 #解压
10 tar = tarfile.open('your.tar','r')11 tar.extractall()  #可设置解压地址
12 tar.close()13
14 tarfile 压缩解压15
16 tarfile 压缩解压

tarfile 压缩解压

1 classZipFile(object):2     """Class with methods to open, read, write, close, list zip files.3
4 z = ZipFile(file, mode="r", compression=ZIP_STORED, allowZip64=False)5
6 file: Either the path to the file, or a file-like object.7 If it is a path, the file will be opened and closed by ZipFile.8 mode: The mode can be either read "r", write "w" or append "a".9 compression: ZIP_STORED (no compression) or ZIP_DEFLATED (requires zlib).10 allowZip64: if True ZipFile will create files with ZIP64 extensions when11 needed, otherwise it will raise an exception when this would12 be necessary.13
14     """
15
16     fp = None                   #Set here since __del__ checks it
17
18     def __init__(self, file, mode="r", compression=ZIP_STORED, allowZip64=False):19         """Open the ZIP file with mode read "r", write "w" or append "a"."""
20         if mode not in ("r", "w", "a"):21             raise RuntimeError('ZipFile() requires mode "r", "w", or "a"')22
23         if compression ==ZIP_STORED:24             pass
25         elif compression ==ZIP_DEFLATED:26             if notzlib:27                 raiseRuntimeError,\28                       "Compression requires the (missing) zlib module"
29         else:30             raise RuntimeError, "That compression method is not supported"
31
32         self._allowZip64 =allowZip6433         self._didModify =False34         self.debug = 0  #Level of printing: 0 through 3
35         self.NameToInfo = {}    #Find file info given name
36         self.filelist = []      #List of ZipInfo instances for archive
37         self.compression = compression  #Method of compression
38         self.mode = key = mode.replace('b', '')[0]39         self.pwd =None40         self._comment = ''
41
42         #Check if we were passed a file-like object
43         ifisinstance(file, basestring):44             self._filePassed =045             self.filename =file46             modeDict = {'r' : 'rb', 'w': 'wb', 'a' : 'r+b'}47             try:48                 self.fp =open(file, modeDict[mode])49             exceptIOError:50                 if mode == 'a':51                     mode = key = 'w'
52                     self.fp =open(file, modeDict[mode])53                 else:54                     raise
55         else:56             self._filePassed = 1
57             self.fp =file58             self.filename = getattr(file, 'name', None)59
60         try:61             if key == 'r':62 self._RealGetContents()63             elif key == 'w':64                 #set the modified flag so central directory gets written
65                 #even if no files are added to the archive
66                 self._didModify =True67             elif key == 'a':68                 try:69                     #See if file is a zip file
70 self._RealGetContents()71                     #seek to start of directory and overwrite
72 self.fp.seek(self.start_dir, 0)73                 exceptBadZipfile:74                     #file is not a zip file, just append
75                     self.fp.seek(0, 2)76
77                     #set the modified flag so central directory gets written
78                     #even if no files are added to the archive
79                     self._didModify =True80             else:81                 raise RuntimeError('Mode must be "r", "w" or "a"')82         except:83             fp =self.fp84             self.fp =None85             if notself._filePassed:86 fp.close()87             raise
88
89     def __enter__(self):90         returnself91
92     def __exit__(self, type, value, traceback):93 self.close()94
95     def_RealGetContents(self):96         """Read in the table of contents for the ZIP file."""
97         fp =self.fp98         try:99             endrec =_EndRecData(fp)100         exceptIOError:101             raise BadZipfile("File is not a zip file")102         if notendrec:103             raise BadZipfile, "File is not a zip file"
104         if self.debug > 1:105             printendrec106         size_cd = endrec[_ECD_SIZE]             #bytes in central directory
107         offset_cd = endrec[_ECD_OFFSET]         #offset of central directory
108         self._comment = endrec[_ECD_COMMENT]    #archive comment
109
110         #"concat" is zero, unless zip was concatenated to another file
111         concat = endrec[_ECD_LOCATION] - size_cd -offset_cd112         if endrec[_ECD_SIGNATURE] ==stringEndArchive64:113             #If Zip64 extension structures are present, account for them
114             concat -= (sizeEndCentDir64 +sizeEndCentDir64Locator)115
116         if self.debug > 2:117             inferred = concat +offset_cd118             print "given, inferred, offset", offset_cd, inferred, concat119         #self.start_dir:  Position of start of central directory
120         self.start_dir = offset_cd +concat121 fp.seek(self.start_dir, 0)122         data =fp.read(size_cd)123         fp =cStringIO.StringIO(data)124         total =0125         while total <size_cd:126             centdir =fp.read(sizeCentralDir)127             if len(centdir) !=sizeCentralDir:128                 raise BadZipfile("Truncated central directory")129             centdir =struct.unpack(structCentralDir, centdir)130             if centdir[_CD_SIGNATURE] !=stringCentralDir:131                 raise BadZipfile("Bad magic number for central directory")132             if self.debug > 2:133                 printcentdir134             filename =fp.read(centdir[_CD_FILENAME_LENGTH])135             #Create ZipInfo instance to store file information
136             x =ZipInfo(filename)137             x.extra =fp.read(centdir[_CD_EXTRA_FIELD_LENGTH])138             x.comment =fp.read(centdir[_CD_COMMENT_LENGTH])139             x.header_offset =centdir[_CD_LOCAL_HEADER_OFFSET]140 (x.create_version, x.create_system, x.extract_version, x.reserved,141 x.flag_bits, x.compress_type, t, d,142                 x.CRC, x.compress_size, x.file_size) = centdir[1:12]143             x.volume, x.internal_attr, x.external_attr = centdir[15:18]144             #Convert date/time code to (year, month, day, hour, min, sec)
145             x._raw_time =t146             x.date_time = ( (d>>9)+1980, (d>>5)&0xF, d&0x1F,147                                      t>>11, (t>>5)&0x3F, (t&0x1F) * 2)148
149 x._decodeExtra()150             x.header_offset = x.header_offset +concat151             x.filename =x._decodeFilename()152 self.filelist.append(x)153             self.NameToInfo[x.filename] =x154
155             #update total bytes read from central directory
156             total = (total + sizeCentralDir +centdir[_CD_FILENAME_LENGTH]157                      +centdir[_CD_EXTRA_FIELD_LENGTH]158                      +centdir[_CD_COMMENT_LENGTH])159
160             if self.debug > 2:161                 print "total", total162
163
164     defnamelist(self):165         """Return a list of file names in the archive."""
166         l =[]167         for data inself.filelist:168 l.append(data.filename)169         returnl170
171     definfolist(self):172         """Return a list of class ZipInfo instances for files in the173 archive."""
174         returnself.filelist175
176     defprintdir(self):177         """Print a table of contents for the zip file."""
178         print "%-46s %19s %12s" % ("File Name", "Modified", "Size")179         for zinfo inself.filelist:180             date = "%d-%02d-%02d %02d:%02d:%02d" % zinfo.date_time[:6]181             print "%-46s %s %12d" %(zinfo.filename, date, zinfo.file_size)182
183     deftestzip(self):184         """Read all the files and check the CRC."""
185         chunk_size = 2 ** 20
186         for zinfo inself.filelist:187             try:188                 #Read by chunks, to avoid an OverflowError or a
189                 #MemoryError with very large embedded files.
190                 with self.open(zinfo.filename, "r") as f:191                     while f.read(chunk_size):     #Check CRC-32
192                         pass
193             exceptBadZipfile:194                 returnzinfo.filename195
196     defgetinfo(self, name):197         """Return the instance of ZipInfo given 'name'."""
198         info =self.NameToInfo.get(name)199         if info isNone:200             raiseKeyError(201                 'There is no item named %r in the archive' %name)202
203         returninfo204
205     defsetpassword(self, pwd):206         """Set default password for encrypted files."""
207         self.pwd =pwd208
209 @property210     defcomment(self):211         """The comment text associated with the ZIP file."""
212         returnself._comment213
214 @comment.setter215     defcomment(self, comment):216         #check for valid comment length
217         if len(comment) >ZIP_MAX_COMMENT:218             importwarnings219             warnings.warn('Archive comment is too long; truncating to %d bytes'
220                           % ZIP_MAX_COMMENT, stacklevel=2)221             comment =comment[:ZIP_MAX_COMMENT]222         self._comment =comment223         self._didModify =True224
225     def read(self, name, pwd=None):226         """Return file bytes (as a string) for name."""
227         return self.open(name, "r", pwd).read()228
229     def open(self, name, mode="r", pwd=None):230         """Return file-like object for 'name'."""
231         if mode not in ("r", "U", "rU"):232             raise RuntimeError, 'open() requires mode "r", "U", or "rU"'
233         if notself.fp:234             raiseRuntimeError, \235                   "Attempt to read ZIP archive that was already closed"
236
237         #Only open a new file for instances where we were not
238         #given a file object in the constructor
239         ifself._filePassed:240             zef_file =self.fp241             should_close =False242         else:243             zef_file = open(self.filename, 'rb')244             should_close =True245
246         try:247             #Make sure we have an info object
248             ifisinstance(name, ZipInfo):249                 #'name' is already an info object
250                 zinfo =name251             else:252                 #Get info object for name
253                 zinfo =self.getinfo(name)254
255 zef_file.seek(zinfo.header_offset, 0)256
257             #Skip the file header:
258             fheader =zef_file.read(sizeFileHeader)259             if len(fheader) !=sizeFileHeader:260                 raise BadZipfile("Truncated file header")261             fheader =struct.unpack(structFileHeader, fheader)262             if fheader[_FH_SIGNATURE] !=stringFileHeader:263                 raise BadZipfile("Bad magic number for file header")264
265             fname =zef_file.read(fheader[_FH_FILENAME_LENGTH])266             iffheader[_FH_EXTRA_FIELD_LENGTH]:267 zef_file.read(fheader[_FH_EXTRA_FIELD_LENGTH])268
269             if fname !=zinfo.orig_filename:270                 raiseBadZipfile, \271                         'File name in directory "%s" and header "%s" differ.' %(272 zinfo.orig_filename, fname)273
274             #check for encrypted flag & handle password
275             is_encrypted = zinfo.flag_bits & 0x1
276             zd =None277             ifis_encrypted:278                 if notpwd:279                     pwd =self.pwd280                 if notpwd:281                     raise RuntimeError, "File %s is encrypted,"\282                         "password required for extraction" %name283
284                 zd =_ZipDecrypter(pwd)285                 #The first 12 bytes in the cypher stream is an encryption header
286                 #used to strengthen the algorithm. The first 11 bytes are
287                 #completely random, while the 12th contains the MSB of the CRC,
288                 #or the MSB of the file time depending on the header type
289                 #and is used to check the correctness of the password.
290                 bytes = zef_file.read(12)291                 h = map(zd, bytes[0:12])292                 if zinfo.flag_bits & 0x8:293                     #compare against the file type from extended local headers
294                     check_byte = (zinfo._raw_time >> 8) & 0xff
295                 else:296                     #compare against the CRC otherwise
297                     check_byte = (zinfo.CRC >> 24) & 0xff
298                 if ord(h[11]) !=check_byte:299                     raise RuntimeError("Bad password for file", name)300
301             returnZipExtFile(zef_file, mode, zinfo, zd,302                     close_fileobj=should_close)303         except:304             ifshould_close:305 zef_file.close()306             raise
307
308     def extract(self, member, path=None, pwd=None):309         """Extract a member from the archive to the current working directory,310 using its full name. Its file information is extracted as accurately311 as possible. `member' may be a filename or a ZipInfo object. You can312 specify a different directory using `path'.313         """
314         if notisinstance(member, ZipInfo):315             member =self.getinfo(member)316
317         if path isNone:318             path =os.getcwd()319
320         returnself._extract_member(member, path, pwd)321
322     def extractall(self, path=None, members=None, pwd=None):323         """Extract all members from the archive to the current working324 directory. `path' specifies a different directory to extract to.325 `members' is optional and must be a subset of the list returned326 by namelist().327         """
328         if members isNone:329             members =self.namelist()330
331         for zipinfo inmembers:332 self.extract(zipinfo, path, pwd)333
334     def_extract_member(self, member, targetpath, pwd):335         """Extract the ZipInfo object 'member' to a physical336 file on the path targetpath.337         """
338         #build the destination pathname, replacing
339         #forward slashes to platform specific separators.
340         arcname = member.filename.replace('/', os.path.sep)341
342         ifos.path.altsep:343             arcname =arcname.replace(os.path.altsep, os.path.sep)344         #interpret absolute pathname as relative, remove drive letter or
345         #UNC path, redundant separators, "." and ".." components.
346         arcname = os.path.splitdrive(arcname)[1]347         arcname = os.path.sep.join(x for x inarcname.split(os.path.sep)348                     if x not in ('', os.path.curdir, os.path.pardir))349         if os.path.sep == '\\':350             #filter illegal characters on Windows
351             illegal = ':<>|"?*'
352             ifisinstance(arcname, unicode):353                 table = {ord(c): ord('_') for c inillegal}354             else:355                 table = string.maketrans(illegal, '_' *len(illegal))356             arcname =arcname.translate(table)357             #remove trailing dots
358             arcname = (x.rstrip('.') for x inarcname.split(os.path.sep))359             arcname = os.path.sep.join(x for x in arcname ifx)360
361         targetpath =os.path.join(targetpath, arcname)362         targetpath =os.path.normpath(targetpath)363
364         #Create all upper directories if necessary.
365         upperdirs =os.path.dirname(targetpath)366         if upperdirs and notos.path.exists(upperdirs):367 os.makedirs(upperdirs)368
369         if member.filename[-1] == '/':370             if notos.path.isdir(targetpath):371 os.mkdir(targetpath)372             returntargetpath373
374         with self.open(member, pwd=pwd) as source, \375              file(targetpath, "wb") as target:376 shutil.copyfileobj(source, target)377
378         returntargetpath379
380     def_writecheck(self, zinfo):381         """Check for errors before writing a file to the archive."""
382         if zinfo.filename inself.NameToInfo:383             importwarnings384             warnings.warn('Duplicate name: %r' % zinfo.filename, stacklevel=3)385         if self.mode not in ("w", "a"):386             raise RuntimeError, 'write() requires mode "w" or "a"'
387         if notself.fp:388             raiseRuntimeError, \389                   "Attempt to write ZIP archive that was already closed"
390         if zinfo.compress_type == ZIP_DEFLATED and notzlib:391             raiseRuntimeError, \392                   "Compression requires the (missing) zlib module"
393         if zinfo.compress_type not in(ZIP_STORED, ZIP_DEFLATED):394             raiseRuntimeError, \395                   "That compression method is not supported"
396         if notself._allowZip64:397             requires_zip64 =None398             if len(self.filelist) >=ZIP_FILECOUNT_LIMIT:399                 requires_zip64 = "Files count"
400             elif zinfo.file_size >ZIP64_LIMIT:401                 requires_zip64 = "Filesize"
402             elif zinfo.header_offset >ZIP64_LIMIT:403                 requires_zip64 = "Zipfile size"
404             ifrequires_zip64:405                 raise LargeZipFile(requires_zip64 +
406                                    "would require ZIP64 extensions")407
408     def write(self, filename, arcname=None, compress_type=None):409         """Put the bytes from filename into the archive under the name410 arcname."""
411         if notself.fp:412             raiseRuntimeError(413                   "Attempt to write to ZIP archive that was already closed")414
415         st =os.stat(filename)416         isdir =stat.S_ISDIR(st.st_mode)417         mtime =time.localtime(st.st_mtime)418         date_time = mtime[0:6]419         #Create ZipInfo instance to store file information
420         if arcname isNone:421             arcname =filename422         arcname = os.path.normpath(os.path.splitdrive(arcname)[1])423         while arcname[0] in(os.sep, os.altsep):424             arcname = arcname[1:]425         ifisdir:426             arcname += '/'
427         zinfo =ZipInfo(arcname, date_time)428         zinfo.external_attr = (st[0] & 0xFFFF) << 16L      #Unix attributes
429         if compress_type isNone:430             zinfo.compress_type =self.compression431         else:432             zinfo.compress_type =compress_type433
434         zinfo.file_size =st.st_size435         zinfo.flag_bits = 0x00
436         zinfo.header_offset = self.fp.tell()    #Start of header bytes
437
438 self._writecheck(zinfo)439         self._didModify =True440
441         ifisdir:442             zinfo.file_size =0443             zinfo.compress_size =0444             zinfo.CRC =0445             zinfo.external_attr |= 0x10  #MS-DOS directory flag
446 self.filelist.append(zinfo)447             self.NameToInfo[zinfo.filename] =zinfo448 self.fp.write(zinfo.FileHeader(False))449             return
450
451         with open(filename, "rb") as fp:452             #Must overwrite CRC and sizes with correct data later
453             zinfo.CRC = CRC =0454             zinfo.compress_size = compress_size =0455             #Compressed size can be larger than uncompressed size
456             zip64 = self._allowZip64 and\457                     zinfo.file_size * 1.05 >ZIP64_LIMIT458 self.fp.write(zinfo.FileHeader(zip64))459             if zinfo.compress_type ==ZIP_DEFLATED:460                 cmpr =zlib.compressobj(zlib.Z_DEFAULT_COMPRESSION,461                      zlib.DEFLATED, -15)462             else:463                 cmpr =None464             file_size =0465             while 1:466                 buf = fp.read(1024 * 8)467                 if notbuf:468                     break
469                 file_size = file_size +len(buf)470                 CRC = crc32(buf, CRC) & 0xffffffff
471                 ifcmpr:472                     buf =cmpr.compress(buf)473                     compress_size = compress_size +len(buf)474 self.fp.write(buf)475         ifcmpr:476             buf =cmpr.flush()477             compress_size = compress_size +len(buf)478 self.fp.write(buf)479             zinfo.compress_size =compress_size480         else:481             zinfo.compress_size =file_size482         zinfo.CRC =CRC483         zinfo.file_size =file_size484         if not zip64 andself._allowZip64:485             if file_size >ZIP64_LIMIT:486                 raise RuntimeError('File size has increased during compressing')487             if compress_size >ZIP64_LIMIT:488                 raise RuntimeError('Compressed size larger than uncompressed size')489         #Seek backwards and write file header (which will now include
490         #correct CRC and file sizes)
491         position = self.fp.tell()       #Preserve current position in file
492 self.fp.seek(zinfo.header_offset, 0)493 self.fp.write(zinfo.FileHeader(zip64))494 self.fp.seek(position, 0)495 self.filelist.append(zinfo)496         self.NameToInfo[zinfo.filename] =zinfo497
498     def writestr(self, zinfo_or_arcname, bytes, compress_type=None):499         """Write a file into the archive.  The contents is the string500 'bytes'.  'zinfo_or_arcname' is either a ZipInfo instance or501 the name of the file in the archive."""
502         if notisinstance(zinfo_or_arcname, ZipInfo):503             zinfo = ZipInfo(filename=zinfo_or_arcname,504                             date_time=time.localtime(time.time())[:6])505
506             zinfo.compress_type =self.compression507             if zinfo.filename[-1] == '/':508                 zinfo.external_attr = 0o40775 << 16   #drwxrwxr-x
509                 zinfo.external_attr |= 0x10           #MS-DOS directory flag
510             else:511                 zinfo.external_attr = 0o600 << 16     #?rw-------
512         else:513             zinfo =zinfo_or_arcname514
515         if notself.fp:516             raiseRuntimeError(517                   "Attempt to write to ZIP archive that was already closed")518
519         if compress_type is notNone:520             zinfo.compress_type =compress_type521
522         zinfo.file_size = len(bytes)            #Uncompressed size
523         zinfo.header_offset = self.fp.tell()    #Start of header bytes
524 self._writecheck(zinfo)525         self._didModify =True526         zinfo.CRC = crc32(bytes) & 0xffffffff       #CRC-32 checksum
527         if zinfo.compress_type ==ZIP_DEFLATED:528             co =zlib.compressobj(zlib.Z_DEFAULT_COMPRESSION,529                  zlib.DEFLATED, -15)530             bytes = co.compress(bytes) +co.flush()531             zinfo.compress_size = len(bytes)    #Compressed size
532         else:533             zinfo.compress_size =zinfo.file_size534         zip64 = zinfo.file_size > ZIP64_LIMIT or\535                 zinfo.compress_size >ZIP64_LIMIT536         if zip64 and notself._allowZip64:537             raise LargeZipFile("Filesize would require ZIP64 extensions")538 self.fp.write(zinfo.FileHeader(zip64))539 self.fp.write(bytes)540         if zinfo.flag_bits & 0x08:541             #Write CRC and file sizes after the file data
542             fmt = '<LQQ' if zip64 else '<LLL'
543 self.fp.write(struct.pack(fmt, zinfo.CRC, zinfo.compress_size,544 zinfo.file_size))545 self.fp.flush()546 self.filelist.append(zinfo)547         self.NameToInfo[zinfo.filename] =zinfo548
549     def __del__(self):550         """Call the "close()" method in case the user forgot."""
551 self.close()552
553     defclose(self):554         """Close the file, and for mode "w" and "a" write the ending555 records."""
556         if self.fp isNone:557             return
558
559         try:560             if self.mode in ("w", "a") and self._didModify: #write ending records
561                 pos1 =self.fp.tell()562                 for zinfo in self.filelist:         #write central directory
563                     dt =zinfo.date_time564                     dosdate = (dt[0] - 1980) << 9 | dt[1] << 5 | dt[2]565                     dostime = dt[3] << 11 | dt[4] << 5 | (dt[5] // 2)566                     extra =[]567                     if zinfo.file_size >ZIP64_LIMIT \568                             or zinfo.compress_size >ZIP64_LIMIT:569 extra.append(zinfo.file_size)570 extra.append(zinfo.compress_size)571                         file_size = 0xffffffff
572                         compress_size = 0xffffffff
573                     else:574                         file_size =zinfo.file_size575                         compress_size =zinfo.compress_size576
577                     if zinfo.header_offset >ZIP64_LIMIT:578 extra.append(zinfo.header_offset)579                         header_offset = 0xffffffffL
580                     else:581                         header_offset =zinfo.header_offset582
583                     extra_data =zinfo.extra584                     ifextra:585                         #Append a ZIP64 field to the extra's
586                         extra_data =struct.pack(587                                 '<HH' + 'Q'*len(extra),588                                 1, 8*len(extra), *extra) +extra_data589
590                         extract_version = max(45, zinfo.extract_version)591                         create_version = max(45, zinfo.create_version)592                     else:593                         extract_version =zinfo.extract_version594                         create_version =zinfo.create_version595
596                     try:597                         filename, flag_bits =zinfo._encodeFilenameFlags()598                         centdir =struct.pack(structCentralDir,599 stringCentralDir, create_version,600 zinfo.create_system, extract_version, zinfo.reserved,601 flag_bits, zinfo.compress_type, dostime, dosdate,602 zinfo.CRC, compress_size, file_size,603 len(filename), len(extra_data), len(zinfo.comment),604 0, zinfo.internal_attr, zinfo.external_attr,605 header_offset)606                     exceptDeprecationWarning:607                         print >>sys.stderr, (structCentralDir,608 stringCentralDir, create_version,609 zinfo.create_system, extract_version, zinfo.reserved,610 zinfo.flag_bits, zinfo.compress_type, dostime, dosdate,611 zinfo.CRC, compress_size, file_size,612 len(zinfo.filename), len(extra_data), len(zinfo.comment),613 0, zinfo.internal_attr, zinfo.external_attr,614 header_offset)615                         raise
616 self.fp.write(centdir)617 self.fp.write(filename)618 self.fp.write(extra_data)619 self.fp.write(zinfo.comment)620
621                 pos2 =self.fp.tell()622                 #Write end-of-zip-archive record
623                 centDirCount =len(self.filelist)624                 centDirSize = pos2 -pos1625                 centDirOffset =pos1626                 requires_zip64 =None627                 if centDirCount >ZIP_FILECOUNT_LIMIT:628                     requires_zip64 = "Files count"
629                 elif centDirOffset >ZIP64_LIMIT:630                     requires_zip64 = "Central directory offset"
631                 elif centDirSize >ZIP64_LIMIT:632                     requires_zip64 = "Central directory size"
633                 ifrequires_zip64:634                     #Need to write the ZIP64 end-of-archive records
635                     if notself._allowZip64:636                         raise LargeZipFile(requires_zip64 +
637                                            "would require ZIP64 extensions")638                     zip64endrec =struct.pack(639 structEndArchive64, stringEndArchive64,640                             44, 45, 45, 0, 0, centDirCount, centDirCount,641 centDirSize, centDirOffset)642 self.fp.write(zip64endrec)643
644                     zip64locrec =struct.pack(645 structEndArchive64Locator,646                             stringEndArchive64Locator, 0, pos2, 1)647 self.fp.write(zip64locrec)648                     centDirCount = min(centDirCount, 0xFFFF)649                     centDirSize = min(centDirSize, 0xFFFFFFFF)650                     centDirOffset = min(centDirOffset, 0xFFFFFFFF)651
652                 endrec =struct.pack(structEndArchive, stringEndArchive,653 0, 0, centDirCount, centDirCount,654 centDirSize, centDirOffset, len(self._comment))655 self.fp.write(endrec)656 self.fp.write(self._comment)657 self.fp.flush()658         finally:659             fp =self.fp660             self.fp =None661             if notself._filePassed:662 fp.close()663
664 ZipFile665
666 ZipFile

ZipFile

1 classTarFile(object):2     """The TarFile Class provides an interface to tar archives.3     """
4
5     debug = 0                   #May be set from 0 (no msgs) to 3 (all msgs)
6
7     dereference = False         #If true, add content of linked file to the
8                                 #tar file, else the link.
9
10     ignore_zeros = False        #If true, skips empty or invalid blocks and
11                                 #continues processing.
12
13     errorlevel = 1              #If 0, fatal errors only appear in debug
14                                 #messages (if debug >= 0). If > 0, errors
15                                 #are passed to the caller as exceptions.
16
17     format = DEFAULT_FORMAT     #The format to use when creating an archive.
18
19     encoding = ENCODING         #Encoding for 8-bit character strings.
20
21     errors = None               #Error handler for unicode conversion.
22
23     tarinfo = TarInfo           #The default TarInfo class to use.
24
25     fileobject = ExFileObject   #The default ExFileObject class to use.
26
27     def __init__(self, name=None, mode="r", fileobj=None, format=None,28             tarinfo=None, dereference=None, ignore_zeros=None, encoding=None,29             errors=None, pax_headers=None, debug=None, errorlevel=None):30         """Open an (uncompressed) tar archive `name'. `mode' is either 'r' to31 read from an existing archive, 'a' to append data to an existing32 file or 'w' to create a new file overwriting an existing one. `mode'33 defaults to 'r'.34 If `fileobj' is given, it is used for reading or writing data. If it35 can be determined, `mode' is overridden by `fileobj's mode.36 `fileobj' is not closed, when TarFile is closed.37         """
38         modes = {"r": "rb", "a": "r+b", "w": "wb"}39         if mode not inmodes:40             raise ValueError("mode must be 'r', 'a' or 'w'")41         self.mode =mode42         self._mode =modes[mode]43
44         if notfileobj:45             if self.mode == "a" and notos.path.exists(name):46                 #Create nonexistent files in append mode.
47                 self.mode = "w"
48                 self._mode = "wb"
49             fileobj =bltn_open(name, self._mode)50             self._extfileobj =False51         else:52             if name is None and hasattr(fileobj, "name"):53                 name =fileobj.name54             if hasattr(fileobj, "mode"):55                 self._mode =fileobj.mode56             self._extfileobj =True57         self.name = os.path.abspath(name) if name elseNone58         self.fileobj =fileobj59
60         #Init attributes.
61         if format is notNone:62             self.format =format63         if tarinfo is notNone:64             self.tarinfo =tarinfo65         if dereference is notNone:66             self.dereference =dereference67         if ignore_zeros is notNone:68             self.ignore_zeros =ignore_zeros69         if encoding is notNone:70             self.encoding =encoding71
72         if errors is notNone:73             self.errors =errors74         elif mode == "r":75             self.errors = "utf-8"
76         else:77             self.errors = "strict"
78
79         if pax_headers is not None and self.format ==PAX_FORMAT:80             self.pax_headers =pax_headers81         else:82             self.pax_headers ={}83
84         if debug is notNone:85             self.debug =debug86         if errorlevel is notNone:87             self.errorlevel =errorlevel88
89         #Init datastructures.
90         self.closed =False91         self.members = []       #list of members as TarInfo objects
92         self._loaded = False    #flag if all members have been read
93         self.offset =self.fileobj.tell()94                                 #current position in the archive file
95         self.inodes = {}        #dictionary caching the inodes of
96                                 #archive members already added
97
98         try:99             if self.mode == "r":100                 self.firstmember =None101                 self.firstmember =self.next()102
103             if self.mode == "a":104                 #Move to the end of the archive,
105                 #before the first empty block.
106                 whileTrue:107 self.fileobj.seek(self.offset)108                     try:109                         tarinfo =self.tarinfo.fromtarfile(self)110 self.members.append(tarinfo)111                     exceptEOFHeaderError:112 self.fileobj.seek(self.offset)113                         break
114                     exceptHeaderError, e:115                         raiseReadError(str(e))116
117             if self.mode in "aw":118                 self._loaded =True119
120                 ifself.pax_headers:121                     buf =self.tarinfo.create_pax_global_header(self.pax_headers.copy())122 self.fileobj.write(buf)123                     self.offset +=len(buf)124         except:125             if notself._extfileobj:126 self.fileobj.close()127             self.closed =True128             raise
129
130     def_getposix(self):131         return self.format ==USTAR_FORMAT132     def_setposix(self, value):133         importwarnings134         warnings.warn("use the format attribute instead", DeprecationWarning,135                       2)136         ifvalue:137             self.format =USTAR_FORMAT138         else:139             self.format =GNU_FORMAT140     posix =property(_getposix, _setposix)141
142     #--------------------------------------------------------------------------
143     #Below are the classmethods which act as alternate constructors to the
144     #TarFile class. The open() method is the only one that is needed for
145     #public use; it is the "super"-constructor and is able to select an
146     #adequate "sub"-constructor for a particular compression using the mapping
147     #from OPEN_METH.
148     #149     #This concept allows one to subclass TarFile without losing the comfort of
150     #the super-constructor. A sub-constructor is registered and made available
151     #by adding it to the mapping in OPEN_METH.
152
153 @classmethod154     def open(cls, name=None, mode="r", fileobj=None, bufsize=RECORDSIZE, **kwargs):155         """Open a tar archive for reading, writing or appending. Return156 an appropriate TarFile class.157
158 mode:159 'r' or 'r:*' open for reading with transparent compression160 'r:'         open for reading exclusively uncompressed161 'r:gz'       open for reading with gzip compression162 'r:bz2'      open for reading with bzip2 compression163 'a' or 'a:'  open for appending, creating the file if necessary164 'w' or 'w:'  open for writing without compression165 'w:gz'       open for writing with gzip compression166 'w:bz2'      open for writing with bzip2 compression167
168 'r|*'        open a stream of tar blocks with transparent compression169 'r|'         open an uncompressed stream of tar blocks for reading170 'r|gz'       open a gzip compressed stream of tar blocks171 'r|bz2'      open a bzip2 compressed stream of tar blocks172 'w|'         open an uncompressed stream for writing173 'w|gz'       open a gzip compressed stream for writing174 'w|bz2'      open a bzip2 compressed stream for writing175         """
176
177         if not name and notfileobj:178             raise ValueError("nothing to open")179
180         if mode in ("r", "r:*"):181             #Find out which *open() is appropriate for opening the file.
182             for comptype incls.OPEN_METH:183                 func =getattr(cls, cls.OPEN_METH[comptype])184                 if fileobj is notNone:185                     saved_pos =fileobj.tell()186                 try:187                     return func(name, "r", fileobj, **kwargs)188                 except(ReadError, CompressionError), e:189                     if fileobj is notNone:190 fileobj.seek(saved_pos)191                     continue
192             raise ReadError("file could not be opened successfully")193
194         elif ":" inmode:195             filemode, comptype = mode.split(":", 1)196             filemode = filemode or "r"
197             comptype = comptype or "tar"
198
199             #Select the *open() function according to
200             #given compression.
201             if comptype incls.OPEN_METH:202                 func =getattr(cls, cls.OPEN_METH[comptype])203             else:204                 raise CompressionError("unknown compression type %r" %comptype)205             return func(name, filemode, fileobj, **kwargs)206
207         elif "|" inmode:208             filemode, comptype = mode.split("|", 1)209             filemode = filemode or "r"
210             comptype = comptype or "tar"
211
212             if filemode not in ("r", "w"):213                 raise ValueError("mode must be 'r' or 'w'")214
215             stream =_Stream(name, filemode, comptype, fileobj, bufsize)216             try:217                 t = cls(name, filemode, stream, **kwargs)218             except:219 stream.close()220                 raise
221             t._extfileobj =False222             returnt223
224         elif mode in ("a", "w"):225             return cls.taropen(name, mode, fileobj, **kwargs)226
227         raise ValueError("undiscernible mode")228
229 @classmethod230     def taropen(cls, name, mode="r", fileobj=None, **kwargs):231         """Open uncompressed tar archive name for reading or writing.232         """
233         if mode not in ("r", "a", "w"):234             raise ValueError("mode must be 'r', 'a' or 'w'")235         return cls(name, mode, fileobj, **kwargs)236
237 @classmethod238     def gzopen(cls, name, mode="r", fileobj=None, compresslevel=9, **kwargs):239         """Open gzip compressed tar archive name for reading or writing.240 Appending is not allowed.241         """
242         if mode not in ("r", "w"):243             raise ValueError("mode must be 'r' or 'w'")244
245         try:246             importgzip247 gzip.GzipFile248         except(ImportError, AttributeError):249             raise CompressionError("gzip module is not available")250
251         try:252             fileobj =gzip.GzipFile(name, mode, compresslevel, fileobj)253         exceptOSError:254             if fileobj is not None and mode == 'r':255                 raise ReadError("not a gzip file")256             raise
257
258         try:259             t = cls.taropen(name, mode, fileobj, **kwargs)260         exceptIOError:261 fileobj.close()262             if mode == 'r':263                 raise ReadError("not a gzip file")264             raise
265         except:266 fileobj.close()267             raise
268         t._extfileobj =False269         returnt270
271 @classmethod272     def bz2open(cls, name, mode="r", fileobj=None, compresslevel=9, **kwargs):273         """Open bzip2 compressed tar archive name for reading or writing.274 Appending is not allowed.275         """
276         if mode not in ("r", "w"):277             raise ValueError("mode must be 'r' or 'w'.")278
279         try:280             importbz2281         exceptImportError:282             raise CompressionError("bz2 module is not available")283
284         if fileobj is notNone:285             fileobj =_BZ2Proxy(fileobj, mode)286         else:287             fileobj = bz2.BZ2File(name, mode, compresslevel=compresslevel)288
289         try:290             t = cls.taropen(name, mode, fileobj, **kwargs)291         except(IOError, EOFError):292 fileobj.close()293             if mode == 'r':294                 raise ReadError("not a bzip2 file")295             raise
296         except:297 fileobj.close()298             raise
299         t._extfileobj =False300         returnt301
302     #All *open() methods are registered here.
303     OPEN_METH ={304         "tar": "taropen",   #uncompressed tar
305         "gz":  "gzopen",    #gzip compressed tar
306         "bz2": "bz2open"    #bzip2 compressed tar
307 }308
309     #--------------------------------------------------------------------------
310     #The public methods which TarFile provides:
311
312     defclose(self):313         """Close the TarFile. In write-mode, two finishing zero blocks are314 appended to the archive.315         """
316         ifself.closed:317             return
318
319         if self.mode in "aw":320             self.fileobj.write(NUL * (BLOCKSIZE * 2))321             self.offset += (BLOCKSIZE * 2)322             #fill up the end with zero-blocks
323             #(like option -b20 for tar does)
324             blocks, remainder =divmod(self.offset, RECORDSIZE)325             if remainder >0:326                 self.fileobj.write(NUL * (RECORDSIZE -remainder))327
328         if notself._extfileobj:329 self.fileobj.close()330         self.closed =True331
332     defgetmember(self, name):333         """Return a TarInfo object for member `name'. If `name' can not be334 found in the archive, KeyError is raised. If a member occurs more335 than once in the archive, its last occurrence is assumed to be the336 most up-to-date version.337         """
338         tarinfo =self._getmember(name)339         if tarinfo isNone:340             raise KeyError("filename %r not found" %name)341         returntarinfo342
343     defgetmembers(self):344         """Return the members of the archive as a list of TarInfo objects. The345 list has the same order as the members in the archive.346         """
347 self._check()348         if not self._loaded:    #if we want to obtain a list of
349             self._load()        #all members, we first have to
350                                 #scan the whole archive.
351         returnself.members352
353     defgetnames(self):354         """Return the members of the archive as a list of their names. It has355 the same order as the list returned by getmembers().356         """
357         return [tarinfo.name for tarinfo inself.getmembers()]358
359     def gettarinfo(self, name=None, arcname=None, fileobj=None):360         """Create a TarInfo object for either the file `name' or the file361 object `fileobj' (using os.fstat on its file descriptor). You can362 modify some of the TarInfo's attributes before you add it using363 addfile(). If given, `arcname' specifies an alternative name for the364 file in the archive.365         """
366         self._check("aw")367
368         #When fileobj is given, replace name by
369         #fileobj's real name.
370         if fileobj is notNone:371             name =fileobj.name372
373         #Building the name of the member in the archive.
374         #Backward slashes are converted to forward slashes,
375         #Absolute paths are turned to relative paths.
376         if arcname isNone:377             arcname =name378         drv, arcname =os.path.splitdrive(arcname)379         arcname = arcname.replace(os.sep, "/")380         arcname = arcname.lstrip("/")381
382         #Now, fill the TarInfo object with
383         #information specific for the file.
384         tarinfo =self.tarinfo()385         tarinfo.tarfile =self386
387         #Use os.stat or os.lstat, depending on platform
388         #and if symlinks shall be resolved.
389         if fileobj isNone:390             if hasattr(os, "lstat") and notself.dereference:391                 statres =os.lstat(name)392             else:393                 statres =os.stat(name)394         else:395             statres =os.fstat(fileobj.fileno())396         linkname = ""
397
398         stmd =statres.st_mode399         ifstat.S_ISREG(stmd):400             inode =(statres.st_ino, statres.st_dev)401             if not self.dereference and statres.st_nlink > 1 and\402                     inode in self.inodes and arcname !=self.inodes[inode]:403                 #Is it a hardlink to an already
404                 #archived file?
405                 type =LNKTYPE406                 linkname =self.inodes[inode]407             else:408                 #The inode is added only if its valid.
409                 #For win32 it is always 0.
410                 type =REGTYPE411                 ifinode[0]:412                     self.inodes[inode] =arcname413         elifstat.S_ISDIR(stmd):414             type =DIRTYPE415         elifstat.S_ISFIFO(stmd):416             type =FIFOTYPE417         elifstat.S_ISLNK(stmd):418             type =SYMTYPE419             linkname =os.readlink(name)420         elifstat.S_ISCHR(stmd):421             type =CHRTYPE422         elifstat.S_ISBLK(stmd):423             type =BLKTYPE424         else:425             returnNone426
427         #Fill the TarInfo object with all
428         #information we can get.
429         tarinfo.name =arcname430         tarinfo.mode =stmd431         tarinfo.uid =statres.st_uid432         tarinfo.gid =statres.st_gid433         if type ==REGTYPE:434             tarinfo.size =statres.st_size435         else:436             tarinfo.size =0L437         tarinfo.mtime =statres.st_mtime438         tarinfo.type =type439         tarinfo.linkname =linkname440         ifpwd:441             try:442                 tarinfo.uname =pwd.getpwuid(tarinfo.uid)[0]443             exceptKeyError:444                 pass
445         ifgrp:446             try:447                 tarinfo.gname =grp.getgrgid(tarinfo.gid)[0]448             exceptKeyError:449                 pass
450
451         if type in(CHRTYPE, BLKTYPE):452             if hasattr(os, "major") and hasattr(os, "minor"):453                 tarinfo.devmajor =os.major(statres.st_rdev)454                 tarinfo.devminor =os.minor(statres.st_rdev)455         returntarinfo456
457     def list(self, verbose=True):458         """Print a table of contents to sys.stdout. If `verbose' is False, only459 the names of the members are printed. If it is True, an `ls -l'-like460 output is produced.461         """
462 self._check()463
464         for tarinfo inself:465             ifverbose:466                 printfilemode(tarinfo.mode),467                 print "%s/%s" % (tarinfo.uname ortarinfo.uid,468                                  tarinfo.gname ortarinfo.gid),469                 if tarinfo.ischr() ortarinfo.isblk():470                     print "%10s" % ("%d,%d"\471                                     %(tarinfo.devmajor, tarinfo.devminor)),472                 else:473                     print "%10d" %tarinfo.size,474                 print "%d-%02d-%02d %02d:%02d:%02d"\475                       % time.localtime(tarinfo.mtime)[:6],476
477             print tarinfo.name + ("/" if tarinfo.isdir() else ""),478
479             ifverbose:480                 iftarinfo.issym():481                     print "->", tarinfo.linkname,482                 iftarinfo.islnk():483                     print "link to", tarinfo.linkname,484             print
485
486     def add(self, name, arcname=None, recursive=True, exclude=None, filter=None):487         """Add the file `name' to the archive. `name' may be any type of file488 (directory, fifo, symbolic link, etc.). If given, `arcname'489 specifies an alternative name for the file in the archive.490 Directories are added recursively by default. This can be avoided by491 setting `recursive' to False. `exclude' is a function that should492 return True for each filename to be excluded. `filter' is a function493 that expects a TarInfo object argument and returns the changed494 TarInfo object, if it returns None the TarInfo object will be495 excluded from the archive.496         """
497         self._check("aw")498
499         if arcname isNone:500             arcname =name501
502         #Exclude pathnames.
503         if exclude is notNone:504             importwarnings505             warnings.warn("use the filter argument instead",506                     DeprecationWarning, 2)507             ifexclude(name):508                 self._dbg(2, "tarfile: Excluded %r" %name)509                 return
510
511         #Skip if somebody tries to archive the archive...
512         if self.name is not None and os.path.abspath(name) ==self.name:513             self._dbg(2, "tarfile: Skipped %r" %name)514             return
515
516         self._dbg(1, name)517
518         #Create a TarInfo object from the file.
519         tarinfo =self.gettarinfo(name, arcname)520
521         if tarinfo isNone:522             self._dbg(1, "tarfile: Unsupported type %r" %name)523             return
524
525         #Change or exclude the TarInfo object.
526         if filter is notNone:527             tarinfo =filter(tarinfo)528             if tarinfo isNone:529                 self._dbg(2, "tarfile: Excluded %r" %name)530                 return
531
532         #Append the tar header and data to the archive.
533         iftarinfo.isreg():534             with bltn_open(name, "rb") as f:535 self.addfile(tarinfo, f)536
537         eliftarinfo.isdir():538 self.addfile(tarinfo)539             ifrecursive:540                 for f inos.listdir(name):541 self.add(os.path.join(name, f), os.path.join(arcname, f),542 recursive, exclude, filter)543
544         else:545 self.addfile(tarinfo)546
547     def addfile(self, tarinfo, fileobj=None):548         """Add the TarInfo object `tarinfo' to the archive. If `fileobj' is549 given, tarinfo.size bytes are read from it and added to the archive.550 You can create TarInfo objects using gettarinfo().551 On Windows platforms, `fileobj' should always be opened with mode552 'rb' to avoid irritation about the file size.553         """
554         self._check("aw")555
556         tarinfo =copy.copy(tarinfo)557
558         buf =tarinfo.tobuf(self.format, self.encoding, self.errors)559 self.fileobj.write(buf)560         self.offset +=len(buf)561
562         #If there's data to follow, append it.
563         if fileobj is notNone:564 copyfileobj(fileobj, self.fileobj, tarinfo.size)565             blocks, remainder =divmod(tarinfo.size, BLOCKSIZE)566             if remainder >0:567                 self.fileobj.write(NUL * (BLOCKSIZE -remainder))568                 blocks += 1
569             self.offset += blocks *BLOCKSIZE570
571 self.members.append(tarinfo)572
573     def extractall(self, path=".", members=None):574         """Extract all members from the archive to the current working575 directory and set owner, modification time and permissions on576 directories afterwards. `path' specifies a different directory577 to extract to. `members' is optional and must be a subset of the578 list returned by getmembers().579         """
580         directories =[]581
582         if members isNone:583             members =self584
585         for tarinfo inmembers:586             iftarinfo.isdir():587                 #Extract directories with a safe mode.
588 directories.append(tarinfo)589                 tarinfo =copy.copy(tarinfo)590                 tarinfo.mode = 0700
591 self.extract(tarinfo, path)592
593         #Reverse sort directories.
594         directories.sort(key=operator.attrgetter('name'))595 directories.reverse()596
597         #Set correct owner, mtime and filemode on directories.
598         for tarinfo indirectories:599             dirpath =os.path.join(path, tarinfo.name)600             try:601 self.chown(tarinfo, dirpath)602 self.utime(tarinfo, dirpath)603 self.chmod(tarinfo, dirpath)604             exceptExtractError, e:605                 if self.errorlevel > 1:606                     raise
607                 else:608                     self._dbg(1, "tarfile: %s" %e)609
610     def extract(self, member, path=""):611         """Extract a member from the archive to the current working directory,612 using its full name. Its file information is extracted as accurately613 as possible. `member' may be a filename or a TarInfo object. You can614 specify a different directory using `path'.615         """
616         self._check("r")617
618         ifisinstance(member, basestring):619             tarinfo =self.getmember(member)620         else:621             tarinfo =member622
623         #Prepare the link target for makelink().
624         iftarinfo.islnk():625             tarinfo._link_target =os.path.join(path, tarinfo.linkname)626
627         try:628 self._extract_member(tarinfo, os.path.join(path, tarinfo.name))629         exceptEnvironmentError, e:630             if self.errorlevel >0:631                 raise
632             else:633                 if e.filename isNone:634                     self._dbg(1, "tarfile: %s" %e.strerror)635                 else:636                     self._dbg(1, "tarfile: %s %r" %(e.strerror, e.filename))637         exceptExtractError, e:638             if self.errorlevel > 1:639                 raise
640             else:641                 self._dbg(1, "tarfile: %s" %e)642
643     defextractfile(self, member):644         """Extract a member from the archive as a file object. `member' may be645 a filename or a TarInfo object. If `member' is a regular file, a646 file-like object is returned. If `member' is a link, a file-like647 object is constructed from the link's target. If `member' is none of648 the above, None is returned.649 The file-like object is read-only and provides the following650 methods: read(), readline(), readlines(), seek() and tell()651         """
652         self._check("r")653
654         ifisinstance(member, basestring):655             tarinfo =self.getmember(member)656         else:657             tarinfo =member658
659         iftarinfo.isreg():660             returnself.fileobject(self, tarinfo)661
662         elif tarinfo.type not inSUPPORTED_TYPES:663             #If a member's type is unknown, it is treated as a
664             #regular file.
665             returnself.fileobject(self, tarinfo)666
667         elif tarinfo.islnk() ortarinfo.issym():668             ifisinstance(self.fileobj, _Stream):669                 #A small but ugly workaround for the case that someone tries
670                 #to extract a (sym)link as a file-object from a non-seekable
671                 #stream of tar blocks.
672                 raise StreamError("cannot extract (sym)link as file object")673             else:674                 #A (sym)link's file object is its target's file object.
675                 returnself.extractfile(self._find_link_target(tarinfo))676         else:677             #If there's no data associated with the member (directory, chrdev,
678             #blkdev, etc.), return None instead of a file object.
679             returnNone680
681     def_extract_member(self, tarinfo, targetpath):682         """Extract the TarInfo object tarinfo to a physical683 file called targetpath.684         """
685         #Fetch the TarInfo object for the given name
686         #and build the destination pathname, replacing
687         #forward slashes to platform specific separators.
688         targetpath = targetpath.rstrip("/")689         targetpath = targetpath.replace("/", os.sep)690
691         #Create all upper directories.
692         upperdirs =os.path.dirname(targetpath)693         if upperdirs and notos.path.exists(upperdirs):694             #Create directories that are not part of the archive with
695             #default permissions.
696 os.makedirs(upperdirs)697
698         if tarinfo.islnk() ortarinfo.issym():699             self._dbg(1, "%s -> %s" %(tarinfo.name, tarinfo.linkname))700         else:701             self._dbg(1, tarinfo.name)702
703         iftarinfo.isreg():704 self.makefile(tarinfo, targetpath)705         eliftarinfo.isdir():706 self.makedir(tarinfo, targetpath)707         eliftarinfo.isfifo():708 self.makefifo(tarinfo, targetpath)709         elif tarinfo.ischr() ortarinfo.isblk():710 self.makedev(tarinfo, targetpath)711         elif tarinfo.islnk() ortarinfo.issym():712 self.makelink(tarinfo, targetpath)713         elif tarinfo.type not inSUPPORTED_TYPES:714 self.makeunknown(tarinfo, targetpath)715         else:716 self.makefile(tarinfo, targetpath)717
718 self.chown(tarinfo, targetpath)719         if nottarinfo.issym():720 self.chmod(tarinfo, targetpath)721 self.utime(tarinfo, targetpath)722
723     #--------------------------------------------------------------------------
724     #Below are the different file methods. They are called via
725     #_extract_member() when extract() is called. They can be replaced in a
726     #subclass to implement other functionality.
727
728     defmakedir(self, tarinfo, targetpath):729         """Make a directory called targetpath.730         """
731         try:732             #Use a safe mode for the directory, the real mode is set
733             #later in _extract_member().
734             os.mkdir(targetpath, 0700)735         exceptEnvironmentError, e:736             if e.errno !=errno.EEXIST:737                 raise
738
739     defmakefile(self, tarinfo, targetpath):740         """Make a file called targetpath.741         """
742         source =self.extractfile(tarinfo)743         try:744             with bltn_open(targetpath, "wb") as target:745 copyfileobj(source, target)746         finally:747 source.close()748
749     defmakeunknown(self, tarinfo, targetpath):750         """Make a file from a TarInfo object with an unknown type751 at targetpath.752         """
753 self.makefile(tarinfo, targetpath)754         self._dbg(1, "tarfile: Unknown file type %r,"\755                      "extracted as regular file." %tarinfo.type)756
757     defmakefifo(self, tarinfo, targetpath):758         """Make a fifo called targetpath.759         """
760         if hasattr(os, "mkfifo"):761 os.mkfifo(targetpath)762         else:763             raise ExtractError("fifo not supported by system")764
765     defmakedev(self, tarinfo, targetpath):766         """Make a character or block device called targetpath.767         """
768         if not hasattr(os, "mknod") or not hasattr(os, "makedev"):769             raise ExtractError("special devices not supported by system")770
771         mode =tarinfo.mode772         iftarinfo.isblk():773             mode |=stat.S_IFBLK774         else:775             mode |=stat.S_IFCHR776
777 os.mknod(targetpath, mode,778 os.makedev(tarinfo.devmajor, tarinfo.devminor))779
780     defmakelink(self, tarinfo, targetpath):781         """Make a (symbolic) link called targetpath. If it cannot be created782 (platform limitation), we try to make a copy of the referenced file783 instead of a link.784         """
785         if hasattr(os, "symlink") and hasattr(os, "link"):786             #For systems that support symbolic and hard links.
787             iftarinfo.issym():788                 ifos.path.lexists(targetpath):789 os.unlink(targetpath)790 os.symlink(tarinfo.linkname, targetpath)791             else:792                 #See extract().
793                 ifos.path.exists(tarinfo._link_target):794                     ifos.path.lexists(targetpath):795 os.unlink(targetpath)796 os.link(tarinfo._link_target, targetpath)797                 else:798 self._extract_member(self._find_link_target(tarinfo), targetpath)799         else:800             try:801 self._extract_member(self._find_link_target(tarinfo), targetpath)802             exceptKeyError:803                 raise ExtractError("unable to resolve link inside archive")804
805     defchown(self, tarinfo, targetpath):806         """Set owner of targetpath according to tarinfo.807         """
808         if pwd and hasattr(os, "geteuid") and os.geteuid() ==0:809             #We have to be root to do so.
810             try:811                 g = grp.getgrnam(tarinfo.gname)[2]812             exceptKeyError:813                 g =tarinfo.gid814             try:815                 u = pwd.getpwnam(tarinfo.uname)[2]816             exceptKeyError:817                 u =tarinfo.uid818             try:819                 if tarinfo.issym() and hasattr(os, "lchown"):820 os.lchown(targetpath, u, g)821                 else:822                     if sys.platform != "os2emx":823 os.chown(targetpath, u, g)824             exceptEnvironmentError, e:825                 raise ExtractError("could not change owner")826
827     defchmod(self, tarinfo, targetpath):828         """Set file permissions of targetpath according to tarinfo.829         """
830         if hasattr(os, 'chmod'):831             try:832 os.chmod(targetpath, tarinfo.mode)833             exceptEnvironmentError, e:834                 raise ExtractError("could not change mode")835
836     defutime(self, tarinfo, targetpath):837         """Set modification time of targetpath according to tarinfo.838         """
839         if not hasattr(os, 'utime'):840             return
841         try:842 os.utime(targetpath, (tarinfo.mtime, tarinfo.mtime))843         exceptEnvironmentError, e:844             raise ExtractError("could not change modification time")845
846     #--------------------------------------------------------------------------
847     defnext(self):848         """Return the next member of the archive as a TarInfo object, when849 TarFile is opened for reading. Return None if there is no more850 available.851         """
852         self._check("ra")853         if self.firstmember is notNone:854             m =self.firstmember855             self.firstmember =None856             returnm857
858         #Read the next block.
859 self.fileobj.seek(self.offset)860         tarinfo =None861         whileTrue:862             try:863                 tarinfo =self.tarinfo.fromtarfile(self)864             exceptEOFHeaderError, e:865                 ifself.ignore_zeros:866                     self._dbg(2, "0x%X: %s" %(self.offset, e))867                     self.offset +=BLOCKSIZE868                     continue
869             exceptInvalidHeaderError, e:870                 ifself.ignore_zeros:871                     self._dbg(2, "0x%X: %s" %(self.offset, e))872                     self.offset +=BLOCKSIZE873                     continue
874                 elif self.offset ==0:875                     raiseReadError(str(e))876             exceptEmptyHeaderError:877                 if self.offset ==0:878                     raise ReadError("empty file")879             exceptTruncatedHeaderError, e:880                 if self.offset ==0:881                     raiseReadError(str(e))882             exceptSubsequentHeaderError, e:883                 raiseReadError(str(e))884             break
885
886         if tarinfo is notNone:887 self.members.append(tarinfo)888         else:889             self._loaded =True890
891         returntarinfo892
893     #--------------------------------------------------------------------------
894     #Little helper methods:
895
896     def _getmember(self, name, tarinfo=None, normalize=False):897         """Find an archive member by name from bottom to top.898 If tarinfo is given, it is used as the starting point.899         """
900         #Ensure that all members have been loaded.
901         members =self.getmembers()902
903         #Limit the member search list up to tarinfo.
904         if tarinfo is notNone:905             members =members[:members.index(tarinfo)]906
907         ifnormalize:908             name =os.path.normpath(name)909
910         for member inreversed(members):911             ifnormalize:912                 member_name =os.path.normpath(member.name)913             else:914                 member_name =member.name915
916             if name ==member_name:917                 returnmember918
919     def_load(self):920         """Read through the entire archive file and look for readable921 members.922         """
923         whileTrue:924             tarinfo =self.next()925             if tarinfo isNone:926                 break
927         self._loaded =True928
929     def _check(self, mode=None):930         """Check if TarFile is still open, and if the operation's mode931 corresponds to TarFile's mode.932         """
933         ifself.closed:934             raise IOError("%s is closed" % self.__class__.__name__)935         if mode is not None and self.mode not inmode:936             raise IOError("bad operation for mode %r" %self.mode)937
938     def_find_link_target(self, tarinfo):939         """Find the target member of a symlink or hardlink member in the940 archive.941         """
942         iftarinfo.issym():943             #Always search the entire archive.
944             linkname = "/".join(filter(None, (os.path.dirname(tarinfo.name), tarinfo.linkname)))945             limit =None946         else:947             #Search the archive before the link, because a hard link is
948             #just a reference to an already archived file.
949             linkname =tarinfo.linkname950             limit =tarinfo951
952         member = self._getmember(linkname, tarinfo=limit, normalize=True)953         if member isNone:954             raise KeyError("linkname %r not found" %linkname)955         returnmember956
957     def __iter__(self):958         """Provide an iterator object.959         """
960         ifself._loaded:961             returniter(self.members)962         else:963             returnTarIter(self)964
965     def_dbg(self, level, msg):966         """Write debugging output to sys.stderr.967         """
968         if level <=self.debug:969             print >>sys.stderr, msg970
971     def __enter__(self):972 self._check()973         returnself974
975     def __exit__(self, type, value, traceback):976         if type isNone:977 self.close()978         else:979             #An exception occurred. We must not call close() because
980             #it would try to write end-of-archive blocks and padding.
981             if notself._extfileobj:982 self.fileobj.close()983             self.closed =True984 #class TarFile
985
986 TarFile987
988 TarFile

TarFile

转载于:https://www.cnblogs.com/GraceZ/p/8025016.html

shutil——高级的 文件、文件夹、压缩包 处理模块相关推荐

  1. python中shutil模块_python文件、文件夹、压缩包处理模块-shutil模块-阿里云开发者社区...

    shutil模块 高级的文件.文件夹.压缩包 处理模块 本节内容基本在linux下python交互环境实现 复制移动文件.文件夹 将文件内容拷贝到另一个文件中,可以部分内容 格式如下: ``` shu ...

  2. python创建模块文件夹_python文件、文件夹、压缩包处理模块-shutil模块

    shutil模块 高级的文件.文件夹.压缩包 处理模块 本节内容基本在linux下python交互环境实现 复制移动文件.文件夹 将文件内容拷贝到另一个文件中,可以部分内容 格式如下: ``` shu ...

  3. python shutil_Python3 shutil(高级文件操作模块)

    1.shutil是shell utility的缩写 shutil.move直接从一个地方挪到另一个地方,而os.rename常常只能重命名,不能挪动位置. 功能是:>>>shutil ...

  4. python文件文件夹操作

    Python文件夹操作 1.提要 ​ python中可以使用os及shutil模块进行文件和目录的操作.其中os可以执行简单的文件及文件夹操作.而shutil则提供了大量文件的高级操作,特别针对文件的 ...

  5. python读取、写入、移动、复制文件(夹)以及其他关于文件(夹)的操作

    文章目录 基础操作 判断文件或者目录是否存在 创建目录 连接两个路径成为一个路径 读取文件 写入文件 移动文件(夹) 复制文件(夹) 突然发现,经常需要进行文件操作,因为如果程序运行时间很长,我们需要 ...

  6. 通俗易懂【Springboot】 单文件下载和批量下载(多个文件合成一个压缩包下载)

    文章目录 一.单文件下载 1.简单理解文件下载 2.单文件下载的具体代码实现 3.测试 4.单文件下载整体代码 二.多文件批量下载(多个文件合成一个压缩包下载) 1.多文件下载的实现方式,这里使用了Z ...

  7. 文件上传控件-如何上传文件-文件夹上传

    需求: 项目要支持大文件上传功能,经过讨论,初步将文件上传大小控制在20G内,因此自己需要在项目中进行文件上传部分的调整和配置,自己将大小都以20G来进行限制. PC端全平台支持,要求支持Window ...

  8. 个人永久性免费-Excel催化剂功能第41波-文件文件夹相关函数

    对于日常办公过程中,每天面对的操作离不开文件.文件夹的操作,当然可以用资源管理器.Everything之类的管理软件来管理. 但涉及到批量操作时,在Excel环境或许是个更好的方式,前面很多的内容中不 ...

  9. XRename(文件文件夹超级重命名工具)简介

    测试版下载:https://gitee.com/sysdzw/XRename 开放源代码:https://blog.csdn.net/sysdzw/article/details/6213821 gi ...

最新文章

  1. struts 页面取值问题
  2. is not None与not某某 的区别
  3. 学Java可以看哪些书?6本Java书籍推荐
  4. 很长很真实!但会对你有所帮助的(关于职业规划)
  5. python程序文件扩展名有_python程序文件的扩展名称是什么
  6. python数组合并、输入三行数据,如何在python中合并两个或三个3D数组?
  7. 关于控件postback 后viewstate加载失败的问题
  8. Java——List集合特有的功能
  9. 自动化测试--8种定位方式以及等待详解
  10. 中学生python程序设计大赛_贵州民族大学第一届程序设计大赛圆满举办
  11. PHP代码中解决出现中文乱码的问题
  12. SpannableString 给TextView添加不同的显示样式
  13. 2022-01-11每日刷题打卡
  14. linux 怎么关闭输入法快捷键设置方法,关闭输入法快捷键
  15. kangle安装php7.0_搭建Kangle+EasyPanel对接SWAP IDC虚拟主机自助开通完整教程
  16. 用SDK包开发K66FX18学习笔记(3)
  17. 书法作品落款时间的农历传统雅称摘要
  18. PHP编程 967 个函式 值得收藏
  19. 【教学类-30-04】10以内减法题不重复(一页两份)(包括6以内、7以内、8以内、9以内、10以内减法题 只抽取25个)
  20. 【STM32F407】第4章 RTX5操作系统移植(MDK AC5)

热门文章

  1. 【OpenCV】OpenCV实战从入门到精通之 -- 输入输出XML和YAML文件详解
  2. Github | 人脸识别手册 (分类论文集)
  3. 计算机基本知识实训报告,计算机实训报告小结
  4. 写给小白的机器学习之决策树算法详解(附实战源码)
  5. 最先进单插槽专业绘图解决方案
  6. Git操作记录方便查阅
  7. echarts 三环圆环_echarts实现环形进度图
  8. python自动发送邮件_Python自动发送邮件
  9. matlab 下采样_Lattice规划与Matlab实现(1)
  10. 2018前端学习总结