zoukankan      html  css  js  c++  java
  • Python(27):shutil文件操作、subprocess运行子程序模块、psutil

    一、shutil模块(了解):高级的文件、文件夹、压缩包处理模块。

    import shutil
    
    # shutil.copyfileobj(fsrc, fdst[, length]),将文件内容拷贝到另一个文件中
    shutil.copyfileobj(open('old.xml', 'r'), open('new.xml', 'w'))
    
    # shutil.copyfile(src, dst),拷贝文件
    shutil.copyfile('f1.log', 'f2.log')  # 目标文件无需存在
    
    # shutil.copymode(src, dst),仅拷贝权限。内容、组、用户均不变
    shutil.copymode('f1.log', 'f2.log')  # 目标文件必须存在
    
    # shutil.copystat(src, dst),仅拷贝状态的信息,包括:mode bits, atime, mtime, flags
    shutil.copystat('f1.log', 'f2.log')  # 目标文件必须存在
    
    # shutil.copy(src, dst),拷贝文件和权限
    shutil.copy('f1.log', 'f2.log')
    
    # shutil.copy2(src, dst),拷贝文件和状态信息
    shutil.copy2('f1.log', 'f2.log')
    
    # shutil.ignore_patterns(*patterns)
    # shutil.copytree(src, dst, symlinks=False, ignore=None),递归的去拷贝文件夹
    # 目标目录不能存在,注意对folder2目录父级目录要有可写权限,ignore的意思是排除
    shutil.copytree('folder1', 'folder2', ignore=shutil.ignore_patterns('*.pyc', 'tmp*'))
    
    # shutil.rmtree(path[, ignore_errors[, onerror]]),递归的去删除文件
    shutil.rmtree('folder1')
    
    # shutil.move(src, dst),递归的去移动文件,它类似mv命令,其实就是重命名
    shutil.move('folder1', 'folder3')
    
    # shutil.make_archive(base_name, format, ...),创建压缩包并返回文件路径,例如:zip、tar
    '''
    base_name: 压缩包的文件名,也可以是压缩包的路径。只是文件名时,则保存至当前目录,否则保存至指定路径,如 data_bak = >保存至当前路径;/ tmp/data_bak = >保存至/tmp/
    format:压缩包种类,“zip”, “tar”, “bztar”,“gztar”
    root_dir:要压缩的文件夹路径(默认当前目录)
    owner:用户,默认当前用户
    group:组,默认当前组
    logger:用于记录日志,通常是logging.Logger对象
    '''
    
    # 将 /data 下的文件打包放置当前程序目录
    ret = shutil.make_archive("data_bak", 'gztar', root_dir='/data')
    
    # 将 /data下的文件打包放置 /tmp/目录
    ret = shutil.make_archive("/tmp/data_bak", 'gztar', root_dir='/data')

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

    1、 zipfile压缩解压缩

    import zipfile
    
    # 压缩
    z = zipfile.ZipFile('laxi.zip', 'w')
    z.write('a.log')
    z.write('data.data')
    z.close()
    
    # 解压
    z = zipfile.ZipFile('laxi.zip', 'r')
    z.extractall(path='.')
    z.close()

    2、 tarfile压缩解压缩

    import tarfile
    
    # 压缩
    t=tarfile.open('/tmp/egon.tar','w')
    t.add('/test1/a.py',arcname='a.bak')
    t.add('/test1/b.py',arcname='b.bak')
    t.close()
    
    
    # 解压
    t=tarfile.open('/tmp/egon.tar','r')
    t.extractall('/egon')
    t.close()

    二、subprocess模块(了解):运行子程序

    subprocess模块允许你去创建一个新的进程让其执行另外的程序,并与它进行通信,获取标准的输入、标准输出、标准错误以及返回码等。

    更多查看官网:https://docs.python.org/2/library/subprocess.html?highlight=subprocess#frequently-used-arguments

    import subprocess
    '''
    sh-3.2# ls /Users/nick/Desktop |grep txt$
    mysql.txt
    tt.txt
    事物.txt
    '''
    
    res1 = subprocess.Popen('ls /Users/jieli/Desktop',shell=True, stdout=subprocess.PIPE)
    res = subprocess.Popen('grep txt$', shell=True,stdin=res1.stdout, stdout=subprocess.PIPE)
    print(res.stdout.read().decode('utf-8'))
    
    # 下面这段等同于上面,但是上面的优势在于,一个数据流可以和另外一个数据流交互,可以通过爬虫得到结果然后交给grep。
    res1 = subprocess.Popen('ls /Users/jieli/Desktop |grep txt$',shell=True, stdout=subprocess.PIPE)
    print(res1.stdout.read().decode('utf-8'))
    
    # windows下:
    # dir | findstr 'test*'
    # dir | findstr 'txt$'
    res1 = subprocess.Popen(r'dir “C:UsersAdministratorPycharmProjects	est函数备课”',  shell=True, stdout=subprocess.PIPE)
    res = subprocess.Popen('findstr test*', shell=True,  stdin=res1.stdout,   stdout=subprocess.PIPE)
    
    # subprocess使用当前系统默认编码,得到结果为bytes类型,在windows下需要用gbk解码
    print(res.stdout.read().decode('gbk') )

    三、psutil

    用Python来编写脚本简化日常的运维工作是Python的一个重要用途。在Linux下,有许多系统命令可以让我们时刻监控系统运行的状态,如pstopfree等等。要获取这些系统信息,Python可以通过subprocess模块调用并获取结果。但这样做显得很麻烦,尤其是要写很多解析代码。

    在Python中获取系统信息的另一个好办法是使用psutil这个第三方模块。顾名思义,psutil = process and system utilities,它不仅可以通过一两行代码实现系统监控,还可以跨平台使用,支持Linux/UNIX/OSX/Windows等,是系统管理员和运维小伙伴不可或缺的必备模块。

    1、安装psutil

    如果安装了Anaconda,psutil就已经可用了。否则,需要在命令行下通过pip安装:

    $ pip install psutil
    

    如果遇到Permission denied安装失败,请加上sudo重试。

    2、获取CPU信息

    我们先来获取CPU的信息:

    >>> import psutil
    >>> psutil.cpu_count() # CPU逻辑数量
    4
    >>> psutil.cpu_count(logical=False) # CPU物理核心
    2
    # 2说明是双核超线程, 4则是4核非超线程
    

    image

    统计CPU的用户/系统/空闲时间:

    >>> psutil.cpu_times()
    scputimes(user=10963.31, nice=0.0, system=5138.67, idle=356102.45)
    

    再实现类似top命令的CPU使用率,每秒刷新一次,累计10次:

    >>> for x in range(10):
    ...     print(psutil.cpu_percent(interval=1, percpu=True))
    ... 
    [14.0, 4.0, 4.0, 4.0]
    [12.0, 3.0, 4.0, 3.0]
    [8.0, 4.0, 3.0, 4.0]
    [12.0, 3.0, 3.0, 3.0]
    [18.8, 5.1, 5.9, 5.0]
    [10.9, 5.0, 4.0, 3.0]
    [12.0, 5.0, 4.0, 5.0]
    [15.0, 5.0, 4.0, 4.0]
    [19.0, 5.0, 5.0, 4.0]
    [9.0, 3.0, 2.0, 3.0]
    

    3、获取内存信息

    使用psutil获取物理内存和交换内存信息,分别使用:

    >>> psutil.virtual_memory()
    svmem(total=8589934592, available=2866520064, percent=66.6, used=7201386496, free=216178688, active=3342192640, inactive=2650341376, wired=1208852480)
    >>> psutil.swap_memory()
    sswap(total=1073741824, used=150732800, free=923009024, percent=14.0, sin=10705981440, sout=40353792)
    

    返回的是字节为单位的整数,可以看到,总内存大小是8589934592 = 8 GB,已用7201386496 = 6.7 GB,使用了66.6%。

    而交换区大小是1073741824 = 1 GB。

    获取磁盘信息

    可以通过psutil获取磁盘分区、磁盘使用率和磁盘IO信息:

    >>> psutil.disk_partitions() # 磁盘分区信息
    [sdiskpart(device='/dev/disk1', mountpoint='/', fstype='hfs', opts='rw,local,rootfs,dovolfs,journaled,multilabel')]
    >>> psutil.disk_usage('/') # 磁盘使用情况
    sdiskusage(total=998982549504, used=390880133120, free=607840272384, percent=39.1)
    >>> psutil.disk_io_counters() # 磁盘IO
    sdiskio(read_count=988513, write_count=274457, read_bytes=14856830464, write_bytes=17509420032, read_time=2228966, write_time=1618405)
    

    可以看到,磁盘'/'的总容量是998982549504 = 930 GB,使用了39.1%。文件格式是HFS,opts中包含rw表示可读写,journaled表示支持日志。

    4、获取网络信息

    psutil可以获取网络接口和网络连接信息:

    >>> psutil.net_io_counters() # 获取网络读写字节/包的个数
    snetio(bytes_sent=3885744870, bytes_recv=10357676702, packets_sent=10613069, packets_recv=10423357, errin=0, errout=0, dropin=0, dropout=0)
    >>> psutil.net_if_addrs() # 获取网络接口信息
    {
      'lo0': [snic(family=<AddressFamily.AF_INET: 2>, address='127.0.0.1', netmask='255.0.0.0'), ...],
      'en1': [snic(family=<AddressFamily.AF_INET: 2>, address='10.0.1.80', netmask='255.255.255.0'), ...],
      'en0': [...],
      'en2': [...],
      'bridge0': [...]
    }
    >>> psutil.net_if_stats() # 获取网络接口状态
    {
      'lo0': snicstats(isup=True, duplex=<NicDuplex.NIC_DUPLEX_UNKNOWN: 0>, speed=0, mtu=16384),
      'en0': snicstats(isup=True, duplex=<NicDuplex.NIC_DUPLEX_UNKNOWN: 0>, speed=0, mtu=1500),
      'en1': snicstats(...),
      'en2': snicstats(...),
      'bridge0': snicstats(...)
    }
    

    要获取当前网络连接信息,使用net_connections()

    >>> psutil.net_connections()
    Traceback (most recent call last):
      ...
    PermissionError: [Errno 1] Operation not permitted
    
    During handling of the above exception, another exception occurred:
    
    Traceback (most recent call last):
      ...
    psutil.AccessDenied: psutil.AccessDenied (pid=3847)
    

    你可能会得到一个AccessDenied错误,原因是psutil获取信息也是要走系统接口,而获取网络连接信息需要root权限,这种情况下,可以退出Python交互环境,用sudo重新启动:

    $ sudo python3
    Password: ******
    Python 3.8 ... on darwin
    Type "help", ... for more information.
    >>> import psutil
    >>> psutil.net_connections()
    [
        sconn(fd=83, family=<AddressFamily.AF_INET6: 30>, type=1, laddr=addr(ip='::127.0.0.1', port=62911), raddr=addr(ip='::127.0.0.1', port=3306), status='ESTABLISHED', pid=3725),
        sconn(fd=84, family=<AddressFamily.AF_INET6: 30>, type=1, laddr=addr(ip='::127.0.0.1', port=62905), raddr=addr(ip='::127.0.0.1', port=3306), status='ESTABLISHED', pid=3725),
        sconn(fd=93, family=<AddressFamily.AF_INET6: 30>, type=1, laddr=addr(ip='::', port=8080), raddr=(), status='LISTEN', pid=3725),
        sconn(fd=103, family=<AddressFamily.AF_INET6: 30>, type=1, laddr=addr(ip='::127.0.0.1', port=62918), raddr=addr(ip='::127.0.0.1', port=3306), status='ESTABLISHED', pid=3725),
        sconn(fd=105, family=<AddressFamily.AF_INET6: 30>, type=1, ..., pid=3725),
        sconn(fd=106, family=<AddressFamily.AF_INET6: 30>, type=1, ..., pid=3725),
        sconn(fd=107, family=<AddressFamily.AF_INET6: 30>, type=1, ..., pid=3725),
        ...
        sconn(fd=27, family=<AddressFamily.AF_INET: 2>, type=2, ..., pid=1)
    ]
    

    5、获取进程信息

    通过psutil可以获取到所有进程的详细信息:

    >>> psutil.pids() # 所有进程ID
    [3865, 3864, 3863, 3856, 3855, 3853, 3776, ..., 45, 44, 1, 0]
    >>> p = psutil.Process(3776) # 获取指定进程ID=3776,其实就是当前Python交互环境
    >>> p.name() # 进程名称
    'python3.6'
    >>> p.exe() # 进程exe路径
    '/Users/michael/anaconda3/bin/python3.6'
    >>> p.cwd() # 进程工作目录
    '/Users/michael'
    >>> p.cmdline() # 进程启动的命令行
    ['python3']
    >>> p.ppid() # 父进程ID
    3765
    >>> p.parent() # 父进程
    <psutil.Process(pid=3765, name='bash') at 4503144040>
    >>> p.children() # 子进程列表
    []
    >>> p.status() # 进程状态
    'running'
    >>> p.username() # 进程用户名
    'michael'
    >>> p.create_time() # 进程创建时间
    1511052731.120333
    >>> p.terminal() # 进程终端
    '/dev/ttys002'
    >>> p.cpu_times() # 进程使用的CPU时间
    pcputimes(user=0.081150144, system=0.053269812, children_user=0.0, children_system=0.0)
    >>> p.memory_info() # 进程使用的内存
    pmem(rss=8310784, vms=2481725440, pfaults=3207, pageins=18)
    >>> p.open_files() # 进程打开的文件
    []
    >>> p.connections() # 进程相关网络连接
    []
    >>> p.num_threads() # 进程的线程数量
    1
    >>> p.threads() # 所有线程信息
    [pthread(id=1, user_time=0.090318, system_time=0.062736)]
    >>> p.environ() # 进程环境变量
    {'SHELL': '/bin/bash', 'PATH': '/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin:...', 'PWD': '/Users/michael', 'LANG': 'zh_CN.UTF-8', ...}
    >>> p.terminate() # 结束进程
    Terminated: 15 <-- 自己把自己结束了
    

    和获取网络连接类似,获取一个root用户的进程需要root权限,启动Python交互环境或者.py文件时,需要sudo权限。

    psutil还提供了一个test()函数,可以模拟出ps命令的效果:

    $ sudo python3
    Password: ******
    Python 3.6.3 ... on darwin
    Type "help", ... for more information.
    >>> import psutil
    >>> psutil.test()
    USER         PID %MEM     VSZ     RSS TTY           START    TIME  COMMAND
    root           0 24.0 74270628 2016380 ?             Nov18   40:51  kernel_task
    root           1  0.1 2494140    9484 ?             Nov18   01:39  launchd
    root          44  0.4 2519872   36404 ?             Nov18   02:02  UserEventAgent
    root          45    ? 2474032    1516 ?             Nov18   00:14  syslogd
    root          47  0.1 2504768    8912 ?             Nov18   00:03  kextd
    root          48  0.1 2505544    4720 ?             Nov18   00:19  fseventsd
    _appleeven    52  0.1 2499748    5024 ?             Nov18   00:00  appleeventsd
    root          53  0.1 2500592    6132 ?             Nov18   00:02  configd
    ...
    

    小结

    psutil使得Python程序获取系统信息变得易如反掌。

    psutil还可以获取用户信息、Windows服务等很多有用的系统信息,具体请参考psutil的官网:https://github.com/giampaolo/psutil

  • 相关阅读:
    KINDLE 小说下载--超级书库
    修改PR Cs6,PS Cs6,AU Cs6的启动界面
    SQLMAP用户手册
    Burp Suite 实战指南--说明书
    XSS跨站测试代码
    万能密码字典
    python数据结构之队列(一)
    python数据结构之栈
    python实现链表(二)
    python实现链表(一)
  • 原文地址:https://www.cnblogs.com/springsnow/p/11974749.html
Copyright © 2011-2022 走看看