栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 系统运维 > 运维 > Linux

Python实现上下文管理详解

Linux 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Python实现上下文管理详解

1、如何让对象支持上下文管理?

        实际案例:

                我们实现了一个telnet客户端TelnetClient,调用实例的start()方法启动客户端与服务器交互,交互完毕后需调用cleanup()方法,关闭已连接的socket,以及将操作历史记录写入文件并关闭。

                能否让TelnetClient的实例支持上下文管理协议,从而替代手工调用cleanup()方法。

        解决方案:

                实现上下文管理协议,需定义实例的__enter__,__exit__方法,它们分别在with开始和结束时被调用。

2、代码演示

(1)上下文管理的例子

'''
经常在对文件操作的时候使用上下文管理使用with语句,
使用上下文管理的好处就是在with语句结束以后不用显示的close去关闭文件, 
这样一个清理工作会交给上下文管理协议,让它们自动帮我们完成。
'''
with open('demo.txt', 'w') as f:
    f.write('abcdef')
    f.writelines(['xyzn', '123n'])
# f.close()

(2)实现一个类的对象支持上下文管理协议

from telnetlib import Telnet
from sys import stdin, stdout
from collections import deque


class TelnetClient(object):
    def __init__(self, addr, port=23):
        self.addr = addr
        self.port = port
        self.tn = None

    def start(self):
        # 人为显示抛出一个异常,用来证明即使有异常__exit__同样会被调用
        # raise Exception('Test')
        # 进行登录操作输入用户名和密码
        # user
        t = self.tn.read_until(b'login: ')
        stdout.write(t)
        user = stdin.readline()
        self.tn.write(user)
        # password
        t = self.tn.read_until(b'password: ')
        if t.startswith(user[:-1]):
            t = t[len(user) + 1:]
        stdout.write(t)
        self.tn.write(stdin.readline())

        # 登录到服务器的shell当中
        t = self.tn.read_until('$ ')
        stdout.write(t)
        while True:
            uinput = stdin.readline()
            if not uinput:
                break
            self.history.append(uinput)
            self.tn.write(uinput)
            t = self.tn.read_until('$ ')
            stdout.write(t[len(uinput) + 1:])

    def cleanup(self):
        pass

    def __enter__(self):
        # Telnet连接对象
        self.tn = Telnet(self.addr, self.port)
        # 创建队列存储用户的操作历史记录
        self.history = deque()
        # __enter__()返回的才是with中as的对象,self自身
        return self

    # 3个参数分别为异常类型、异常值、跟踪栈信息
    # 无异常的情况下三个参数都是None
    def __exit__(self, exc_type, exc_val, exc_tb):
        # print('In __exit__')
        # 关闭服务器连接的socket
        self.tn.close()
        self.tn = None
        with open(self.addr + 'history.txt', 'w') as f:
            f.writelines(self.history)
        # 默认返回None
        # return True # 不抛出异常


with TelnetClient('127.0.0.1') as client:
    client.start()

print('END')

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/881776.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号