博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
用PYTHON实现MORE(最终版)
阅读量:6659 次
发布时间:2019-06-25

本文共 7421 字,大约阅读时间需要 24 分钟。

hot3.png

#!/usr/bin/env python# -*- coding:utf-8-*-#名字:more.py#功能:#           实现linux中more的基本功能,当more 后加一个文件名参数时候,分屏显示,按空格换页,按回车换行',在左下角显示百分比;#           可以处理管道参数的输入,处理选项+num:从指定行开始显示,+/string :查找字符串,从指定字符串之后开始显示#运行环境:#           安装有PYTHON的linux系统#调用示例:#            more.py [+num ]     [+/pattern]  filename#            command|./more.py [+num ] [+/pattern]#            more.p  --help     输出帮助信息#            num 是 要从第几行开始显示,pattern是要在文件中查找的字符串           import osimport sysimport curses      #用于获取终端的尺寸import re              #用于字符匹配import signal      #用于处理ctrl+c中断import fcntl        # 处理显示过程中屏幕的变化import termios  #获取终端信息import structpage_len = 0   #满屏时可以显示的最大行数line_len = 0   #满屏时每行可以显示的最大字节数sig_up = 0    #中断信号标志winsz_chg = 0   #窗口大小改变标志def win_sz_chg(signum, frame):    '''  函数功能:本函数是屏幕变化信号的处理函数'''    global page_len, line_len, winsz_chg    winsz_chg = 1    signal.signal(signal.SIGWINCH, signal.SIG_IGN)    s = struct.pack("HHHH", 0, 0, 0, 0)      a = struct.unpack('hhhh', fcntl.ioctl(sys.stdout.fileno(), termios.TIOCGWINSZ , s))    #获取当前窗口的大小    page_len = int(a[0]) - 1  #留一行显示进度    line_len = int(a[1])    signal.signal(signal.SIGWINCH, win_sz_chg) #不调用会导致只能检测一次屏幕变化        signal.signal(signal.SIGWINCH, win_sz_chg)  #接收处理窗口改变信号def term_do_exit(signum, frame):    '''  函数功能:键盘中断信号的响应函数'''    global sig_up    sig_up = 1           #将键盘中断标志置1    os.system("stty -F /dev/tty echo") #恢复终端输出回车有效状态    os.system("stty -F /dev/tty -cbreak")#重新设着屏幕为输入字符回显状态    return        signal.signal(signal.SIGINT, term_do_exit)   #接收并处理键盘中断信号def usage():    '''显示脚本的各参数的含义和调用格式'''    print "-----------------usage-----------------"    print "1./more.py [+num] [+/pattern] filename"    print "2 command | ./more.py"    print "num: Start at line number num. "    print "pattern:Start at line number num."    print "space: next page"    print "q :do_exit the program"    print "enter:next line"    print"----------------------------------------"    sys.exit()def do_exit():    '''用于系统退出 '''    os.system("stty -F /dev/tty echo") #恢复终端输出回车有效状态    os.system("stty -F /dev/tty -cbreak")#重新设着屏幕为输入字符回显状态    sys.exit()    def is_input():    ''' 检测是否有管道数据输入 '''    try:        f = sys.stdin.fileno()  #判断时候有管道输入        init_tty = termios.tcgetattr(f)     #当没有管道输入,也没有参数时候,显示提示        return 0    except:        return 1    def get_line_num(args):    ''' 从命令行参数中获取开始显示的指定行    参数:args:从命令行获取的参数返回值:要开始显示的指定行  '''    line_num = 1    for i in args:            #匹配要从第几行开始的行数        ln = re.search(r'\+\d+', str(i))        if ln:            line_num = int(ln.group().lstrip('+'))#采用正则表达式处理去掉‘+’,得到开始显示的行号            break    return line_num                     def get_patstr(args):    '''从命令行中获取要查找的字符串     参数:args:从命令行中获取的参数为返回值:要查找的字符串:  '''    patstr = ""    for i in args:            #获取要匹配的字符串        pa = re.search(r'(\+\/\w*[^\s])', str(i))        if pa:            break    if pa:        patstr = str(pa.group().lstrip('+/'))    return patstr        def get_args():    '''用于从命令行获取参数,解析各参数    返回值:(line_num,patstr,fp):要开始显示的指定行,要查找的字符串,要操作的文件对象 '''    line_num = 1    patstr = ""    args = sys.argv[1:]    if not args:        if is_input():      #在没有参数时候,判断是否为管道命令输入,不是提示正确输入参数            fp = sys.stdin            return (line_num, patstr, fp)        else:            usage()    else:        if args[-1] == "--help":            usage()        line_num = get_line_num(args)        patstr = get_patstr(args)        if '+' not in args[-1]:            filename = args[-1]            if not os.path.exists(filename):                print " 没有那个文件或目录"                do_exit()            else:                fp = open(filename)        else:            if not is_input():                usage()            else:                fp = sys.stdin    return (line_num, patstr, fp)def get_screen_size():    ''' 用于获取文件显示终端的尺寸   '''    global page_len, line_len           screen = curses.initscr()           page_len, line_len = screen.getmaxyx()#获取屏幕显示尺寸    page_len = page_len - 2   #去掉输入命令那行,和最后要显示more的那一行    curses.endwin()     #此处不结束会导致后面显示的乱码    def show_more(pre_pg_len):    ''' 等待键盘输入命令 ,进行相应的处理。    :param pre_pg_len:屏幕改变以前保存的可显示的最大行数'''    global  sig_up, winsz_chg, page_len    ft = open('/dev/tty')   #打开标准终端输入    sys.stdout.flush()    #刷新缓存输出,否则显示会出现问题    c = ''    while True:        try:            c = ft.read(1)#读取一个字符        except IOError:            if sig_up:                do_exit()   #键盘中断退出程序        if c == " ":            print "\033[20D\033[K" #控制光标回到more--反白字体的行首,删除此行以达到more不随文字滚动效果            if winsz_chg:        #如果此时屏幕大小变化,第一次返回之前屏幕满屏行数                winsz_chg = 0                return pre_pg_len            else:                return page_len     #当输入是空格时候,分屏显示,显示下一屏        elif c == "q":            print "\033[20D\033[K"            return 0          #当输入是"q"时,退出显示        elif c == '\n':            print "\033[20D\033[K",            return 1           #当输入是换行符时候,多显示一行            def skip_ln(fp, line_num):    ''' 读取文件到指定开始显示的行  '''    n = line_num - 1    while n:        fp.readline()        if not fp:            return        n = n - 1def search(fp, patstr):    ''' 在文件中寻找要查找的字符串。    :param fp:要显示的文件对象        :param patstr:要查找的检索词  '''    global  sig_up    text = ' '    while True:        try:            s = fp.read(1)                     if not s:                print "can not find the string in the file"                do_exit()            text = text + s            if patstr in text:                return        except IOError:            if sig_up:                do_exit()      def show_prog(read_size, total_size):             '''  在显示屏幕的左下角显示反显的"More"        当要显示的是一个文件时,同时显示已经显示文件的百分比        当显示的是一个管道输入时,只显示“More”    :param read_size: 已经显示的文件    :param total_size:要显示的文件的总大小  '''    if total_size:        prog = int(read_size * 100 / float(total_size))        print  "\033[7m --More--" + str(prog) + '%' + "\033[0m", #输出反白的文字“more”和显示百分数    else:        print  "\033[7m --More--" + "\033[0m", #输出反白的文字“more”    return            def do_more(fp , line_num , patstr):    '''分屏显示文件内容    :param fp:要显示的文件对象    :param page_len: 可显示的最大行数    :param line_len:可每行可显示的最大字节数   '''    global page_len, line_len    read_size = 0    total_size = 0    os.system("stty -F /dev/tty cbreak") #调用linux命令设置屏幕为不等回车    os.system("stty -F /dev/tty -echo ")#设置屏幕为输入字符不回显    if fp != sys.stdin:            fp.seek(0, 2)  #获取文件的总字节数,以便后来显示输出的百分比            total_size = fp.tell()            fp.seek(0, 0)    if line_num != 1:        skip_ln(fp, line_num)    if patstr:        search(fp, patstr)    try:        line = fp.readline(line_len)#按行读取文件,可以设置最大读取字数,当遇到“\n”时候,将"\n"读入结束。        read_size = len(line)        num_lns = 0        while line:            if num_lns == page_len: #每次输出满屏后,等待指令                pre_pg_len = page_len                show_prog(read_size, total_size)                reply = show_more(pre_pg_len)                           if reply == 0:                    break                num_lns -= reply            print line.strip('\n') #用,来消除print 输出的换行符            sys.stdout.flush()    #刷新缓存,否则会出现文件显示迟缓的问题            read_size = read_size + len(line)            num_lns += 1            line = fp.readline(line_len)        fp.close()    except IOError:        if sig_up:            do_exit()               if __name__ == '__main__':    get_screen_size()  #获取显示终端的尺寸    (line_num, patstr, fp) = get_args()    do_more(fp, line_num, patstr)    do_exit()

转载于:https://my.oschina.net/annieduoduo/blog/71360

你可能感兴趣的文章
我为公司做的总体架构,欢迎提建议
查看>>
如何成为一名优秀的web前端工程师(前端攻城师)?
查看>>
【8086汇编基础】04--中断
查看>>
laravel 5.5 数据库迁移报错
查看>>
小记:《技术进步引发的灵感革命》网易游戏学院第二届公开日
查看>>
【C#并发】00概述
查看>>
[转] Webpack的devtool和source maps
查看>>
查询表结构视图
查看>>
input type file onchange上传文件的过程中,同一个文件二次上传无效的问题。
查看>>
Java入门篇(一)——如何编写一个简单的Java程序
查看>>
Kotlin Coroutine(协程) 基本知识
查看>>
JavaScript系列-this详解
查看>>
浅谈 Objective-C Associated Objects
查看>>
陆瀚卿:非农做单十大技巧,掌握六条算你赢!
查看>>
9章 RxJava混合实战
查看>>
1002 A+B for Polynomials
查看>>
实在不能忍了,记一下浏览器事件环与node事件环的区别
查看>>
组件:Text
查看>>
第十四篇 SpringBoot2 x整合MyBatisGenerator
查看>>
js实现的大根堆算法(基于链式的m叉树)
查看>>