这也是我第一次系统对软件系统进行测试,从中了解到了很多之前我从来没有注意到的一些情况,也让我对自己的编码以及软件设计等方面重新进行了深刻的反思,意识到了之前自己的思维上的很多的不足。
个人认为,对软件或程序进行测试,很大一部分就是对当初对软件或者程序的设计过程中的思考是否具有完备性的一次自我检查。
下面列举了很多我在测试过程得到的一些感悟和思考:
你很难预测用户会采取怎样的行为,因此要做“最坏”的打算。
登陆过程,一定要防止用户使用SQL注入。以本系统中的代码为例,一定要采用数据库连接类中对应的API完成字符串拼接,而不是直接进行拼接。
sql = "SELECT * FROM " + DB_TABLE_NAME + " WHERE username=%s AND password=%s" self._cursor.execute(sql, (username, password))
而不是
sql = "SELECT * FROM " + DB_TABLE_NAME + " WHERE username=%s AND password=%s" % (username, password) self._cursor.execute(sql)
该部分总结:因此在测试过程中应该用“最坏的恶意”去揣测用户,尤其是在给予用户极大自主权的部分(例如自行输入的文本框,或者自行选择文件的文本对话框等),一定要对用户的行为进行严格的判断并给出对应合理的解决措施。
很多情况下,程序都会具有在开发时很难注意到的、较为极端的,但却能又能让程序直接崩溃的特例。
self._ax.axis([0, (length - 1) * interval, 0, 100])
self.draw()
因为该部分横坐标最大长度是通过截取帧的数目减去1乘上提取帧间隔得到的。当length的值为0或者1,length - 1的值为-1或0,自然出现异常。
当然这部分的确不应该在测试过程中才发现,在当时进行编码的过程中就应该注意到这里length - 1是可能是负数或零的,所以一定要进行异常分析。
因此这部分需要严格设定情绪的初始值,而不是寄望于不停地使用上一帧的情绪值。
该部分总结:编码过程中,一定要养成良好的习惯!
例如,出现对数组的索引访问,一定要考虑一下,这个索引会不会在某种特别特别极端的情况下变成负的或者出现溢出?
对任何对象的成员函数进行调用时,一定要考虑一下,这个对象是否正确的初始化?是否可能为None从而出现空指针错误?
...
建立有效的异常处理与测试机制并打印控制台日志能够极大地为测试过程提供方便。
所幸我在最开始编码上就极力要求自己在代码中加入非常严格的异常处理以及测试部分,这让我们在最后测试的过程中少走了很多的弯路。
例如,在用户管理的数据库帮助类下面,就有这样的主函数帮助测试:
if __name__ == ‘__main__‘: usersHelper = UsersHelper() username = ‘456‘ password = ‘456‘ # print(usersHelper.register(username, password)) print(usersHelper.confirm(username, password))
例如,对视频进行情绪曲线作图时,控制台日志如下,里面的一些警告信息(例如找到了多对眼睛等)都一目了然。
同样的,在通过摄像头进行识别时,也能清楚地看到中间的过程,包括缓存的存储信息,找不到人脸的错误信息,找不到人眼的警告信息等。
例如该系统中对人脸的截取部分。本系统不仅对人脸进行截取,还能够对人脸进行摆正操作,然而这些过程都可能出现很多异常的东西,如找不到人脸,找到多个人脸,找不到眼睛(因为人脸摆正依赖对眼睛的识别)的异常处理必须严谨慎重。代码如下:
def get_face(self, image, rotate=False): """ :param image: 需要截取的人脸矩阵 :param rotate: 是否进行摆正处理,默认为False :return: 返回截取的脸,与一个flag值,为True表示有找到多个脸 """ if rotate: # 试图旋转 try: angle = self.get_angle(image) except: # 捕捉眼睛过程出现异常,此时放弃旋转 pass else: # 未出现异常,正常旋转 image = self.rotate_about_center(image, angle) faces = self._face_classifier.detectMultiScale(image) if len(faces) == 0: # 找不到人脸 print(‘ERROR: No Face Found‘.center(80, ‘-‘)) raise Exception(‘No Face Found‘) # 抛出异常 elif len(faces) > 1: # 找到多个人脸时,只在控制台上打印警告信息,不抛出异常,但令flag值为True print(‘WARNING: Multiple Faces Found‘.center(80, ‘-‘)) return image[faces[0][1]:faces[0][1] + faces[0][3], faces[0][0]:faces[0][0] + faces[0][2]], True return image[faces[0][1]:faces[0][1] + faces[0][3], faces[0][0]:faces[0][0] + faces[0][2]], False
在设计类的过程中抛出的异常,在用到对象的时候自然就要用到,例如
try: image, flag = self._faceIdentifier.get_face(image, rotate=True) # 尝试获取脸部 except Exception as e: # 若找不到人脸,直接返回错误信息,函数结束 return ‘‘, [], str(e) if flag: # 若找到多个人脸,暂时不结束程序,但记录错误信息 exception = ‘Multiple Faces Found‘
此外,对很多容易产生异常的函数,一定要对其传入参数进行严格判断,并抛出对应异常
def get_data(self, X, Y, n): """ 从CK+数据集获取数据存到给定矩阵中 :param X: numpy(n, IMAGE_SIZE, IMAGE_SIZE, CHANNEL_NUM), 输入矩阵 :param Y: numpy(n), 结果矩阵 :param n: int, 需要获取的数据数量 :return: """ if len(X) != n or len(Y) != n: # 输入矩阵或输出矩阵的维度不符合要求 raise Exception(‘Illegal Length of Lists‘) if n > self._size: # 试图获取的数据数量超出SQL查询结果 raise Exception(‘CK+ Samples Out of Range‘)
该部分总结:磨刀不误砍柴工,写代码时写好注释、测试函数、异常处理等等,在测试的时候则会真的省去很多事。到底是什么地方出现问题一目了然。
总结
软件测试是一个相当具有技巧,并相当必要的过程。由于我们下学期才上系统的软件测试的课程,现在我们能做的也是必须要做的就是严谨自己的思维,增加自己考虑问题的全面性,并对用户的任何可能行为作出全面的预测分析,在编码中多做工夫,从而就能在软件测试过程中少做工夫。
原文:https://www.cnblogs.com/cadenza/p/10222166.html