千行代码入门Python

转自:https://github.com/xianhu/LearnPython

   1 # _*_ coding: utf-8 _*_
   2 
   3 '''类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算----类型和运算'''
   4 
   5 #--    寻求帮助:
   6     dir(obj)                    # 简单的列出对象obj所包含的方法名称,返回一个字符串列表
   7     help(obj.func)        # 查询obj.func的具体介绍和用法
   8     
   9 #--    测试类型的三种方法,推荐第三种
  10     if type(L) == type([]): print("L is list")
  11     if type(L) == list: print("L is list")
  12     if isinstance(L, list): print("L is list")
  13         
  14 #-- Python数据类型:哈希类型、不可哈希类型
  15     # 哈希类型,即在原地不能改变的变量类型,不可变类型。可利用hash函数查看其hash值,也可以作为字典的key
  16         "数字类型:int, float, decimal.Decimal, fractions.Fraction, complex"
  17         "字符串类型:str, bytes"
  18         "元组:tuple"
  19         "冻结集合:frozenset"
  20         "布尔类型:True, False"
  21         "None"
  22     # 不可hash类型:原地可变类型:list、dict和set。它们不可以作为字典的key。
  23     
  24 #--    数字常量
  25     1234, -1234, 0, 999999999                            # 整数
  26     1.23, 1., 3.14e-10, 4E210, 4.0e+210        # 浮点数
  27     0o177, 0x9ff, 0X9FF, 0b101010                    # 八进制、十六进制、二进制数字
  28     3+4j, 3.0+4.0j, 3J                                        # 复数常量,也可以用complex(real, image)来创建
  29     hex(I), oct(I), bin(I)                                # 将十进制数转化为十六进制、八进制、二进制表示的“字符串”
  30     int(str, base)                                                # 将字符串转化为整数,base为进制数
  31     # 2.x中,有两种整数类型:一般整数(32位)和长整数(无穷精度)。可以用l或L结尾,迫使一般整数成为长整数
  32     float('inf'), float('-inf'), float('nan')        # 无穷大, 无穷小, 非数
  33     
  34 #-- 数字的表达式操作符
  35     yield x                                                                # 生成器函数发送协议
  36     lambda args: expression                                # 生成匿名函数
  37     x if y else z                                                    # 三元选择表达式
  38     x and y, x or y, not x                                # 逻辑与、逻辑或、逻辑非
  39     x in y, x not in y                                        # 成员对象测试
  40     x is y, x is not y                                        # 对象实体测试
  41     x<y, x<=y, x>y, x>=y, x==y, x!=y            # 大小比较,集合子集或超集值相等性操作符
  42     1 < a < 3                                                            # Python中允许连续比较
  43     x|y, x&y, x^y                                               # 位或、位与、位异或
  44     x<<y, x>>y                                                        # 位操作:x左移、右移y位
  45     +, -, *, /, //, %, **                                    # 真除法、floor除法:返回不大于真除法结果的整数值、取余、幂运算
  46     -x, +x, ~x                                                        # 一元减法、识别、按位求补(取反)
  47     x[i], x[i:j:k], x(……)                                    # 索引、分片、调用
  48     int(3.14),  float(3)                                    # 强制类型转换
  49     
  50 #-- 整数可以利用bit_length函数测试所占的位数
  51     a = 1;        a.bit_length()                            # 1
  52     a = 1024    a.bit_length()                            # 11
  53     
  54 #--    repr和str显示格式的区别
  55     '''
  56     repr格式:默认的交互模式回显,产生的结果看起来它们就像是代码。
  57     str格式:打印语句,转化成一种对用户更加友好的格式。
  58     '''
  59     
  60 #--    数字相关的模块
  61     # math模块
  62     # Decimal模块:小数模块
  63         from decimal import Decimal
  64         Decimal("0.01") + Decimal("0.02")        # 返回Decimal("0.03")
  65         decimal.getcontext().prec = 4                # 设置全局精度为4 即小数点后边4位
  66     # Fraction模块:分数模块
  67         from fractions import Fraction
  68         x = fractions.Fraction(4, 6)                # 分数类型 4/6
  69         x = fractions.Fraction("0.25")            # 分数类型 1/4 接收字符串类型的参数
  70 
  71 #--    集合set
  72     '''
  73     set是一个无序不重复元素集, 基本功能包括关系测试和消除重复元素。
  74     set支持union(联合), intersection(交), difference(差)和sysmmetric difference(对称差集)等数学运算。
  75     set支持x in set, len(set), for x in set。
  76     set不记录元素位置或者插入点, 因此不支持indexing, slicing, 或其它类序列的操作
  77     '''
  78     s = set([3,5,9,10])                                  # 创建一个数值集合,返回{3, 5, 9, 10}
  79     t = set("Hello")                                     # 创建一个唯一字符的集合返回{}
  80     a = t | s          t.union(s)                            # t 和 s的并集  
  81     b = t & s         t.intersection(s)              # t 和 s的交集  
  82     c = t – s         t.difference(s)                    # 求差集(项在t中, 但不在s中)  
  83     d = t ^ s          t.symmetric_difference(s)             # 对称差集(项在t或s中, 但不会同时出现在二者中)  
  84     t.add('x')         t.remove('H')                        # 增加/删除一个item
  85      t.update([10,37,42])                                  # 利用[......]更新s集合      
  86     x in s,  x not in s                                        # 集合中是否存在某个值
  87     s.issubset(t)      s.issuperset(t)   s.copy()  s.discard(x)  s.clear()
  88     {x**2 for x in [1, 2, 3, 4]}                    # 集合解析,结果:{16, 1, 4, 9}
  89     {x for x in 'spam'}                                        # 集合解析,结果:{'a', 'p', 's', 'm'}
  90     
  91 #-- 集合frozenset,不可变对象
  92     '''
  93     set是可变对象,即不存在hash值,不能作为字典的键值。同样的还有list、tuple等
  94     frozenset是不可变对象,即存在hash值,可作为字典的键值
  95     frozenset对象没有add、remove等方法,但有union/intersection/difference等方法
  96     '''
  97     a = set([1, 2, 3])    b = set()
  98     b.add(a)                                                            # error: set是不可哈希类型
  99     b.add(frozenset(a))                                        # ok,将set变为frozenset,可哈希
 100 
 101 #--    布尔类型bool
 102     type(True)                                                        # 返回<class 'bool'>
 103     isinstance(False, int)                                # bool类型属于整形,所以返回True
 104     True == 1, True is 1                                    # 输出(True, False)
 105     
 106 #-- 动态类型简介
 107     '''
 108     变量名通过引用,指向对象。
 109     Python中的“类型”属于对象,而不是变量,每个对象都包含有头部信息,比如"类型标示符" "引用计数器"等
 110     '''
 111     #共享引用及在原处修改:对于可变对象,要注意尽量不要共享引用!
 112     #共享引用和相等测试:
 113         L = [1], M = [1], L is M                        # 返回False
 114         L = M = [1, 2, 3], L is M                        # 返回True,共享引用
 115     #增强赋值和共享引用:普通+号会生成新的对象,而增强赋值+=会在原处修改
 116         L = M = [1, 2]
 117         L = L + [3, 4]                                            # L = [1, 2, 3, 4], M = [1, 2]
 118         L += [3, 4]                                                    # L = [1, 2, 3, 4], M = [1, 2, 3, 4]
 119 
 120 #-- 常见字符串常量和表达式
 121     S = ''                                                                # 空字符串
 122     S = "spam’s"                                                    # 双引号和单引号相同
 123     S = "s\np\ta\x00m"                                        # 转义字符
 124     S = '''spam'''                                                # 三重引号字符串,一般用于函数说明
 125     S = r'\temp'                                                    # Raw字符串,不会进行转义,抑制转义
 126     S = b'Spam'                                                        # Python3中的字节字符串
 127     S = u'spam'                                                        # Python2.6中的Unicode字符串
 128     s1+s2, s1*3, s[i], s[i:j], len(s)            # 字符串操作
 129     'a %s parrot' % 'kind'                                # 字符串格式化表达式
 130     'a {0} parrot'.format('kind')                    # 字符串格式化方法
 131     for x in s: print(x)                                    # 字符串迭代,成员关系
 132     [x*2 for x in s]                                            # 字符串列表解析
 133     ','.join(['a', 'b', 'c'])                            # 字符串输出,结果:a,b,c
 134     
 135 #-- 内置str处理函数:
 136     str.upper()        str.lower()        str.swapcase()        str.capitalize()        str.title()                # 全部大写,全部小写、大小写转换,首字母大写,每个单词的首字母都大写
 137     str.ljust(width)                                            # 获取固定长度,右对齐,左边不够用空格补齐
 138     str.rjust(width)                                            # 获取固定长度,左对齐,右边不够用空格补齐
 139     str.center(width)                                            # 获取固定长度,中间对齐,两边不够用空格补齐
 140     str.zfill(width)                                            # 获取固定长度,右对齐,左边不足用0补齐
 141     str.find('t',start,end)                                # 查找字符串,可以指定起始及结束位置搜索
 142     str.rfind('t')                                                # 从右边开始查找字符串
 143     str.count('t')                                                # 查找字符串出现的次数
 144     #上面所有方法都可用index代替,不同的是使用index查找不到会抛异常,而find返回-1
 145     str.replace('old','new')                            # 替换函数,替换old为new,参数中可以指定maxReplaceTimes,即替换指定次数的old为new
 146     str.strip()        str.lstrip()        str.rstrip()        str.strip('d')        str.lstrip('d')        str.rstrip('d')
 147     str.startswith('start')                                # 是否以start开头
 148     str.endswith('end')                                        # 是否以end结尾
 149     str.isalnum()        str.isalpha()        str.isdigit()        str.islower()            str.isupper()            # 判断字符串是否全为字符、数字、大写、小写
 150 
 151 #-- 三重引号编写多行字符串块,并且在代码折行处嵌入换行字符\n
 152     mantra = '''hello world
 153             hello python
 154             hello my friend'''
 155     #mantra为'''hello world \n hello python \n hello my friend'''
 156     
 157 #-- 索引和分片:
 158     S[0], S[len(S) – 1], S[-1]                        # 索引
 159     S[1:3], S[1:], S[:-1], S[1:10:2]            # 分片,第三个参数指定步长
 160 
 161 #-- 字符串转换工具:
 162     int('42'),  str(42)                                        # 返回(42, '42')
 163     float('4.13'),  str(4.13)                            # 返回(4.13, '4.13')
 164     ord('s'),  chr(115)                                        # 返回(115, 's')
 165     int('1001', 2)                                                # 将字符串作为二进制数字,转化为数字,返回13
 166     bin(13), oct(13), hex(13)                            # 将整数转化为二进制/八进制/十六进制字符串,返回('1001', '0o15', '0xd')
 167     
 168 #-- 另类字符串连接
 169     name = "wang" "hong"    #单行,name = "wanghong"
 170     name = "wang" \
 171             "hong"            #多行,name = "wanghong"
 172 
 173 #-- Python中的字符串格式化实现1--字符串格式化表达式
 174     '''
 175     基于C语言的'print'模型,并且在大多数的现有的语言中使用。
 176     通用结构:%[(name)][flag][width].[precision]typecode
 177     '''
 178     "this is %d %s bird" % (1, 'dead')                                                    # 一般的格式化表达式
 179     "%s---%s---%s" % (42, 3.14, [1, 2, 3])                                            # 字符串输出:'42---3.14---[1, 2, 3]'
 180     "%d...%6d...%-6d...%06d" % (1234, 1234, 1234, 1234)                    # 对齐方式及填充:"1234...  1234...1234  ...001234"
 181     x = 1.23456789
 182     "%e | %f | %g" % (x, x, x)                                                                    # 对齐方式:"1.234568e+00 | 1.234568 | 1.23457"
 183     "%6.2f*%-6.2f*%06.2f*%+6.2f" % (x, x, x, x)                                    # 对齐方式:'  1.23*1.23  *001.23* +1.23'
 184     "%(name1)d---%(name2)s" % {"name1":23, "name2":"value2"}        # 基于字典的格式化表达式
 185     "%(name)s is %(age)d" % vars()                                                            # vars()函数调用返回一个字典,包含了所有本函数调用时存在的变量
 186     
 187 #-- Python中的字符串格式化实现2--字符串格式化调用方法
 188     # 普通调用
 189     "{0}, {1} and {2}".format('spam', 'ham', 'eggs')                        # 基于位置的调用
 190     "{motto} and {pork}".format(motto = 'spam', pork = 'ham')        # 基于Key的调用
 191     "{motto} and {0}".format(ham, motto = 'spam')                                # 混合调用
 192     # 添加键 属性 偏移量 (import sys)
 193     "my {1[spam]} runs {0.platform}".format(sys, {'spam':'laptop'})                                    # 基于位置的键和属性
 194     "{config[spam]} {sys.platform}".format(sys = sys, config = {'spam':'laptop'})        # 基于Key的键和属性
 195     "first = {0[0]}, second = {0[1]}".format(['A', 'B', 'C'])                                                # 基于位置的偏移量
 196     # 具体格式化
 197     "{0:e}, {1:.3e}, {2:g}".format(3.14159, 3.14159, 3.14159)        # 输出'3.141590e+00, 3.142e+00, 3.14159'
 198     "{fieldname:format_spec}".format(......)
 199     # 说明:
 200     ''' 
 201         fieldname是指定参数的一个数字或关键字, 后边可跟可选的".name"或"[index]"成分引用
 202         format_spec ::=  [[fill]align][sign][#][0][width][,][.precision][type]
 203         fill        ::=  <any character>                    #填充字符
 204         align       ::=  "<" | ">" | "=" | "^"        #对齐方式
 205         sign        ::=  "+" | "-" | " "                    #符号说明
 206         width       ::=  integer                                    #字符串宽度
 207         precision   ::=  integer                                    #浮点数精度
 208         type        ::=  "b" | "c" | "d" | "e" | "E" | "f" | "F" | "g" | "G" | "n" | "o" | "s" | "x" | "X" | "%"
 209     '''
 210     # 例子:
 211         '={0:10} = {1:10}'.format('spam', 123.456)                                # 输出'=spam       =    123.456'
 212         '={0:>10}='.format('test')                                                                # 输出'=      test='
 213         '={0:<10}='.format('test')                                                                # 输出'=test      ='
 214         '={0:^10}='.format('test')                                                                # 输出'=   test   ='
 215         '{0:X}, {1:o}, {2:b}'.format(255, 255, 255)                                # 输出'FF, 377, 11111111'
 216         'My name is {0:{1}}.'.format('Fred', 8)                                        # 输出'My name is Fred    .'  动态指定参数
 217 
 218 #-- 常用列表常量和操作
 219     L = [[1, 2], 'string', {}]                        # 嵌套列表
 220     L = list('spam')                                            # 列表初始化
 221     L = list(range(0, 4))                                    # 列表初始化
 222     list(map(ord, 'spam'))                                # 列表解析
 223     len(L)                                                                # 求列表长度
 224     L.count(value)                                                # 求列表中某个值的个数
 225     L.append(obj)                                                    # 向列表的尾部添加数据,比如append(2),添加元素2
 226     L.insert(index, obj)                                    # 向列表的指定index位置添加数据,index及其之后的数据后移
 227     L.extend(interable)                                        # 通过添加iterable中的元素来扩展列表,比如extend([2]),添加元素2,注意和append的区别
 228     L.index(value, [start, [stop]])                # 返回列表中值value的第一个索引
 229     L.pop([index])                                                # 删除并返回index处的元素,默认为删除并返回最后一个元素
 230     L.remove(value)                                                # 删除列表中的value值,只删除第一次出现的value的值
 231     L.reverse()                                                        # 反转列表
 232     L.sort(cmp=None, key=None, reverse=False)        # 排序列表
 233     a = [1, 2, 3], b = a[10:]                            # 注意,这里不会引发IndexError异常,只会返回一个空的列表[]
 234     a = [], a += [1]                                            # 这里实在原有列表的基础上进行操作,即列表的id没有改变
 235     a = [], a = a + [1]                                        # 这里最后的a要构建一个新的列表,即a的id发生了变化
 236      
 237 
 238 #-- 用切片来删除序列的某一段
 239     a = [1, 2, 3, 4, 5, 6, 7]
 240     a[1:4] = []                                                        # a = [1, 5, 6, 7]
 241     a = [0, 1, 2, 3, 4, 5, 6, 7]
 242     del a[::2]                                                        # 去除偶数项(偶数索引的),a = [1, 3, 5, 7]
 243     
 244 #--    常用字典常量和操作
 245     D = {}
 246     D = {'spam':2, 'tol':{'ham':1}}                                    # 嵌套字典
 247     D = dict.fromkeys(['s', 'd'], 8)                                # {'d': 8, 's': 8}
 248     D = dict(name = 'tom', age = 12)                                # {'age': 12, 'name': 'tom'}
 249     D = dict([('name', 'tom'), ('age', 12)])                # {'age': 12, 'name': 'tom'}
 250     D = dict(zip(['name', 'age'], ['tom', 12]))            # {'age': 12, 'name': 'tom'}
 251     D.keys()    D.values()    D.items()                                    # 字典键、值以及键值对
 252     D.get(key, default)                                                            # get函数
 253     D.update(D_other)                                                                # 合并字典,如果存在相同的键值,D_other的数据会覆盖掉D的数据
 254     D.pop(key, [D])                                                                    # 删除字典中键值为key的项,返回键值为key的值,如果不存在,返回默认值D,否则异常
 255     D.popitem()                                                                            # pop字典中的一项(一个键值对)
 256     D.setdefault(k[, d])                                                        # 设置D中某一项的默认值。如果k存在,则返回D[k],否则设置D[k]=d,同时返回D[k]。
 257     del D                                                                                        # 删除字典
 258     del D['key']                                                                        # 删除字典的某一项
 259     if key in D:        if key not in D:                                # 测试字典键是否存在
 260     # 字典注意事项:(1)对新索引赋值会添加一项(2)字典键不一定非得是字符串,也可以为任何的不可变对象
 261 
 262 #--    字典解析
 263     D = {k:8 for k in ['s', 'd']}                                        # {'d': 8, 's': 8}
 264     D = {k:v for (k, v) in zip(['name', 'age'], ['tom', 12])}
 265     
 266 #-- 字典的特殊方法__missing__:当查找找不到key时,会执行该方法
 267     class Dict(dict):
 268         def __missing__(self, key):
 269             self[key] = []
 270             return self[key]
 271     dct = Dict()
 272     dct["foo"].append(1)                                                        # 这有点类似于collections.defalutdict
 273     dct["foo"]                                                                            # [1]
 274     
 275 #-- 元组和列表的唯一区别在于元组是不可变对象,列表时可变对象
 276     a = [1, 2, 3]            # a[1] = 0, OK
 277     a = (1, 2, 3)            # a[1] = 0, Error
 278     a = ([1, 2])            # a[0][1] = 0, OK
 279     a = [(1, 2)]            # a[0][1] = 0, Error
 280     
 281 #--    元组的特殊语法: 逗号和圆括号
 282     D = (12)                    # 此时D为一个整数 即D = 12
 283     D = (12, )                # 此时D为一个元组 即D = (12, )
 284     
 285 #-- 文件基本操作
 286     output = open(r'C:\spam', 'w')                # 打开输出文件,用于写
 287     input = open('data', 'r')                            # 打开输入文件,用于读。打开的方式可以为'w', 'r', 'a', 'wb', 'rb', 'ab'等
 288     fp.read([size])                         # size为读取的长度,以byte为单位
 289     fp.readline([size])                     # 读一行,如果定义了size,有可能返回的只是一行的一部分
 290     fp.readlines([size])                    # 把文件每一行作为一个list的一个成员,并返回这个list。其实它的内部是通过循环调用readline()来实现的。如果提供size参数,size是表示读取内容的总长。
 291     fp.readable()                                                    # 是否可读
 292     fp.write(str)                             # 把str写到文件中,write()并不会在str后加上一个换行符
 293     fp.writelines(seq)                            # 把seq的内容全部写到文件中(多行一次性写入)
 294     fp.writeable()                                                # 是否可写
 295     fp.close()                                # 关闭文件。
 296     fp.flush()                               # 把缓冲区的内容写入硬盘
 297     fp.fileno()                                                        # 返回一个长整型的”文件标签“
 298     fp.isatty()                                  # 文件是否是一个终端设备文件(unix系统中的)
 299     fp.tell()                                        # 返回文件操作标记的当前位置,以文件的开头为原点
 300     fp.next()                                # 返回下一行,并将文件操作标记位移到下一行。把一个file用于for … in file这样的语句时,就是调用next()函数来实现遍历的。
 301     fp.seek(offset[,whence])                # 将文件打操作标记移到offset的位置。whence可以为0表示从头开始计算,1表示以当前位置为原点计算。2表示以文件末尾为原点进行计算。
 302     fp.seekable()                                                    # 是否可以seek
 303     fp.truncate([size])                      # 把文件裁成规定的大小,默认的是裁到当前文件操作标记的位置。
 304     for line in open('data'): 
 305         print(line)                                                    # 使用for语句,比较适用于打开比较大的文件
 306     open('f.txt', encoding = 'latin-1')        # Python3.x Unicode文本文件
 307     open('f.bin', 'rb')                                        # Python3.x 二进制bytes文件
 308     # 文件对象还有相应的属性:buffer closed encoding errors line_buffering name newlines等
 309     
 310 #-- 其他
 311     # Python中的真假值含义:1. 数字如果非零,则为真,0为假。 2. 其他对象如果非空,则为真
 312     # 通常意义下的类型分类:1. 数字、序列、映射。 2. 可变类型和不可变类型
 313 
 314 
 315 '''语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句----语法和语句'''
 316 
 317 #--    赋值语句的形式
 318     spam = 'spam'                                                    # 基本形式
 319     spam, ham = 'spam', 'ham'                            # 元组赋值形式
 320     [spam, ham] = ['s', 'h']                            # 列表赋值形式
 321     a, b, c, d = 'abcd'                                        # 序列赋值形式
 322     a, *b, c = 'spam'                                            # 序列解包形式(Python3.x中才有)
 323     spam = ham = 'no'                                            # 多目标赋值运算,涉及到共享引用
 324     spam += 42                                                        # 增强赋值,涉及到共享引用
 325 
 326 #-- 序列赋值 序列解包
 327     [a, b, c] = (1, 2, 3)                                    # a = 1, b = 2, c = 3
 328     a, b, c, d = "spam"                                        # a = 's', b = 'p'
 329     a, b, c = range(3)                                        # a = 0, b = 1
 330     a, *b = [1, 2, 3, 4]                                    # a = 1, b = [2, 3, 4]
 331     *a, b = [1, 2, 3, 4]                                    # a = [1, 2, 3], b = 4
 332     a, *b, c = [1, 2, 3, 4]                                # a = 1, b = [2, 3], c = 4
 333     # 带有*时 会优先匹配*之外的变量 如
 334     a, *b, c = [1, 2]                                            # a = 1, c = 2, b = []
 335 
 336 #--    print函数原型
 337     print(value, ..., sep=' ', end='\n', file=sys.stdout, flush=False)
 338     # 流的重定向
 339     print('hello world')                                # 等于sys.stdout.write('hello world')
 340     temp = sys.stdout                                        # 原有流的保存
 341     sys.stdout = open('log.log', 'a')        # 流的重定向
 342     print('hello world')                                # 写入到文件log.log
 343     sys.stdout.close()
 344     sys.stdout = temp                                        # 原有流的复原
 345     
 346 #--    Python中and或or总是返回对象(左边的对象或右边的对象) 且具有短路求值的特性
 347     1 or 2 or 3                                                    # 返回 1
 348     1 and 2 and 3                                                # 返回 3
 349 
 350 #--    if/else三元表达符(if语句在行内)
 351     A = 1 if X else 2
 352     A = 1 if X else (2 if Y else 3)
 353     # 也可以使用and-or语句(一条语句实现多个if-else)
 354     result = (a > 20 and "big than 20" or a > 10 and "big than 10" or a > 5 and "big than 5")
 355 
 356 #--    Python的while语句或者for语句可以带else语句 当然也可以带continue/break/pass语句
 357     while a > 1:
 358         ......
 359     else:
 360         ......
 361     # else语句会在循环结束后执行,除非在循环中执行了break,同样的还有for语句
 362     for i in range(5):
 363         ......
 364     else:
 365         ......
 366 
 367 #--    for循环的元组赋值
 368     for (a, b) in [(1, 2), (3, 4)]:                                    # 最简单的赋值
 369     for ((a, b), c) in [((1, 2), 3), ((4, 5), 6)]:    # 自动解包赋值
 370     for ((a, b), c) in [((1, 2), 3), ("XY", 6)]:        # 自动解包 a = X, b = Y, c = 6
 371     for (a, *b) in [(1, 2, 3), (4, 5, 6)]:                    # 自动解包赋值
 372 
 373 #--    列表解析语法
 374     M = [[1,2,3], [4,5,6], [7,8,9]]
 375     res = [sum(row) for row in M]                                        # G = [6, 15, 24] 一般的列表解析 生成一个列表
 376     res = [c * 2 for c in 'spam']                                        # ['ss', 'pp', 'aa', 'mm']
 377     res = [a * b for a in [1, 2] for b in [4, 5]]        # 多解析过程 返回[4, 5, 8, 10]
 378     res = [a for a in [1, 2, 3] if a < 2]                        # 带判断条件的解析过程
 379     res = [a if a > 0 else 0 for a in [-1, 0, 1]]     # 带判断条件的高级解析过程
 380     # 两个列表同时解析:使用zip函数
 381     for teama, teamb in zip(["Packers", "49ers"], ["Ravens", "Patriots"]):
 382         print(teama + " vs. " + teamb)
 383     # 带索引的列表解析:使用enumerate函数
 384     for index, team in enumerate(["Packers", "49ers", "Ravens", "Patriots"]):
 385         print(index, team)                                                        # 输出0, Packers \n 1, 49ers \n ......
 386     
 387 #-- 生成器表达式
 388     G = (sum(row) for row in M)                                         # 使用小括号可以创建所需结果的生成器generator object
 389     next(G), next(G), next(G)                                           # 输出(6, 15, 24)
 390     G = {sum(row) for row in M}                                         # G = {6, 15, 24} 解析语法还可以生成集合和字典
 391     G = {i:sum(M[i]) for i in range(3)}                            # G = {0: 6, 1: 15, 2: 24}
 392 
 393 #--    文档字符串:出现在Module的开端以及其中函数或类的开端 使用三重引号字符串
 394     '''
 395     module document
 396     '''
 397     def func():
 398         '''
 399         function document
 400         '''
 401         print()
 402     class Employee:
 403         '''
 404         class document
 405         '''
 406         ...
 407     print(func.__doc__)                                        # 输出函数文档字符串
 408     print(Employee.__doc__)                                # 输出类的文档字符串
 409     
 410 #--    命名惯例:
 411     '''
 412     以单一下划线开头的变量名(_X)不会被from module import*等语句导入
 413     前后有两个下划线的变量名(__X__)是系统定义的变量名,对解释器有特殊意义
 414     以两个下划线开头但不以下划线结尾的变量名(__X)是类的本地(私有)变量
 415     '''
 416 
 417 #-- 列表解析 in成员关系测试 map sorted zip enumerate内置函数等都使用了迭代协议
 418     'first line' in open('test.txt')            # in测试 返回True或False
 419     list(map(str.upper, open('t')))                # map内置函数
 420     sorted(iter([2, 5, 8, 3, 1]))                    # sorted内置函数
 421     list(zip([1, 2], [3, 4]))                            # zip内置函数 [(1, 3), (2, 4)] 
 422 
 423 #-- del语句: 手动删除某个变量
 424     del X
 425 
 426 #-- 获取列表的子表的方法:
 427     x = [1,2,3,4,5,6]
 428     x[:3]                                                                    # 前3个[1,2,3]
 429     x[1:5]                                                                # 中间4个[2,3,4,5]
 430     x[-3:]                                                                # 最后3个[4,5,6]
 431     x[::2]                                                                # 奇数项[1,3,5]
 432     x[1::2]                                                                # 偶数项[2,4,6]
 433     
 434 #-- 手动迭代:iter和next
 435     L = [1, 2]
 436     I = iter(L)                                                        # I为L的迭代器
 437     I.next()                                                            #    返回1
 438     I.next()                                                            # 返回2
 439     I.next()                                                            # Error:StopIteration
 440     
 441 #-- Python中的可迭代对象
 442     '''
 443     1.range迭代器
 444     2.map、zip和filter迭代器
 445     3.字典视图迭代器:D.keys()), D.items()等
 446     4.文件类型
 447     '''
 448 
 449 
 450 '''函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则----函数语法规则'''
 451 
 452 #-- 函数相关的语句和表达式
 453     myfunc('spam')                            # 函数调用
 454     def myfunc():                                # 函数定义
 455     return None                                    # 函数返回值
 456     global a                                        # 全局变量
 457     nonlocal x                                    # 在函数或其他作用域中使用外层(非全局)变量
 458     yield x                                            # 生成器函数返回
 459     lambda                                            # 匿名函数
 460     
 461 #--    Python函数变量名解析:LEGB原则,即:
 462     '''
 463     local(functin) --> encloseing function locals --> global(module) --> build-in(python)
 464     说明:以下边的函数maker为例 则相对于action而言 X为Local N为Encloseing
 465     '''
 466 
 467 #--    嵌套函数举例:工厂函数
 468     def maker(N):
 469         def action(X):
 470             return X ** N
 471         return action
 472     f = maker(2)                                # pass 2 to N
 473     f(3)                                                # 9, pass 3 to X
 474 
 475 #--    嵌套函数举例:lambda实例
 476     def maker(N):
 477         action = (lambda X: X**N)
 478         return action
 479     f = maker(2)                                # pass 2 to N
 480     f(3)                                                # 9, pass 3 to X
 481 
 482 #--    nonlocal和global语句的区别
 483     # nonlocal应用于一个嵌套的函数的作用域中的一个名称 例如:
 484     start = 100
 485     def tester(start):
 486         def nested(label):
 487             nonlocal start                    # 指定start为tester函数内的local变量 而不是global变量start
 488             print(label, start)
 489             start += 3
 490         return nested
 491     # global为全局的变量 即def之外的变量
 492     def tester(start):
 493         def nested(label):
 494             global start                        # 指定start为global变量start
 495             print(label, start)
 496             start += 3
 497         return nested    
 498     
 499 #--    函数参数,不可变参数通过“值”传递,可变参数通过“引用”传递
 500     def f(a, b, c): print(a, b, c)
 501     f(1, 2, 3)                                                        # 参数位置匹配
 502     f(1, c = 3, b = 2)                                        # 参数关键字匹配
 503     def f(a, b = 1, c = 2): print(a, b, c)
 504     f(1)                                                                    # 默认参数匹配
 505     f(1, 2)                                                                # 默认参数匹配
 506     f(a = 1, c = 3)                                                # 关键字参数和默认参数的混合
 507     # Keyword-Only参数:出现在*args之后 必须用关键字进行匹配
 508     def keyOnly(a, *b, c): print('')            # c就为keyword-only匹配 必须使用关键字c = value匹配
 509     def keyOnly(a, *, b, c): ......                # b c为keyword-only匹配 必须使用关键字匹配
 510     def keyOnly(a, *, b = 1): ......            # b有默认值 或者省略 或者使用关键字参数b = value
 511 
 512 #--    可变参数匹配: * 和 **
 513     def f(*args): print(args)                            # 在元组中收集不匹配的位置参数
 514     f(1, 2, 3)                                                        # 输出(1, 2, 3)
 515     def f(**args): print(args)                        # 在字典中收集不匹配的关键字参数
 516     f(a = 1, b = 2)                                                # 输出{'a':1, 'b':2}
 517     def f(a, *b **c): print(a, b, c)            # 两者混合使用
 518     f(1, 2, 3, x = 4, y = 5)                            # 输出1, (2, 3), {'x':4, 'y':5}
 519     
 520 #--    函数调用时的参数解包: * 和 ** 分别解包元组和字典
 521     func(1, *(2, 3))  <==>  func(1, 2, 3)
 522     func(1, **{'c':3, 'b':2})  <==>  func(1, b = 2, c = 3)
 523     func(1, *(2, 3), **{'c':3, 'b':2})  <==>  func(1, 2, 3, b = 2, c = 3)
 524     
 525 #-- 函数属性:(自己定义的)函数可以添加属性
 526     def func():.....
 527     func.count = 1                                                # 自定义函数添加属性
 528     print.count = 1                                                # Error 内置函数不可以添加属性
 529     
 530 #-- 函数注解: 编写在def头部行 主要用于说明参数范围、参数类型、返回值类型等
 531     def func(a:'spam', b:(1, 10), c:float) -> int :
 532         print(a, b, c)
 533     func.__annotations__                                    # {'c':<class 'float'>, 'b':(1, 10), 'a':'spam', 'return':<class 'int'>}
 534     # 编写注解的同时 还是可以使用函数默认值 并且注解的位置位于=号的前边
 535     def func(a:'spam'='a', b:(1, 10)=2, c:float=3) -> int :
 536         print(a, b, c)
 537 
 538 #-- 匿名函数:lambda
 539     f = lambda x, y, z : x + y + z                # 普通匿名函数,使用方法f(1, 2, 3)
 540     f = lambda x = 1, y = 1: x + y                # 带默认参数的lambda函数
 541     def action(x):                                                # 嵌套lambda函数
 542         return (lambda y : x + y)
 543     f = lambda: a if xxx() else b                    # 无参数的lambda函数,使用方法f()
 544 
 545 #-- lambda函数与map filter reduce函数的结合
 546     list(map((lambda x: x + 1), [1, 2, 3]))                            # [2, 3, 4]
 547     list(filter((lambda x: x > 0), range(-4, 5)))                # [1, 2, 3, 4]
 548     functools.reduce((lambda x, y: x + y), [1, 2, 3])        # 6
 549     functools.reduce((lambda x, y: x * y), [2, 3, 4])        # 24
 550     
 551 #-- 生成器函数:yield VS return
 552     def gensquare(N):
 553         for i in range(N):
 554             yield i** 2                                                # 状态挂起 可以恢复到此时的状态
 555     for i in gensquare(5):                                # 使用方法
 556         print(i, end = ' ')                                    # [0, 1, 4, 9, 16]
 557     x = gensquare(2)                                            # x是一个生成对象
 558     next(x)                                                                # 等同于x.__next__() 返回0
 559     next(x)                                                                # 等同于x.__next__() 返回1
 560     next(x)                                                                # 等同于x.__next__() 抛出异常StopIteration
 561     
 562 #-- 生成器表达式:小括号进行列表解析
 563     G = (x ** 2 for x in range(3))                 # 使用小括号可以创建所需结果的生成器generator object
 564     next(G), next(G), next(G)                            # 和上述中的生成器函数的返回值一致
 565     #(1)生成器(生成器函数/生成器表达式)是单个迭代对象
 566     G = (x ** 2 for x in range(4))
 567     I1 = iter(G)                                                    # 这里实际上iter(G) = G
 568     next(I1)                                                            # 输出0
 569     next(G)                                                                # 输出1
 570     next(I1)                                                            # 输出4
 571     #(2)生成器不保留迭代后的结果
 572     gen = (i for i in range(4))
 573     2 in gen                                                            # 返回True
 574     3 in gen                                                            # 返回True
 575     1 in gen                                                             # 返回False,其实检测2的时候,1已经就不在生成器中了,即1已经被迭代过了,同理2、3也不在了
 576 
 577 #-- 本地变量是静态检测的
 578     X = 22                                            # 全局变量X的声明和定义
 579     def test():
 580         print(X)                                    # 如果没有下一语句 则该句合法 打印全局变量X
 581         X = 88                                        # 这一语句使得上一语句非法 因为它使得X变成了本地变量 上一句变成了打印一个未定义的本地变量(局部变量)
 582         if False:                                    # 即使这样的语句 也会把print语句视为非法语句 因为:
 583             X = 88                                    # Python会无视if语句而仍然声明了局部变量X
 584     def test():                                    # 改进
 585         global X                                    # 声明变量X为全局变量
 586         print(X)                                    # 打印全局变量X
 587         X = 88                                        # 改变全局变量X
 588         
 589 #-- 函数的默认值是在函数定义的时候实例化的 而不是在调用的时候 例子:
 590     def foo(numbers=[]):                # 这里的[]是可变的
 591         numbers.append(9)    
 592         print(numbers)
 593     foo()                                             # first time, like before, [9]
 594     foo()                                             # second time, not like before, [9, 9]
 595     foo()                                             # third time, not like before too, [9, 9, 9]
 596     # 改进:
 597     def foo(numbers=None):
 598         if numbers is None: numbers = []
 599         numbers.append(9)
 600         print(numbers)
 601     # 另外一个例子 参数的默认值为不可变的:
 602     def foo(count=0):                        # 这里的0是数字, 是不可变的
 603         count += 1
 604         print(count)
 605     foo()                                                # 输出1
 606     foo()                                                # 还是输出1
 607     foo(3)                                            # 输出4
 608     foo()                                                # 还是输出1
 609     
 610 
 611 '''函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子----函数例子'''
 612 
 613     '''数学运算类'''
 614     abs(x)                                                                # 求绝对值,参数可以是整型,也可以是复数,若参数是复数,则返回复数的模
 615     complex([real[, imag]])                                # 创建一个复数
 616     divmod(a, b)                                                    # 分别取商和余数,注意:整型、浮点型都可以
 617     float([x])                                                        # 将一个字符串或数转换为浮点数。如果无参数将返回0.0
 618     int([x[, base]])                                             # 将一个字符串或浮点数转换为int类型,base表示进制
 619     long([x[, base]])                                         # 将一个字符串或浮点数转换为long类型
 620     pow(x, y)                                                         # 返回x的y次幂
 621     range([start], stop[, step])                     # 产生一个序列,默认从0开始
 622     round(x[, n])                                                 # 四舍五入
 623     sum(iterable[, start])                                 # 对集合求和
 624     oct(x)                                                                # 将一个数字转化为8进制字符串
 625     hex(x)                                                                # 将一个数字转换为16进制字符串
 626     chr(i)                                                                # 返回给定int类型对应的ASCII字符
 627     unichr(i)                                                            # 返回给定int类型的unicode
 628     ord(c)                                                                # 返回ASCII字符对应的整数
 629     bin(x)                                                                # 将整数x转换为二进制字符串
 630     bool([x])                                                            # 将x转换为Boolean类型
 631     
 632     '''集合类操作'''
 633     basestring()                                                    # str和unicode的超类,不能直接调用,可以用作isinstance判断
 634     format(value [, format_spec])                    # 格式化输出字符串,格式化的参数顺序从0开始,如“I am {0},I like {1}”
 635     enumerate(sequence[, start=0])                # 返回一个可枚举的对象,注意它有第二个参数
 636     iter(obj[, sentinel])                                    # 生成一个对象的迭代器,第二个参数表示分隔符
 637     max(iterable[, args...][key])                 # 返回集合中的最大值
 638     min(iterable[, args...][key])                    # 返回集合中的最小值
 639     dict([arg])                                                        # 创建数据字典
 640     list([iterable])                                             # 将一个集合类转换为另外一个集合类
 641     set()                                                                    # set对象实例化
 642     frozenset([iterable])                                    # 产生一个不可变的set
 643     tuple([iterable])                                         # 生成一个tuple类型
 644     str([object])                                                 # 转换为string类型
 645     sorted(iterable[, cmp[, key[, reverse]]])             # 集合排序
 646         L = [('b',2),('a',1),('c',3),('d',4)]
 647         sorted(L, key=lambda x: x[1]), reverse=True)    # 使用Key参数和reverse参数
 648         sorted(L, key=lambda x: (x[0], x[1]))                    # 使用key参数进行多条件排序,即如果x[0]相同,则比较x[1]
 649 
 650     '''逻辑判断'''
 651     all(iterable)                                                    # 集合中的元素都为真的时候为真,特别的,若为空串返回为True
 652     any(iterable)                                                    # 集合中的元素有一个为真的时候为真,特别的,若为空串返回为False
 653     cmp(x, y)                                                            # 如果x < y ,返回负数;x == y, 返回0;x > y,返回正数
 654 
 655     '''IO操作'''
 656     file(filename [, mode [, bufsize]])        # file类型的构造函数。
 657     input([prompt])                                             # 获取用户输入,推荐使用raw_input,因为该函数将不会捕获用户的错误输入
 658     raw_input([prompt])                                     # 设置输入,输入都是作为字符串处理
 659     open(name[, mode[, buffering]])             # 打开文件,与file有什么不同?推荐使用open
 660     
 661     '''其他'''
 662     callable(object)                                            # 检查对象object是否可调用
 663     classmethod(func)                                            # 用来说明这个func是个类方法
 664     staticmethod(func)                                        # 用来说明这个func为静态方法
 665     dir([object])                                                    # 不带参数时,返回当前范围内的变量、方法和定义的类型列表;带参数时,返回参数的属性、方法列表。
 666     help(obj)                                                            # 返回obj的帮助信息
 667     eval(expression)                                            # 计算表达式expression的值,并返回
 668     exec(str)                                                            # 将str作为Python语句执行
 669     execfile(filename)                                        # 用法类似exec(),不同的是execfile的参数filename为文件名,而exec的参数为字符串。
 670     filter(function, iterable)                        # 构造一个序列,等价于[item for item in iterable if function(item)],function返回值为True或False的函数
 671         list(filter(bool, range(-3, 4)))        # 返回[-3, -2, -1, 1, 2, 3], 没有0
 672     hasattr(object, name)                                    # 判断对象object是否包含名为name的特性
 673     getattr(object, name [, defalut])            # 获取一个类的属性
 674     setattr(object, name, value)                    # 设置属性值
 675     delattr(object, name)                                    # 删除object对象名为name的属性
 676     globals()                                                            # 返回一个描述当前全局符号表的字典
 677     hash(object)                                                    # 如果对象object为哈希表类型,返回对象object的哈希值
 678     id(object)                                                        # 返回对象的唯一标识,一串数字
 679     isinstance(object, classinfo)                    # 判断object是否是class的实例
 680         isinstance(1, int)                                    # 判断是不是int类型
 681         isinstance(1, (int, float))                    # isinstance的第二个参数接受一个元组类型
 682     issubclass(class, classinfo)                    # 判断class是否为classinfo的子类
 683     locals()                                                             # 返回当前的变量列表
 684     map(function, iterable, ...)                     # 遍历每个元素,执行function操作
 685         list(map(abs, range(-3, 4)))                # 返回[3, 2, 1, 0, 1, 2, 3]
 686     next(iterator[, default])                         # 类似于iterator.next()
 687     property([fget[, fset[, fdel[, doc]]]])             # 属性访问的包装类,设置后可以通过c.x=value等来访问setter和getter
 688     reduce(function, iterable[, initializer])         # 合并操作,从第一个开始是前两个参数,然后是前两个的结果与第三个合并进行处理,以此类推
 689         def add(x,y):return x + y 
 690         reduce(add, range(1, 11))                     # 返回55 (注:1+2+3+4+5+6+7+8+9+10 = 55)
 691         reduce(add, range(1, 11), 20)             # 返回75 
 692     reload(module)                                                 # 重新加载模块
 693     repr(object)                                                     # 将一个对象变幻为可打印的格式
 694     slice(start, stop[, step])                         # 产生分片对象
 695     type(object)                                                    # 返回该object的类型
 696     vars([object])                                                 # 返回对象的变量名、变量值得字典
 697         a = Class();                                                # Class为一个空类
 698         a.name = 'qi', a.age = 9
 699         vars(a)                                                            # {'name':'qi', 'age':9}
 700     zip([iterable, ...])                                    # 返回对应数组
 701         list(zip([1, 2, 3], [4, 5, 6]))            # [(1, 4), (2, 5), (3, 6)]
 702         a = [1, 2, 3],  b = ["a", "b", "c"]
 703         z = zip(a, b)                                                # 压缩:[(1, "a"), (2, "b"), (3, "c")]
 704         zip(*z)                                                            # 解压缩:[(1, 2, 3), ("a", "b", "c")]
 705     unicode(string, encoding, errors)            # 将字符串string转化为unicode形式,string为encoded string。
 706 
 707     
 708 '''模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle----模块Moudle'''
 709 
 710 #-- Python模块搜索路径:
 711     '''
 712     (1)程序的主目录    (2)PYTHONPATH目录 (3)标准链接库目录 (4)任何.pth文件的内容
 713     '''
 714     
 715 #--    查看全部的模块搜索路径
 716     import sys
 717     sys.path
 718         
 719 #--    模块的使用代码
 720     import module1, module2                                # 导入module1 使用module1.printer()
 721     from module1 import printer                        # 导入module1中的printer变量 使用printer()
 722     from module1 imoprt *                                    # 导入module1中的全部变量 使用不必添加module1前缀
 723 
 724 #-- 重载模块reload: 这是一个内置函数 而不是一条语句
 725     from imp import reload
 726     reload(module)
 727     
 728 #--    模块的包导入:使用点号(.)而不是路径(dir1\dir2)进行导入
 729     import dir1.dir2.mod                                    # d导入包(目录)dir1中的包dir2中的mod模块 此时dir1必须在Python可搜索路径中
 730     from dir1.dir2.mod import *                     # from语法的包导入
 731 
 732 #--    __init__.py包文件:每个导入的包中都应该包含这么一个文件
 733     '''
 734     该文件可以为空
 735     首次进行包导入时 该文件会自动执行
 736     高级功能:在该文件中使用__all__列表来定义包(目录)以from*的形式导入时 需要导入什么
 737     '''
 738     
 739 #--    包相对导入:使用点号(.) 只能使用from语句
 740     from . import spam                                        # 导入当前目录下的spam模块(错误: 当前目录下的模块, 直接导入即可)
 741     from .spam import name                                # 导入当前目录下的spam模块的name属性(错误: 当前目录下的模块, 直接导入即可,不用加.)
 742     from .. import spam                                        # 导入当前目录的父目录下的spam模块
 743     
 744 #--    包相对导入与普通导入的区别
 745     from string import *                                    # 这里导入的string模块为sys.path路径上的 而不是本目录下的string模块(如果存在也不是)
 746     from .string import *                                 # 这里导入的string模块为本目录下的(不存在则导入失败) 而不是sys.path路径上的
 747     
 748 #--    模块数据隐藏:最小化from*的破坏
 749     _X                                                                        # 变量名前加下划线可以防止from*导入时该变量名被复制出去
 750     __all__ = ['x', 'x1', 'x2']                        # 使用__all__列表指定from*时复制出去的变量名(变量名在列表中为字符串形式)
 751 
 752 #-- 可以使用__name__进行模块的单元测试:当模块为顶层执行文件时值为'__main__' 当模块被导入时为模块名
 753     if __name__ == '__main__':
 754         ......
 755     # 模块属性中还有其他属性,例如:
 756     __doc__                                            # 模块的说明文档
 757     __file__                                        # 模块文件的文件名,包括全路径
 758     __name__                                        # 主文件或者被导入文件
 759     __package__                                    # 模块所在的包
 760         
 761 #-- import语句from语句的as扩展
 762     import modulename as name
 763     from modulename import attrname as name
 764     
 765 #-- 得到模块属性的几种方法 假设为了得到name属性的值
 766     M.name
 767     M.__dict__['name']
 768     sys.modules['M'].name
 769     getattr(M, 'name')
 770     
 771 
 772 '''类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象----类与面向对象'''
 773 
 774 #--    最普通的类
 775     class C1(C2, C3):
 776         spam = 42                                                        # 数据属性
 777         def __init__(self, name):                        # 函数属性:构造函数
 778             self.name = name
 779         def __del__(self):                                    # 函数属性:析构函数
 780             print("goodbey ", self.name)    
 781     I1 = C1('bob')
 782     
 783 #--    Python的类没有基于参数的函数重载
 784     class FirstClass:
 785         def test(self, string):
 786             print(string)
 787         def test(self):                                            # 此时类中只有一个test函数 即后者test(self) 它覆盖掉前者带参数的test函数
 788             print("hello world")
 789 
 790 #--    子类扩展超类: 尽量调用超类的方法
 791     class Manager(Person):
 792         def giveRaise(self, percent, bonus = .10):
 793             self.pay = int(self.pay*(1 + percent + bonus))        # 不好的方式 复制粘贴超类代码
 794             Person.giveRaise(self, percent + bonus)                        # 好的方式 尽量调用超类方法
 795 
 796 #-- 类内省工具
 797     bob = Person('bob')
 798     bob.__class__                                                    # <class 'Person'>
 799     bob.__class__.__name__                                # 'Person'
 800     bob.__dict__                                                    # {'pay':0, 'name':'bob', 'job':'Manager'}
 801     
 802 #-- 返回1中 数据属性spam是属于类 而不是对象
 803     I1 = C1('bob'); I2 = C2('tom');                # 此时I1和I2的spam都为42 但是都是返回的C1的spam属性
 804     C1.spam = 24                                                    # 此时I1和I2的spam都为24
 805     I1.spam = 3                                                        # 此时I1新增自有属性spam 值为2 I2和C1的spam还都为24
 806     
 807 #--    类方法调用的两种方式
 808     instance.method(arg...)
 809     class.method(instance, arg...)
 810     
 811 #--    抽象超类的实现方法
 812     # (1)某个函数中调用未定义的函数 子类中定义该函数
 813         def delegate(self):
 814             self.action()                                            # 本类中不定义action函数 所以使用delegate函数时就会出错
 815     # (2)定义action函数 但是返回异常
 816         def action(self):
 817             raise NotImplementedError("action must be defined")
 818     # (3)上述的两种方法还都可以定义实例对象 实际上可以利用@装饰器语法生成不能定义的抽象超类
 819         from abc import ABCMeta, abstractmethod
 820         class Super(metaclass = ABCMeta):
 821             @abstractmethod
 822             def action(self): pass
 823         x = Super()                                                    # 返回 TypeError: Can't instantiate abstract class Super with abstract methods action
 824     
 825 #--    # OOP和继承: "is - a"的关系
 826     class A(B):
 827         pass
 828     a = A()
 829     isinstance(a, B)                                            # 返回True, A是B的子类 a也是B的一种
 830     # OOP和组合: "has- a"的关系
 831     pass
 832     # OOP和委托: "包装"对象 在Python中委托通常是以"__getattr__"钩子方法实现的, 这个方法会拦截对不存在属性的读取
 833     # 包装类(或者称为代理类)可以使用__getattr__把任意读取转发给被包装的对象
 834     class wrapper:
 835         def __init__(self, object):
 836             self.wrapped = object
 837         def __getattr(self, attrname):
 838             print('Trace: ', attrname)
 839             return getattr(self.wrapped, attrname)
 840     # 注:这里使用getattr(X, N)内置函数以变量名字符串N从包装对象X中取出属性 类似于X.__dict__[N]
 841     x = wrapper([1, 2, 3])
 842     x.append(4)                                                        # 返回 "Trace: append" [1, 2, 3, 4]
 843     x = wrapper({'a':1, 'b':2})
 844     list(x.keys())                                                # 返回 "Trace: keys" ['a', 'b']
 845 
 846 #-- 类的伪私有属性:使用__attr
 847     class C1:
 848         def __init__(self, name):
 849             self.__name = name                                # 此时类的__name属性为伪私有属性 原理 它会自动变成self._C1__name = name
 850         def __str__(self):
 851             return 'self.name = %s' % self.__name
 852     I = C1('tom')
 853     print(I)                                                            # 返回 self.name = tom
 854     I.__name = 'jeey'                                            # 这里无法访问 __name为伪私有属性
 855     I._C1__name = 'jeey'                                    # 这里可以修改成功 self.name = jeey
 856     
 857 #-- 类方法是对象:无绑定类方法对象 / 绑定实例方法对象
 858     class Spam:
 859         def doit(self, message):
 860             print(message)
 861         def selfless(message)
 862             print(message)
 863     obj = Spam()
 864     x = obj.doit                                                    # 类的绑定方法对象 实例 + 函数
 865     x('hello world')
 866     x = Spam.doit                                                    # 类的无绑定方法对象 类名 + 函数
 867     x(obj, 'hello world')
 868     x = Spam.selfless                                            # 类的无绑定方法是函数 在3.0之前无效
 869     x('hello world')
 870 
 871 #-- 获取对象信息: 属性和方法
 872     a = MyObject()
 873     dir(a)                                                                # 使用dir函数
 874     hasattr(a, 'x')                                                # 测试是否有x属性或方法 即a.x是否已经存在
 875     setattr(a, 'y', 19)                                        # 设置属性或方法 等同于a.y = 19
 876     getattr(a, 'z', 0)                                        # 获取属性或方法 如果属性不存在 则返回默认值0
 877     #这里有个小技巧,setattr可以设置一个不能访问到的属性,即只能用getattr获取
 878     setattr(a, "can't touch", 100)                # 这里的属性名带有空格,不能直接访问
 879     getattr(a, "can't touch", 0)                    # 但是可以用getattr获取
 880 
 881 #-- 为类动态绑定属性或方法: MethodType方法
 882     # 一般创建了一个class的实例后, 可以给该实例绑定任何属性和方法, 这就是动态语言的灵活性
 883     class Student(object):
 884         pass
 885     s = Student()
 886     s.name = 'Michael'                                                             # 动态给实例绑定一个属性
 887     def set_age(self, age):                                                 # 定义一个函数作为实例方法
 888         self.age = age
 889     from types import MethodType
 890     s.set_age = MethodType(set_age, s)                             # 给实例绑定一个方法 类的其他实例不受此影响
 891     s.set_age(25)                                                                     # 调用实例方法
 892     Student.set_age = MethodType(set_age, Student)    # 为类绑定一个方法 类的所有实例都拥有该方法
 893 
 894     
 895 '''类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题----类的高级话题'''
 896     
 897 #--    多重继承: "混合类", 搜索方式"从下到上 从左到右 广度优先"
 898     class A(B, C):
 899         pass
 900 
 901 #-- 类的继承和子类的初始化
 902     # 1.子类定义了__init__方法时,若未显示调用基类__init__方法,python不会帮你调用。
 903     # 2.子类未定义__init__方法时,python会自动帮你调用首个基类的__init__方法,注意是首个。
 904     # 3.子类显示调用基类的初始化函数:
 905     class FooParent(object):
 906         def __init__(self, a):
 907             self.parent = 'I\'m the Parent.'
 908             print('Parent:a=' + str(a))
 909         def bar(self, message):
 910             print(message + ' from Parent')
 911     class FooChild(FooParent):
 912         def __init__(self, a):
 913             FooParent.__init__(self, a)
 914             print('Child:a=' + str(a))
 915         def bar(self, message):
 916             FooParent.bar(self, message)
 917             print(message + ' from Child')
 918     fooChild = FooChild(10)
 919     fooChild.bar('HelloWorld')
 920     
 921 #--    #实例方法 / 静态方法 / 类方法
 922     class Methods:
 923         def imeth(self, x): print(self, x)    # 实例方法:传入的是实例和数据,操作的是实例的属性
 924         def smeth(x): print(x)                            # 静态方法:只传入数据 不传入实例,操作的是类的属性而不是实例的属性
 925         def cmeth(cls, x): print(cls, x)        # 类方法:传入的是类对象和数据
 926         smeth = staticmethod(smeth)                    # 调用内置函数,也可以使用@staticmethod
 927         cmeth = classmethod(cmeth)                    # 调用内置函数,也可以使用@classmethod
 928     obj = Methods()
 929     obj.imeth(1)                                                    # 实例方法调用 <__main__.Methods object...> 1
 930     Methods.imeth(obj, 2)                                    # <__main__.Methods object...> 2
 931     Methods.smeth(3)                                            # 静态方法调用 3
 932     obj.smeth(4)                                                    # 这里可以使用实例进行调用
 933     Methods.cmeth(5)                                            # 类方法调用 <class '__main__.Methods'> 5
 934     obj.cmeth(6)                                                    # <class '__main__.Methods'> 6
 935     
 936 #--    函数装饰器:是它后边的函数的运行时的声明 由@符号以及后边紧跟的"元函数"(metafunction)组成
 937         @staticmethod
 938         def smeth(x): print(x)
 939     # 等同于:
 940         def smeth(x): print(x)
 941         smeth = staticmethod(smeth)
 942     # 同理
 943         @classmethod
 944         def cmeth(cls, x): print(x)
 945     # 等同于
 946         def cmeth(cls, x): print(x)
 947         cmeth = classmethod(cmeth)
 948     
 949 #--    类修饰器:是它后边的类的运行时的声明 由@符号以及后边紧跟的"元函数"(metafunction)组成
 950         def decorator(aClass):.....
 951         @decorator
 952         class C:....
 953     # 等同于:
 954         class C:....
 955         C = decorator(C)
 956 
 957 #--    限制class属性: __slots__属性
 958     class Student:
 959         __slots__ = ('name', 'age')        #限制Student及其实例只能拥有name和age属性
 960     # __slots__属性只对当前类起作用, 对其子类不起作用
 961     # __slots__属性能够节省内存
 962     # __slots__属性可以为列表list,或者元组tuple
 963     
 964 #--    类属性高级话题: @property
 965     # 假设定义了一个类:C,该类必须继承自object类,有一私有变量_x
 966     class C:
 967          def __init__(self):
 968           self.__x = None
 969     # 第一种使用属性的方法
 970          def getx(self):
 971           return self.__x
 972          def setx(self, value):
 973           self.__x = value
 974          def delx(self):
 975           del self.__x
 976          x = property(getx, setx, delx, '')
 977     # property函数原型为property(fget=None,fset=None,fdel=None,doc=None)
 978     # 使用
 979     c = C()
 980     c.x = 100                                        # 自动调用setx方法
 981     y = c.x                                            # 自动调用getx方法
 982     del c.x                                            # 自动调用delx方法
 983    # 第二种方法使用属性的方法
 984         @property
 985          def x(self):
 986           return self.__x
 987          @x.setter
 988          def x(self, value):
 989           self.__x = value
 990          @x.deleter
 991          def x(self):
 992           del self.__x
 993     # 使用
 994     c = C()
 995     c.x = 100                                        # 自动调用setter方法
 996     y = c.x                                            # 自动调用x方法
 997     del c.x                                            # 自动调用deleter方法
 998     
 999 #-- 定制类: 重写类的方法
1000     # (1)__str__方法、__repr__方法: 定制类的输出字符串
1001     # (2)__iter__方法、next方法: 定制类的可迭代性
1002     class Fib(object):
1003         def __init__(self):
1004             self.a, self.b = 0, 1                         # 初始化两个计数器a,b
1005         def __iter__(self):
1006             return self                                             # 实例本身就是迭代对象,故返回自己
1007         def next(self):
1008             self.a, self.b = self.b, self.a + self.b
1009             if self.a > 100000:                             # 退出循环的条件
1010                 raise StopIteration();
1011             return self.a                                         # 返回下一个值
1012     for n in Fib(): 
1013         print(n)                                                        # 使用
1014     # (3)__getitem__方法、__setitem__方法: 定制类的下标操作[] 或者切片操作slice
1015     class Indexer(object):
1016         def __init__(self):
1017             self.data = {}
1018         def __getitem__(self, n):                        # 定义getitem方法
1019             print('getitem:', n)                
1020             return self.data[n]
1021         def __setitem__(self, key, value):    # 定义setitem方法
1022             print('setitem:key = {0}, value = {1}'.format(key, value))
1023             self.data[key] = value
1024     test = Indexer()
1025     test[0] = 1;   test[3] = '3'                    # 调用setitem方法
1026     print(test[0])                                                # 调用getitem方法
1027     # (4)__getattr__方法: 定制类的属性操作
1028     class Student(object):
1029         def __getattr__(self, attr):                # 定义当获取类的属性时的返回值
1030             if attr=='age':
1031                 return 25                                                # 当获取age属性时返回25
1032         raise AttributeError('object has no attribute: %s' % attr)
1033         # 注意: 只有当属性不存在时 才会调用该方法 且该方法默认返回None 需要在函数最后引发异常
1034     s = Student()
1035     s.age                                                                    # s中age属性不存在 故调用__getattr__方法 返回25
1036     # (5)__call__方法: 定制类的'可调用'性
1037     class Student(object):
1038         def __call__(self):                                    # 也可以带参数
1039             print('Calling......')
1040     s = Student()
1041     s()                                                                        # s变成了可调用的 也可以带参数
1042     callable(s)                                                        # 测试s的可调用性 返回True
1043     #    (6)__len__方法:求类的长度
1044     def __len__(self):
1045         return len(self.data)
1046     
1047 #-- 动态创建类type()
1048     # 一般创建类 需要在代码中提前定义
1049         class Hello(object):
1050             def hello(self, name='world'):
1051                 print('Hello, %s.' % name)
1052         h = Hello()
1053         h.hello()                                                        # Hello, world
1054         type(Hello)                                                    # Hello是一个type类型 返回<class 'type'>
1055         type(h)                                                            # h是一个Hello类型 返回<class 'Hello'>
1056     # 动态类型语言中 类可以动态创建 type函数可用于创建新类型
1057         def fn(self, name='world'):                 # 先定义函数
1058             print('Hello, %s.' % name)
1059         Hello = type('Hello', (object,), dict(hello=fn)) 
1060         # 创建Hello类 type原型: type(name, bases, dict)
1061         h = Hello()                                                    # 此时的h和上边的h一致
1062 
1063 
1064 '''异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关----异常相关'''
1065     
1066 #--    #捕获异常: 
1067         try:
1068         except:                                                            # 捕获所有的异常 等同于except Exception:
1069         except name:                                                # 捕获指定的异常
1070         except name, value:                                    # 捕获指定的异常和额外的数据(实例)
1071         except (name1, name2):
1072         except (name1, name2), value:
1073         except name4 as X:
1074         else:                                                                # 如果没有发生异常
1075         finally:                                                        # 总会执行的部分
1076     # 引发异常: raise子句(raise IndexError)
1077         raise <instance>                                        # raise instance of a class, raise IndexError()
1078         raise <class>                                                # make and raise instance of a class, raise IndexError
1079         raise                                                                # reraise the most recent exception
1080 
1081 #-- Python3.x中的异常链: raise exception from otherException
1082     except Exception as X:
1083         raise IndexError('Bad') from X
1084         
1085 #--    assert子句: assert <test>, <data>
1086     assert x < 0, 'x must be negative'
1087     
1088 #--    with/as环境管理器:作为常见的try/finally用法模式的替代方案
1089     with expression [as variable], expression [as variable]:
1090     # 例子:
1091         with open('test.txt') as myfile:
1092             for line in myfile: print(line)
1093     # 等同于:
1094         myfile = open('test.txt')
1095         try:
1096             for line in myfile: print(line)
1097         finally:
1098             myfile.close()
1099 
1100 #--    用户自定义异常: class Bad(Exception):.....
1101     '''
1102     Exception超类 / except基类即可捕获到其所有子类
1103     Exception超类有默认的打印消息和状态 当然也可以定制打印显示:
1104     '''
1105     class MyBad(Exception):
1106         def __str__(self):
1107             return '定制的打印消息'
1108     try:
1109         MyBad()
1110     except MyBad as x:
1111         print(x)
1112     
1113 #--    用户定制异常数据
1114     class FormatError(Exception):
1115         def __init__(self, line ,file):
1116             self.line = line
1117             self.file = file
1118     try:
1119         raise FormatError(42, 'test.py')
1120     except FormatError as X:
1121         print('Error at ', X.file, X.line)
1122     # 用户定制异常行为(方法):以记录日志为例
1123     class FormatError(Exception):
1124         logfile = 'formaterror.txt'
1125         def __init__(self, line ,file):
1126             self.line = line
1127             self.file = file
1128         def logger(self):
1129             open(self.logfile, 'a').write('Error at ', self.file, self.line)
1130     try:
1131         raise FormatError(42, 'test.py')
1132     except FormatError as X:
1133         X.logger()
1134 
1135 #--    关于sys.exc_info:允许一个异常处理器获取对最近引发的异常的访问
1136     try:
1137         ......
1138     except:
1139         # 此时sys.exc_info()返回一个元组(type, value, traceback)
1140         # type:正在处理的异常的异常类型
1141         # value:引发的异常的实例
1142         # traceback:堆栈信息
1143         
1144 #-- 异常层次
1145     BaseException
1146     +-- SystemExit
1147     +-- KeyboardInterrupt
1148     +-- GeneratorExit
1149     +-- Exception
1150         +-- StopIteration
1151         +-- ArithmeticError
1152         +-- AssertionError
1153         +-- AttributeError
1154         +-- BufferError
1155         +-- EOFError
1156         +-- ImportError
1157         +-- LookupError
1158         +-- MemoryError
1159         +-- NameError
1160         +-- OSError
1161         +-- ReferenceError
1162         +-- RuntimeError
1163         +-- SyntaxError
1164         +-- SystemError
1165         +-- TypeError
1166         +-- ValueError
1167         +-- Warning
1168 
1169     
1170 '''Unicode和字节字符串---Unicode和字节字符串----Unicode和字节字符串----Unicode和字节字符串----Unicode和字节字符串----Unicode和字节字符串----Unicode和字节字符串----Unicode和字节字符串'''
1171 
1172 #-- Python的字符串类型
1173     '''Python2.x'''
1174     # 1.str表示8位文本和二进制数据
1175     # 2.unicode表示宽字符Unicode文本
1176     '''Python3.x'''
1177     # 1.str表示Unicode文本(8位或者更宽)
1178     # 2.bytes表示不可变的二进制数据
1179     # 3.bytearray是一种可变的bytes类型
1180 
1181 #--    字符编码方法
1182     '''ASCII'''                                    # 一个字节,只包含英文字符,0到127,共128个字符,利用函数可以进行字符和数字的相互转换
1183     ord('a')                                        # 字符a的ASCII码为97,所以这里返回97
1184     chr(97)                                            # 和上边的过程相反,返回字符'a'
1185     '''Latin-1'''                                # 一个字节,包含特殊字符,0到255,共256个字符,相当于对ASCII码的扩展
1186     chr(196)                                        # 返回一个特殊字符:Ä
1187     '''Unicode'''                                # 宽字符,一个字符包含多个字节,一般用于亚洲的字符集,比如中文有好几万字
1188     '''UTF-8'''                                    # 可变字节数,小于128的字符表示为单个字节,128到0X7FF之间的代码转换为两个字节,0X7FF以上的代码转换为3或4个字节
1189     # 注意:可以看出来,ASCII码是Latin-1和UTF-8的一个子集
1190     # 注意:utf-8是unicode的一种实现方式,unicode、gbk、gb2312是编码字符集
1191     
1192 #--    查看Python中的字符串编码名称,查看系统的编码
1193     import encodings
1194     help(encoding)
1195     import sys
1196     sys.platform                                # 'win64'
1197     sys.getdefaultencoding()        # 'utf-8'
1198     sys.getdefaultencoding()         # 返回当前系统平台的编码类型
1199     sys.getsizeof(object)                # 返回object占有的bytes的大小
1200     
1201 #--    源文件字符集编码声明: 添加注释来指定想要的编码形式 从而改变默认值 注释必须出现在脚本的第一行或者第二行
1202     """说明:其实这里只会检查#和coding:utf-8,其余的字符都是为了美观加上的"""
1203     # _*_ coding: utf-8 _*_
1204     # coding = utf-8
1205     
1206 #--    #编码: 字符串 --> 原始字节        #解码: 原始字节 --> 字符串
1207 
1208 #-- Python3.x中的字符串应用
1209     s = '...'                                        # 构建一个str对象,不可变对象
1210     b = b'...'                                    # 构建一个bytes对象,不可变对象
1211     s[0], b[0]                                    # 返回('.', 113)
1212     s[1:], b[1:]                                # 返回('..', b'..')
1213     B = B'''
1214         xxxx
1215         yyyy
1216         '''
1217     # B = b'\nxxxx\nyyyy\n'
1218     # 编码,将str字符串转化为其raw bytes形式:
1219         str.encode(encoding = 'utf-8', errors = 'strict')
1220         bytes(str, encoding)
1221     # 编码例子:
1222         S = 'egg'
1223         S.encode()                                                    # b'egg'
1224         bytes(S, encoding = 'ascii')                # b'egg'
1225     # 解码,将raw bytes字符串转化为str形式:
1226         bytes.decode(encoding = 'utf-8', errors = 'strict')
1227         str(bytes_or_buffer[, encoding[, errors]])
1228     # 解码例子:
1229         B = b'spam'
1230         B.decode()                                                    # 'spam'
1231         str(B)                                                            # "b'spam'",不带编码的str调用,结果为打印该bytes对象
1232         str(B, encoding = 'ascii')                    # 'spam',带编码的str调用,结果为转化该bytes对象
1233     
1234 #-- Python2.x的编码问题
1235     u = u''
1236     print repr(u)                                                 # u'\xba\xba'
1237     s = u.encode('UTF-8')
1238     print repr(s)                                                 # '\xc2\xba\xc2\xba'
1239     u2 = s.decode('UTF-8')
1240     print repr(u2)                                                 # u'\xba\xba'
1241     # 对unicode进行解码是错误的
1242     s2 = u.decode('UTF-8')                                # UnicodeEncodeError: 'ascii' codec can't encode characters in position 0-1: ordinal not in range(128)
1243     # 同样,对str进行编码也是错误的
1244     u2 = s.encode('UTF-8')                                # UnicodeDecodeError: 'ascii' codec can't decode byte 0xc2 in position 0: ordinal not in range(128)
1245 
1246 #--    bytes对象
1247     B = b'abc'
1248     B = bytes('abc', 'ascii')
1249     B = bytes([97, 98, 99])
1250     B = 'abc'.encode()
1251     # bytes对象的方法调用基本和str类型一致 但:B[0]返回的是ASCII码值97, 而不是b'a'
1252     
1253 #--    #文本文件: 根据Unicode编码来解释文件内容,要么是平台的默认编码,要么是指定的编码类型
1254     # 二进制文件:表示字节值的整数的一个序列 open('bin.txt', 'rb')
1255     
1256 #--    Unicode文件
1257     s = 'A\xc4B\xe8C'                                            # s = 'A?BèC'  len(s) = 5
1258     #手动编码
1259         l = s.encode('latin-1')                            # l = b'A\xc4B\xe8C'  len(l) = 5
1260         u = s.encode('utf-8')                                # u = b'A\xc3\x84B\xc3\xa8C'  len(u) = 7
1261     #文件输出编码
1262         open('latindata', 'w', encoding = 'latin-1').write(s)
1263         l = open('latindata', 'rb').read()    # l = b'A\xc4B\xe8C'  len(l) = 5
1264         open('uft8data', 'w', encoding = 'utf-8').write(s)
1265         u = open('uft8data', 'rb').read()        # u = b'A\xc3\x84B\xc3\xa8C'  len(u) = 7
1266     #文件输入编码
1267         s = open('latindata', 'r', encoding = 'latin-1').read()    # s = 'A?BèC'  len(s) = 5
1268         s = open('latindata', 'rb').read().decode('latin-1')        # s = 'A?BèC'  len(s) = 5
1269         s = open('utf8data', 'r', encoding = 'utf-8').read()        # s = 'A?BèC'  len(s) = 5
1270         s = open('utf8data', 'rb').read().decode('utf-8')                # s = 'A?BèC'  len(s) = 5
1271         
1272 
1273 '''其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----其他----'''
1274     
1275 #--    60个字符解决FizzBuzz:
1276     '''写一个程序, 打印数字1到100, 3的倍数打印“Fizz”来替换这个数, 5的倍数打印“Buzz”, 既是3又是5的倍数的打印“FizzBuzz”'''
1277     for x in range(101):
1278         print("fizz"[x%3*4::]+"buzz"[x%5*4::] or x)        # 解释:最主要用到列表(字符串)的子表
1279         
1280 #--    Python实现任意深度的赋值 例如a[0] = 'value1'; a[1][2] = 'value2'; a[3][4][5] = 'value3'
1281     class MyDict(dict):
1282         def __setitem__(self, key, value):                        # 该函数不做任何改动 这里只是为了输出
1283             print('setitem:', key, value, self)
1284             super().__setitem__(key, value)
1285         def __getitem__(self, item):                                    # 主要技巧在该函数
1286             print('getitem:', item, self)                                # 输出信息
1287             # 基本思路: a[1][2]赋值时 需要先取出a[1] 然后给a[1]的[2]赋值
1288             if item not in self:                                                # 如果a[1]不存在 则需要新建一个dict 并使得a[1] = dict
1289                 temp = MyDict()                                                        # 新建的dict: temp
1290                 super().__setitem__(item, temp)                        # 赋值a[1] = temp
1291                 return temp                                                                # 返回temp 使得temp[2] = value有效
1292             return super().__getitem__(item)                        # 如果a[1]存在 则直接返回a[1]
1293     # 例子:
1294         test = MyDict()
1295         test[0] = 'test'
1296         print(test[0])
1297         test[1][2] = 'test1'
1298         print(test[1][2])
1299         test[1][3] = 'test2'
1300         print(test[1][3])
1301 
1302 #-- Python中的多维数组
1303     lists = [0] * 3                                                # 扩展list,结果为[0, 0, 0]
1304     lists = [[]] * 3                                            # 多维数组,结果为[[], [], []],但有问题,往下看
1305     lists[0].append(3)                                        # 期望看到的结果[[3], [], []],实际结果[[3], [3], [3]],原因:list*n操作,是浅拷贝,如何避免?往下看
1306     lists = [[] for i in range(3)]                # 多维数组,结果为[[], [], []]
1307     lists[0].append(3)                                        # 结果为[[3], [], []]
1308     lists[1].append(6)                                        # 结果为[[3], [6], []]
1309     lists[2].append(9)                                        # 结果为[[3], [6], [9]]
1310     lists = [[[] for j in range(4)] for i in range(3)]
1311     lists     # 3行4列,且每一个元素为[]
posted @ 2016-10-13 22:33  林博士  阅读(2087)  评论(0)    收藏  举报