一、Python的异常处理
因为想到自己不断尝试写小程序的话会用到抛出异常信息来判断哪里出现了问题:
?
1
usage: raise [Exception [, args [, trackback]]]
上面是Python的raise的用法,下面是自己用这个方法实现异常的抛出方法:
?
1
2
3
4
5
6
7
8
def check_args(args):
if not args.host:
msg = ‘Args missing! One of the following args should be specified \n’ \
‘–host 192.168.1.1 \n’ \
‘-f TargetFile \n’
raise Exception(msg)
#参考别人的代码模式,我这样写来抛出异常.
二、list转str
这个问题是因为自己的无知吧:
命令行传入的host(即ip地址)是list形式,想要通过list转为str的格式之后来进行socket.connect() , 报错:
自己想象的姿势:
?
1
client.connect((str(args.host), args.p))
正确的姿势:
?
1
client.connect((”.join(args.host), args.p))
上图证明自己的愚蠢(可能下次还会犯同样的错):
三、argparse函数
出发的动机是因为自己写的小程序要通过命令行的形式传参并执行命令.
用自己的简单小实例来显示函数的具体用法吧:
?
1
2
3
4
5
6
7
8
9
def parse_args():
parser = argparse.ArgumentParser(prog = ‘hello’,
formatter_class = argparse.RawTextHelpFormatter,
description = ‘* A tiny toy for fun *\n’
‘By ST(www.********)’,
usage = ‘hellPLC.py [options]’)
parser.add_argument(‘-host’, metavar = ‘HOST [HOST2 HOST3 …]’, type = str,
default = ”, nargs = ‘*’,
help = ‘Scan the host form command line’)
代码很简单,一看就懂大概函数是什么样子,重点是add_argument的参数,当你传入的cmd参数是这种方式的话:
我们可以看到,需要在-p之后跟一个int型的参数才可以,还有另外一种形式.
?
1
2
parser.add_argument(‘-p’, metavar = ‘PORT’, type = int,
default = ”)
这种方式不需要跟参数,只需要类似于”python -h”这种形式就可以执行并得到想要的结果:
这里就需要对这个函数的各个参数的功能有个基本的了解,这样才能用起来舒服.
?
1
2
parser.add_argument(‘-b’, default= False, dest=’b’, action=’store_true’,
help = ‘Get the base info’)
下图就是我们可以跟的参数,自己的问题就是区分清楚action和dest这两个参数,还有default,type等.
四、正则匹配
这个东西有点高深了,最初的想法是使用正则来转变接收到的数据的格式,问过达哥之后原本recv到的数据在encode之后的类型是”str”,之前自己的想法是把数据转成list格式,之后提取之类的方便,但”str”类型也可以直接利用偏移来进行分析也可以,直接用”str[]”就可以搞定,因为这个过程中不需要可视化的打印出来,需要打印的是分析之后的结果,所以正则先用不到。
不过还是要把觉得不错的链接放一下:
Python入门篇之正则表达式
http://www.jb51.net/article/56436.htm
Python 匹配任意字符(包括换行符)的正则表达式写法
http://www.jb51.net/article/20654.htm
五、格式化字符串
?
1
2
3
4
temp = ‘123456’
print(“word:%s” %temp)
output: word:123456
很简单就可以搞定,但是当时想要返回取多个返回值,一下子蒙住不知道怎么搞了,呵呵了~
?
1
2
3
4
f.write (“Block Type: %s \n”
“Block count: %s \n”
%(block_type, block_count))
return block_type, block_count
六、文件读写操作
这个问题别人的博客写的很好很详细了
?
1
f = open(r’C:\Movie\test2.txt’,’w’)
直接打开一个文件,如果文件不存在则创建文件,只能只用写命令
?
1
f = open(r’C:\Movie\test2.txt’)
打开一个文件,只能只用读命令
?
1
f.read([size])
size为读取的长度,以byte为单位,如果不写则读取全部内容
?
1
f.readline([size])
读一行,如果定义了size,有可能返回的只是一行的一部分。每读取一次,文件操作符向下移动一行。
?
1
f.readlines([size])
把文件每一行作为一个list的一个成员,并返回这个list。其实它的内部是通过循环调用readline()来实现的。如果提供size参数,size是表示读取内容的总长,也就是说可能只读到文件的一部分。
关于open模式,参数不同达到的效果也是不同的,比如我需要的是在一个日志文件中不断的追加新的东西而不是每次都直接覆盖掉,所以我用到了”a+”
关于open模式的参数:
w 以写方式打开
a 以追加模式打开
r+ 以读写模式打开
w+ 以读写模式打开
a+ 以读写模式打开 (我用到了这个模式,读写模式不断追加新的东西)
rb 以二进制读模式打开
wb 以二进制写模式打开
ab 以二进制追加模式打开
rb+ 以二进制读写模式打开11 wb+ 以二进制读写模式打开12 ab+ 以二进制读写模式打开
还有一些操作的区别,我当时用到的问题是f.readline,每次都是读文件的一行,没有把全部的内容都读出来,所以区别还是很明显,要区分清楚.
写操作的用法和区别:
?
1
f.write(“str”)
把str写到文件中,write()方法不会在str后加上一个换行符
?
1
f.writelines(seq)
把seq的内容全部写到文件中(多行一次性写入)。这个函数也只是忠实地写入,不会在每行后面加上任何东西。
?
1
f.close()
关闭文件。在读命令或者写命令结束时,需要用关闭。如果文件关闭后依然操作,会抛出ValueError: I/O operation on closed file
?
1
f.tell()
返回文件操作标记的当前位置,以文件的开头为起点
?
1
fp.next()
返回下一行内容,并将文件操作标记位移到下一行。把一个file用于for … in file这样的语句时,就是调用next()函数来实现遍历的。
?
1
fp.seek(offset[,whence])
将文件操作标记为移动到offset位置。
七、遇到的报错情况
自己不想要定义太多的函数,个人感觉是想要将不同功能的函数进行分类,因此我尝试用到了类,知道自己这么做可能是想法上就有些不对的,但是想要瞎试试看看效果,所以结果就是报错了.
如图,这是别人遇到的方法,和我的是一样的,我没有定义静态的函数,所以每次使用之前都需要进行实例化才能调用.
错误:
TypeError: ‘NoneType’ object is not callable
【错误分析】我是在别的文件中写了一个函数,然后在python console调用使用该函数,出现了上述错误。原因是因为没有对别的文件中函数进行编译,只要打开该调用函数的文件,然后点击运行,之后再在python console中调用使用该函数就不会报错。
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】第二个参数必须为类,否则会报TypeError,所以正确的应该是这样的:
但如果第二个参数是类型对象,则不会报上面的错误,是允许的,比如说:
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】这个涉及到调用顺序问题,即解析方法的MRO调用顺序,在Python2.7版本之后,这样调用会报错,
必须是子类先放前面,然后才是父类.如下所示,方不会报错.
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】foo()未带参数self,也未带cls参数,属于类的静态方法,类的静态方法调用,实例不能直接调用,需要再声明一个静态方法
或者通过@staticmethod来调用
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】__dict__是实例的特殊属性,但在内建属性中,不存在__dict__属性,一般的情况是:
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】如果定义了构造器,它不应当返回任何对象,因为实例对象是自动在实例化调用后返回的。相应地,__init__()就不应当返回任何对象(应当为None);否则就可能出现冲突,因为只能返回实例。试着返回非None的任何其他对象都会导致TypeError异常
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】不要误以为元组里有两个参数,将元组传进去就可以了,实际上元祖作为一个整体只是一个参数,
实际需要两个参数,所以报错。必需再传一个参数方可.
1 >>> f(t, ‘var2’)
2 (‘a’, ‘b’) var2
更常用的用法: 在前面加*,代表引用元组
1 >>> f(*t)
2 ‘a’, ‘b’
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】在C++,Python中默认参数从左往右防止,而不是相反。这可能跟参数进栈顺序有关。
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】这是Python中字典键错误的提示,如果想让程序继续运行,可以用字典中的get方法,如果键存在,则获取该键对应的值,不存在的,返回None,也可打印提示信息.
1 >>> D1.get(‘z’, ‘Key Not Exist!’)
2 ‘Key Not Exist!’
错误:
复制代码
1 >>> from math import sqrt
2 >>> exec “sqrt = 1”
3 >>> sqrt(4)
4
5 Traceback (most recent call last):
6 File “
7 sqrt(4)
8 TypeError: ‘int’ object is not callable
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
复制代码
1 >>> seq = [1, 2, 3, 4]
2 >>> sep = ‘+’
3 >>> sep.join(seq)
4
5 Traceback (most recent call last):
6 File “
7 sep.join(seq)
8 TypeError: sequence item 0: expected string, int found
复制代码
【错误分析】join是split的逆方法,是非常重要的字符串方法,但不能用来连接整数型列表,所以需要改成:
错误:
【错误分析】Python中原始字符串以r开头,里面可以放置任意原始字符,包括\,包含在字符中的\不做转义。
但是,不能放在末尾!也就是说,最后一个字符不能是\,如果真 需要的话,可以这样写:
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
代码:
1 bad = ‘bad’
2
3 try:
4 raise bad
5 except bad:
6 print ‘Got Bad!’
错误:
【错误分析】因所用的Python版本2.7,比较高的版本,raise触发的异常,只能是自定义类异常,而不能是字符串。所以会报错,字符串改为自定义类,就可以了。
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
复制代码
1 class Super:
2 def method(self):
3 print “Super’s method”
4
5 class Sub(Super):
6 def method(self):
7 print “Sub’s method”
8 Super.method()
9 print “Over…”
10
11 S = Sub()
12 S.method()
复制代码
执行上面一段代码,错误如下:
【错误分析】Python中调用类的方法,必须与实例绑定,或者调用自身.
ClassName.method(x, ‘Parm’)
ClassName.method(self)
所以上面代码,要调用Super类的话,只需要加个self参数即可。
复制代码
1 class Super:
2 def method(self):
3 print “Super’s method”
4
5 class Sub(Super):
6 def method(self):
7 print “Sub’s method”
8 Super.method(self)
9 print “Over…”
10
11 S = Sub()
12 S.method()
13
14
15 #输出结果
16 >>>
17 Sub’s method
18 Super’s method
19 Over…
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】reload期望得到的是对象,所以该模块必须成功导入。在没导入模块前,不能重载.
1 >>> import sys
2 >>> reload(sys)
3
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误
复制代码
1 >>> def f(x, y, z):
2 return x + y + z
3
4 >>> args = (1,2,3)
5 >>> print f(args)
6
7 Traceback (most recent call last):
8 File “
9 print f(args)
10 TypeError: f() takes exactly 3 arguments (1 given)
复制代码
*args,才是将元组中的每个元素作为参数
1 >>> f(*args)
2 6
错误:
【错误分析】错误原因**匹配并收集在字典中所有包含位置的参数,但传递进去的却是个元祖。
所以修改传递参数如下:
1 >>> args = {‘a’:1,’b’:2,’c’:3}
2 >>> args[‘d’] = 4
3 >>> f(**args)
4 1 2 3 4
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】在函数hider()内使用了内置变量open,但根据Python作用域规则LEGB的优先级:
先是查找本地变量==》模块内的其他函数==》全局变量==》内置变量,查到了即停止查找。
所以open在这里只是个字符串,不能作为打开文件来使用,所以报错,更改变量名即可。
可以导入__builtin__模块看到所有内置变量:异常错误、和内置方法
1 import __builtin__
2 >>> dir(__builtin__)
3 [‘ArithmeticError’, ‘AssertionError’, ‘AttributeError’,…
………………………………….zip,filter,map]
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】(1)的类型是整数,所以不能与另一个元祖做合并操作,如果只有一个元素的元祖,应该用(1,)来表示
复制代码
1 In [108]: type(T1)
2 Out[108]: int
3
4 In [109]: T1 = (1,)
5 In [110]: T2 = (2,3)
6 In [111]: T1 + T2
7 Out[111]: (1, 2, 3)
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】字典中的键必须是不可变对象,如(整数,浮点数,字符串,元组).
可用hash()判断某个对象是否可哈希
但列表中元素是可变对象,所以是不可哈希的,所以会报上面的错误.
如果要用列表作为字典中的键,最简单的办法是:
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】列表属于可变对象,其append(),sort(),reverse()会在原处修改对象,不会有返回值,
或者说返回值为空,所以要实现反转并排序,不能并行操作,要分开来写。
或者用下面的方法实现:
1 In [103]: sorted(reversed([2,1,4,3]))
2 Out[103]: [1, 2, 3, 4]
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】class是Python保留字,Python保留字不能做变量名,可以用Class,或klass
同样,保留字不能作为模块名来导入,比如说,有个and.py,但不能将其作为模块导入
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】\n默认为换行,\t默认为TAB键.
所以在D:\目录下找不到ew目录下的ext.data文件,将其改为raw方式输入即可。
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】错误原因,else, finally执行位置;正确的程序应该如下:
复制代码
1 try:
2 print 1 / 0
3
4 except ZeroDivisionError:
5 print ‘integer division or modulo by zero’
6
7
8 else:
9 print ‘Continue Handle other part’
10
11 finally:
12 print ‘Done’
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
错误:
【错误分析】错误原因,列表解析中,x,y必须以数组的方式列出(x,y)
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
报错如下:
复制代码
1 >>>
2 secretCount is: 1
3 secretCount is: 2
4
5
6 Traceback (most recent call last):
7 File “D:\Learn\Python\Learn.py”, line 13, in
8 count1.__secretCount
9 AttributeError: JustCounter instance has no attribute ‘__secretCount’
复制代码
【错误分析】双下划线的类属性__secretCount不可访问,所以会报无此属性的错误.
解决办法如下:
复制代码
1 # 1. 可以通过其内部成员方法访问
2 # 2. 也可以通过访问
3 ClassName._ClassName__Attr
4 #或
5 ClassInstance._ClassName__Attr
6 #来访问,比如:
7 print count1._JustCounter__secretCount
8 print JustCounter._JustCounter__secretCount
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】属性错误,归根到底在于元组是不可变类型,所以没有这几种方法.
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】空元组和空列表,没有索引为0的项
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】一般出在代码缩进的问题
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】如果文件里面没有行了会报这种异常
有可迭代的对象的next方法,会前进到下一个结果,而在一系列结果的末尾时,会引发StopIteration的异常.
next()方法属于Python的魔法方法,这种方法的效果就是:逐行读取文本文件的最佳方式就是根本不要去读取。
取而代之的用for循环去遍历文件,自动调用next()去调用每一行,且不会报错
1 for line in open(‘test.txt’,’r’):
2 print line
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 >>> string = ‘SPAM’
2 >>> a,b,c = string
3 Traceback (most recent call last):
4 File “
5 ValueError: too many values to unpack
【错误分析】接受的变量少了,应该是
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】当映射到字典中的键不存在时候,就会触发此类异常, 或者可以,这样测试
1 >>> ‘a’ in mydic.keys()
2 True
3 >>> ‘c’ in mydic.keys() #用in做成员归属测试
4 False
5 >>> D.get(‘c’,'”c” is not exist!’) #用get或获取键,如不存在,会打印后面给出的错误信息
6 ‘”c” is not exist!’
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】一般是代码缩进问题,TAB键或空格键不一致导致
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
复制代码
1 >>>def A():
2 return A()
3 >>>A() #无限循环,等消耗掉所有内存资源后,报最大递归深度的错误
4 File “
5 class Bird:
6 def __init__(self):
7 self.hungry = True
8 def eat(self):
9 if self.hungry:
10 print “Ahaha…”
11 self.hungry = False
12 else:
13 print “No, Thanks!”
14 该类定义鸟的基本功能吃,吃饱了就不再吃
15 输出结果:
16 >>> b = Bird()
17 >>> b.eat()
18 Ahaha…
19 >>> b.eat()
20 No, Thanks!
21 下面一个子类SingBird,
22 class SingBird(Bird):
23 def __init__(self):
24 self.sound = ‘squawk’
25 def sing(self):
26 print self.sound
27 输出结果:
28 >>> s = SingBird()
29 >>> s.sing()
30 squawk
31 SingBird是Bird的子类,但如果调用Bird类的eat()方法时,
32 >>> s.eat()
33 Traceback (most recent call last):
34 File “
35 s.eat()
36 File “D:\Learn\Python\Person.py”, line 42, in eat
37 if self.hungry:
38 AttributeError: SingBird instance has no attribute ‘hungry’
复制代码
【错误分析】代码错误很清晰,SingBird中初始化代码被重写,但没有任何初始化hungry的代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
复制代码
1 __metaclass__=type
2 class Bird:
3 def __init__(self):
4 self.hungry = True
5 def eat(self):
6 if self.hungry:
7 print “Ahaha…”
8 self.hungry = False
9 else:
10 print “No, Thanks!”
11
12 class SingBird(Bird):
13 def __init__(self):
14 super(SingBird,self).__init__()
15 self.sound = ‘squawk’
16 def sing(self):
17 print self.sound
18 >>> S = SingBird()
19 >>> S.
20 SyntaxError: invalid syntax
21 >>> S.
22 SyntaxError: invalid syntax
23 >>> S.eat()
24 Ahaha…
复制代码
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
【错误分析】元组不可变,所以不可以更改;可以用切片或合并的方式达到目的.
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
复制代码
1 >>> X = 1;
2 >>> Y = 2;
3 >>> X + = Y
4 File “
5 X + = Y
6 ^
7 SyntaxError: invalid syntax
复制代码
【错误分析】增强行赋值不能分开来写,必须连着写比如说 +=, *=
1 >>> X += Y
2 >>> X;Y
3 3
4 2