一、常见的异常

1、NameError 未定义变量异常

print(a)
# 输出:NameError: name 'a' is not defined

2、IndexError 下标越界异常

list1 = [10]
print(list1[1])
# 输出:IndexError: list index out of range

3、IOError 输入输出异常

try:
    raise IOError  # 假装这里有异常,一般针对难以复现的异常
except:
    print('文件读写错误')

4、FileNotFoundError 找不到文件异常

with open('./中国男足永夺世界冠军.txt') as file1:
    file1.read()
# 输出:
# FileNotFoundError: [Errno 2] No such file or directory: 'd/中国男足永夺世界冠军.txt'

5、ZeroDivisionError 0不能作为分母异常

print(1/0)
# 输出:
# ZeroDivisionError: division by zero

6、NoSuchElementException 找不到元素

原因:
1、元素表达式写错了。
2、元素表达式没错,但是元素的确不在当前界面上。
3、元素等待,元素未加载完成,需要等待。
4、内嵌网页(在一个html标签里面嵌套一个html)。
    1、iframe标签相当于大房子里面的小房子。
5、多标签页(点击1个元素打开另外1个页签-1,例如点击'知识手册')。
	1、要定位"页签-1"页面的元素需要进入到"页签-1"页面。

7、StaleElementReferenceException 陈旧的元素引用异常

1、原因:
    1、进行ui自动化的时候,都是先找到元素,然后再操作元素,这是两个动作,如果在这两个动作之间,js 把网页刷新了,
那么操作元素的时候,就会碰到陈旧的元素引用异常。
2、解决:
    1、每次操作元素的时候都去实时获取元素。
    2、在po模式中的解决方法是,把每一个元素定位写成函数。

案例:

class HomePage:
    def __init__(self, url, driver):
        self.driver = driver

    # 我的主页
    def myPageBox(self):
        return self.driver.find_element_by_css_selector(
            "[class=\"nav nav-pills nav-stacked custom-nav js-left-nav\"] > li:nth-child(1)")

    # 项目管理
    def prM(self):
        return self.driver.find_element_by_css_selector(
            "[class=\"nav nav-pills nav-stacked custom-nav js-left-nav\"] > li:nth-child(2)")

    # 功能:依次点击 我的主页、项目管理、我的主页
    def foo(self):
        """
        1、点击之后,页面刷新了。
        2、每一次操作元素时,都是通过函数重新寻找的。
        :return:
        """
        self.myPageBox().click()
        self.prM().click()
        self.myPageBox().click()


hp = HomePage("", driver)
hp.foo()

二、处理异常

1、捕获系统异常,转化成用户能看懂的提示
2、一个标准的try-except语句,至少要有一个except,也可以多个except,也可以"额外"加else语句与finally语句
3、如果事先不知道是什么异常,可以直接用except,会捕获所有的异常

try:
    input = int(input('请输入一个整数:'))
    print(1/input)             # 有异常捕获异常,没异常正常执行
except ZeroDivisionError:      # 捕获0作为分母的异常
    print('0不可以作为分母')
except ValueError:             # 捕获输入不是数字无法转int的异常
    print('您输入的不是整数')
except:                        # 会捕获所有的异常
    print('程序出现异常')
else:                          # 当程序正常执行完毕,没有出现异常,则执行一次else语句
    print('没有出现异常')
finally:                       # 无论程序是否出现异常,都会执行
    print('程序执行完毕!')

三、手动抛出异常

try:
    raise IOError  # 假装这里有异常,一般针对难以复现的异常
except:
    print('文件读写错误')

4、断言 assert,一般用于pytest框架

assert 1 == 2  # 当断言的结果为真时,断言不做事情,为假时,断言生效抛出异常
# 输出:AssertionError

try:
    assert 1 == 2
except AssertionError:
    print('断言为假.')

补充:
1、所有的异常都是Exception的子类,或子类的子类
2、Exception是object的子类的子类

print(NameError.__bases__)
print(IndexError.__bases__)
print(LookupError.__bases__)
print(Exception.__bases__)
print(BaseException.__bases__)

# 输出:
# (<class 'Exception'>,)
# (<class 'LookupError'>,)
# (<class 'Exception'>,)
# (<class 'BaseException'>,)
# (<class 'object'>,)