一直想学习python,虽然编程写了不少,但有时仍不得要领。这篇blog主要是记录python的一些主要特性。

前言

python学习总结,包括python的一些基本语法,高级特性,函数式编程,面向对象编程以及错误调试,测试和多线程。

Python基础

字符串

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-

第一行注释是为了告诉Linux/OS X系统,这是一个Python可执行程序,Windows系统会忽略这个注释;

第二行注释是为了告诉Python解释器,按照UTF-8编码读取源代码,否则,你在源代码中写的中文输出可能会有乱码。

数据类型和变量

整数
浮点数
字符串
布尔值

list和tuple

list与tuple的区别:list是可变的,tuple一旦定义则不可变。

定义

  1. List
  2. List的定义方法: classmates = []
  3. classmates = ['Michael', 'Bob', 'Tracy']
  4. print(classmates)
  5. print(classmates[-1])
  6. 结果:
  7. ['Michael', 'Bob', 'Tracy']
  8. Tracy
  9. Tuple
  10. Tuple的定义方法: classmates = ()
  11. classmates = ('Michael', 'Bob', 'Tracy')
  12. print(classmates)
  13. print(classmates[-1])
  14. 结果:
  15. ('Michael', 'Bob', 'Tracy')
  16. Tracy

条件判断

elif是else if的缩写,完全可以有多个elif,所以if语句的完整形式就是:

  1. if <条件判断1>:
  2. <执行1>
  3. elif <条件判断2>:
  4. <执行2>
  5. elif <条件判断3>:
  6. <执行3>
  7. else:
  8. <执行4>

循环

Python的循环有两种,一种是for…in循环,依次把list或tuple中的每个元素迭代出来

  1. names = ['Michael', 'Bob', 'Tracy']
  2. for name in names:
  3. print(name)

第二种循环是while循环,只要条件满足,就不断循环,条件不满足时退出循环。

  1. sum = 0
  2. n = 99
  3. while n > 0:
  4. sum = sum + n
  5. n = n - 2
  6. print(sum)

与大多数语言类似的 break 与 continue 语句的作用也相似。

dict与set

dict

Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,由于是用红黑树构造的,具有极快的查找速度。

以下介绍dict的定义,索引以及get(),pop()函数。

  1. dict定义:
  2. d = {'Michael': 95, 'Bob': 75, 'Tracy': 85}
  3. 索引:
  4. d = {'Michael': 95, 'Bob': 75, 'Tracy': 85}
  5. print(d['Bob'])
  6. 结果:75
  7. get()方法,dict提供的get方法,如果key不存在,可以返回None,或者自己指定的value
  8. print(d.get('Bob', 0))
  9. 结果:75
  10. print(d.get('B', 0))
  11. 结果:0
  12. pop()方法,要删除一个key,用pop(key)方法,对应的value也会从dict中删除:
  13. d = {'Michael': 95, 'Bob': 75, 'Tracy': 85}
  14. d.pop('Bob')
  15. print(d)
  16. 结果:{'Michael': 95, 'Tracy': 85}

函数

定义函数

定义函数例子:

  1. def my_abs(x):
  2. if x >= 0:
  3. return x
  4. else:
  5. return -x

如果你已经把my_abs()的函数定义保存为abstest.py文件了,那么,可以在该文件的当前目录下启动Python解释器,用from abstest import my_abs来导入my_abs()函数,注意abstest是文件名(不含.py扩展名)

空函数

如果想定义一个什么事也不做的空函数,可以用pass语句:

  1. def nop():
  2. pass

缺少了pass,代码运行就会有语法错误。

参数检查

调用函数时,如果参数个数不对,Python解释器会自动检查出来,并抛出TypeError:

  1. >>> my_abs(1, 2)
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. TypeError: my_abs() takes 1 positional argument but 2 were given

但是如果参数类型不对,Python解释器就无法帮我们检查。

让我们修改一下my_abs的定义,对参数类型做检查,只允许整数和浮点数类型的参数。数据类型检查可以用内置函数isinstance()实现:

  1. def my_abs(x):
  2. if not isinstance(x, (int, float)):
  3. raise TypeError('bad operand type')
  4. if x >= 0:
  5. return x
  6. else:
  7. return -x

返回多个值

函数可以返回多个值吗?答案是肯定的。

  1. import math
  2. def move(x, y, step, angle=0):
  3. nx = x + step * math.cos(angle)
  4. ny = y - step * math.sin(angle)
  5. return nx, ny

以上代码返回的多个参数会自动以tuple的方式返回。

  1. r = move(100, 100, 60, math.pi / 6)
  2. print(r)
  3. 结果:
  4. (151.96152422706632, 70.0)

函数的参数

位置参数

  1. def power(x, n):
  2. s = 1
  3. while n > 0:
  4. n = n - 1
  5. s = s * x
  6. return s

power(x, n)函数有两个参数:x和n,这两个参数都是位置参数,调用函数时,传入的两个值按照位置顺序依次赋给参数x和n。

默认参数

添加默认参数:

  1. def power(x, n=2):
  2. s = 1
  3. while n > 0:
  4. n = n - 1
  5. s = s * x
  6. return s

当我们调用power(5)时,相当于调用power(5, 2)。

默认参数的一个需要注意的问题:

  1. def add_end(L=[]):
  2. L.append('END')
  3. return L

正常调用时,结果似乎不错:

  1. add_end([1, 2, 3])
  2. 结果:[1, 2, 3, 'END']

但多次调用时,就出问题了:

  1. print(add_end())
  2. print(add_end())
  3. print(add_end())
  4. 结果:
  5. ['END']
  6. ['END', 'END']
  7. ['END', 'END', 'END']

Python函数在定义的时候,默认参数L的值就被计算出来了,即[],因为默认参数L也是一个变量,它指向对象[],每次调用该函数,如果改变了L的内容,则下次调用时,默认参数的内容就变了,不再是函数定义时的[]了。

所以,定义默认参数要牢记一点:默认参数必须指向不变对象!

修改后的函数:

  1. def add_end(L=None):
  2. if L is None:
  3. L = []
  4. L.append('END')
  5. return L

可变参数

在Python函数中,还可以定义可变参数。顾名思义,可变参数就是传入的参数个数是可变的,可以是1个、2个到任意个,还可以是0个。

  1. def calc(numbers):
  2. sum = 0
  3. for n in numbers:
  4. sum = sum + n * n
  5. return sum

关键字参数

可变参数允许你传入0个或任意个参数,这些可变参数在函数调用时自动组装为一个tuple。而关键字参数允许你传入0个或任意个含参数名的参数,这些关键字参数在函数内部自动组装为一个dict。

示例:

  1. def person(name, age, **kw):
  2. print('name:', name, 'age:', age, 'other:', kw)
  3. person('Bob', 35, city='Beijing')
  4. person('Adam', 45, gender='M', job='Engineer')
  5. 结果:
  6. name: Bob age: 35 other: {'city': 'Beijing'}
  7. name: Adam age: 45 other: {'gender': 'M', 'job': 'Engineer'}

命名关键字参数

要限制关键字参数的名字,就可以用命名关键字参数。
定义方法如下:

  1. def person(name, age, *, city, job):
  2. print(name, age, city, job)

参数组合

在Python中定义函数,可以用必选参数、默认参数、可变参数、关键字参数和命名关键字参数,这5种参数都可以组合使用。但是请注意,参数定义的顺序必须是:必选参数、默认参数、可变参数、命名关键字参数和关键字参数。

  1. def f1(a, b, c=0, *args, **kw):
  2. print('a =', a, 'b =', b, 'c =', c, 'args =', args, 'kw =', kw)
  3. def f2(a, b, c=0, *, d, **kw):
  4. print('a =', a, 'b =', b, 'c =', c, 'd =', d, 'kw =', kw)

在函数调用的时候,Python解释器自动按照参数位置和参数名把对应的参数传进去。

  1. >>> f1(1, 2)
  2. a = 1 b = 2 c = 0 args = () kw = {}
  3. >>> f1(1, 2, c=3)
  4. a = 1 b = 2 c = 3 args = () kw = {}
  5. >>> f1(1, 2, 3, 'a', 'b')
  6. a = 1 b = 2 c = 3 args = ('a', 'b') kw = {}
  7. >>> f1(1, 2, 3, 'a', 'b', x=99)
  8. a = 1 b = 2 c = 3 args = ('a', 'b') kw = {'x': 99}
  9. >>> f2(1, 2, d=99, ext=None)
  10. a = 1 b = 2 c = 0 d = 99 kw = {'ext': None}

递归函数

经典的例子:

  1. def fibo(x):
  2. if (x == 1 or x == 2):
  3. return 1
  4. else:
  5. return fibo(x-1) + fibo(x-2)
  6. print(fibo(30))

高级特性

切片

取一个list或tuple的部分元素是非常常见的操作。

例子:

  1. L = ['Michael', 'Sarah', 'Tracy', 'Bob', 'Jack']

L[0:3]表示,从索引0开始取,直到索引3为止,但不包括索引3。

  1. print(L[0:3])

结果:

  1. ['Michael', 'Sarah', 'Tracy']
  2. print(L[-2:-1])

结果:

  1. ['Bob']

创建一个0-99的数列

  1. L = list(range(100))

前10个数,每两个取一个

  1. print(L[:10:2])

结果:

  1. [0, 2, 4, 6, 8]

所有数,每5个取一个:

  1. print(L[::5])

结果:

  1. [0, 5, 10, 15, 20, 25, 30, 35, 40, 45, 50, 55, 60, 65, 70, 75, 80, 85, 90, 95]

迭代

在Python中,迭代是通过for … in来完成的。
dict的迭代:

  1. d = {'a': 1, 'b': 2, 'c': 3}
  2. for key in d:
  3. print(key)

结果:

  1. a
  2. b
  3. c

默认情况下,dict迭代的是key。如果要迭代value,可以用for value in d.values(),如果要同时迭代key和value,可以用for k, v in d.items()。

  1. d = {'a': 1, 'b': 2, 'c': 3}
  2. for key in d.values():
  3. print(key)

结果:

  1. 1
  2. 2
  3. 3

同时获取key和value:

  1. d = {'a': 1, 'b': 2, 'c': 3}
  2. for key in d.items():
  3. print(key)

结果:

  1. ('a', 1)
  2. ('b', 2)
  3. ('c', 3)

列表生成式

列表生成式即List Comprehensions,是Python内置的非常简单却强大的可以用来创建list的生成式。

列表生成式则可以用一行语句代替循环生成list:

  1. print(x * x for x in range(1, 11)])

结果:

  1. [1, 4, 9, 16, 25, 36, 49, 64, 81, 100]

随机生成一个40位的随机数:

  1. print("%s"*5%tuple([random.randint(10000000,99999999) for i in range(5)]))

还可以使用两层循环,可以生成全排列:

  1. [m + n for m in 'ABC' for n in 'XYZ']

结果:

  1. ['AX', 'AY', 'AZ', 'BX', 'BY', 'BZ', 'CX', 'CY', 'CZ']

生成器

通过列表生成式,我们可以直接创建一个列表。但是,受到内存限制,列表容量肯定是有限的。而且,创建一个包含100万个元素的列表,不仅占用很大的存储空间,如果我们仅仅需要访问前面几个元素,那后面绝大多数元素占用的空间都白白浪费了。

所以,如果列表元素可以按照某种算法推算出来,那我们是否可以在循环的过程中不断推算出后续的元素呢?这样就不必创建完整的list,从而节省大量的空间。在Python中,这种一边循环一边计算的机制,称为生成器:generator。

让我们看看列表生成式与生成器的区别:

  1. 列表生成式:
  2. L = [x * x for x in range(10)]
  3. 生成器:
  4. g = (x * x for x in range(10))

创建L和g的区别仅在于最外层的[]和(),L是一个list,而g是一个generator。
list可以直接打印出来,但generator只能一个一个的生成。generator也是可以迭代的。

  1. g = (x * x for x in range(10))
  2. for n in g:
  3. print(n)

generator非常强大。如果推算的算法比较复杂,用类似列表生成式的for循环无法实现的时候,还可以用函数来实现。

迭代器

凡是可作用于for循环的对象都是Iterable类型;

凡是可作用于next()函数的对象都是Iterator类型,它们表示一个惰性计算的序列;

集合数据类型如list、dict、str等是Iterable但不是Iterator,不过可以通过iter()函数获得一个Iterator对象。

Python的for循环本质上就是通过不断调用next()函数实现的。

  1.  

python语言特性总结的更多相关文章

  1. 写给.NET开发者的Python教程(一):C# vs Python: 语言特性、Conda和Jupyter Notebook环境

    承接上篇,本文会从语言特性.开发环境和必备工具来带领大家进入Python的世界. 语言特性 首先一起看下C#和Python在语言特性层面的对比,他们作为截然不同的两类面向对象高级语言,在语言层面上有何 ...

  2. Python语言特性之5:自省

    自省是Python中非常耀眼的特性. 自省就是面向对象的语言所写的程序在运行时,所能知道对象的类型.简单一句就是运行时能够获得对象的类型.比如 type() dir() getattr() hasat ...

  3. 转--Python语言特性

    1 Python的函数参数传递 看两个例子: a = 1 def fun(a): a = 2 fun(a) print a # 1 a = [] def fun(a): a.append(1) fun ...

  4. Python语言特性

    1 Python的函数参数传递 看两个例子: a = 1 def fun(a): a = 2 fun(a) print a # 1 a = [] def fun(a): a.append(1) fun ...

  5. python 面试题一:Python语言特性

    1 Python的函数参数传递 两个例子 a = 1 def fun(a): a = 2 fun(a) print a # a = [] def fun(a): a.append(1) fun(a) ...

  6. python 语言特性

    动态强类型: 动态类型语言:在运行期进行类型检查的语言,也就是在编写代码的时候可以不指定变量的数据类型,比如Python和Ruby 静态类型语言:它的数据类型是在编译期进行检查的,也就是说变量在使用前 ...

  7. python语言特性-------python2.7教程学习【廖雪峰版】(一)

    开始学习廖雪峰的py2.7教程: 2017年6月5日12:54:28 笔记: 廖雪峰python2.7教程1.用任何编程语言来开发程序,都是为了让计算机干活.  2.Python是一种相当高级的语言. ...

  8. python语言特性简要记载

    1.python是解释型语言,而c,c++等是编译型语言. 2.python是动态类型语言,这意味着你不需要在声明变量时指定类型. 3.Python是面向对象语言,所有允许定义类并且可以继承和组合.P ...

  9. Python 语言特性:编译+解释、动态类型语言、动态语言

    1. 解释性语言和编译性语言 1.1 定义 1.2 Python 属于编译型还是解释型? 1.3 收获 2. 动态类型语言 2.1 定义 2.2 比较 2. 动态语言(动态编程语言) 3.1 定义 3 ...

随机推荐

  1. 庞老师集群.ziw

    2017年2月17日, 星期五 庞老师集群 链接:http://pan.baidu.com/s/1mhSw2TE 密码:hzz4   更改子网IP,及网关:     null

  2. webapi + windows计划 + mshta 实现定时执行任务

    当然,实现定时任务有更好的操作方式,比如方式一:asp.net mvc+quartz.net +corn +webapi,asp.net mvc做任务管理的平台,使用CronTrigger做定时触发, ...

  3. Crash Consistency : FSCK and Journaling

    现在开始今天的第三篇博客的撰写,不能扯淡了,好多任务啊.但是还是忍不住吐槽一下,之前选择这篇文章纯属是个意外,我把Crash看做了Cache,唉,要不然也就不用写这篇文章了. 1. 这篇博客讲什么? ...

  4. 你知道吗?衡量 Web 性能的几个关键指标

    自网站诞生以来,响应速度/响应时间一直都是大家关心的话题,而速度慢乃是网站的一个杀手,正当大家以为四核和宽带能力的提升能够解决这些问题时,Wi-Fi和移动设备为热点移动互联网又悄然兴起. 在2006年 ...

  5. final关键字详解

    java中,final关键字可以用来修饰类.方法和变量(包括成员变量和局部变量).下面就从这三个方面来了解一下final关键字的基本用法. 1.修饰类 当用final修饰一个类时,表明这个类不能被继承 ...

  6. cin循环输入控制问题

    之前写一个简单的输入节点值自动生成链表的测试程序,发现cin的输入控制好像在VC++6.0和VS2010中不一样,特此记录. 现在有以下代码: vector<int> ivec; int ...

  7. 4B/5B编码原理

    4B/5B编码原理 什么是4B/5B编码? 4B/5B编码是百兆以太网(即快速以太网)中线路层编码类型之一,就是用5bit的二进制数来表示4bit二进制数,映射方式如下表所示: 为什么要进行4B/5B ...

  8. https、socket、http协议

    一.https https 其实是由两部分组成:http+ssl(Secure Sockets Layer 安全套接层)/tls(Transport Layer Security 继任者安全传输层), ...

  9. HDU 5936 朋友

    题意为给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点作为根. 当一方操作时,他们需要先选择一个不为根的点,满足该点到其父亲的边权为1; 然后 ...

  10. Java Web Project Problems

    A: 项目红叉 1. 检验 Java Builder  Path 2. 检查 Projects Facets 3. 查看 Targets Runtimes B:项目红感叹号 1. 查看问题栏 Prob ...