# coding:utf-8

#----------------------------------------  print和import的更多信息  --------------------------------

#使用逗号输出

#打印多个表达式也是可行的,只要将它们用逗号隔开就好:

print 'Age:' , 42           #可以看到,每个参数之间都插入了一个空格符。

#Age: 42

#如果想要同时输出文本和变量值,却不希望使用字符串格式化方法,那这个特性就非常有用:

name = 'Gumby'

salutation = 'Mr.'

greeting = 'Hello,'

print greeting, salutation, name

#Hello, Mr. Gumby

#注意,如果greeting变量里字符串不带逗号:

greeting = 'Hello'

#那么结果中怎么能得到逗号呢?

print greeting, ',', salutation, name       #这样做是不行的,因为语句会在逗号前加入空格。下面是一种解决方案:

#Hello , Mr. Gumby

print greeting + ',', salutation, name

#Hello, Mr. Gumby

#如果在结尾处加上逗号,那么接下来的语句会与前一条语句在同一行打印。

print 'Hello,',

print 'world!'

#Hello, world!

#把某件事作为另一件事导入

#从模块导入函数时候,通常可以使用

#import somemodule

#或者

#from somemodule import somefunction

#或者

#from somemodule import somefumction, anotherfunction, yetanotherfunction

#或者

#from somemodule import *

#只有确定自己想要从给定的模块导入所有功能是,才应该使用最后一个版本。

#如果两个模块都有open函数,使用如下方法:

#import somemodule

#module1.open(...)

#module2.open(...)

#但还有另外的选择,可以在语句末尾增加一个as子句,在该子句后给出想要使用的别名。

import math as foobar       #整个模块提供别名

print foobar.sqrt(4)

#2.0

from math import sqrt as foobar     #为函数提供别名

print foobar(4)

#2.0

#对于open函数,可以像下面这样使用:

#from module1 import open as open1

#from module2 import open as open2

#------------------------------------------  赋值魔法  -----------------------------------------------------

#序列解包

#将多个值得序列解开,然后放到变量的序列中。

x, y, z = 1, 2, 3

print x, y, z

#1 2 3

#交换两个(或更多个)变量

x, y = y, x

print x, y, z

#2 1 3

values = 1, 2, 3

print values

#(1, 2, 3)

x, y, z = values

print x

#1

scoundrel = {

'name': 'Robin', 'girlfriend': 'Marion'}

key, value = scoundrel.popitem()              #随机删除一对键和值,然后删除的键和值作为元组返回,这个元组赋予key和value变量

print key

#girlfriend

print value

#Marion

print scoundrel

#{'name': 'Robin'}

#所解包的序列中的元素数量必须和放置在赋值符号=左边的变量数量完全一致,否则会报错。

#x, y, z = 1, 2             会报错

#x, y, z = 1, 2, 3, 4       会报错

#链式赋值

#链式赋值是将同一个值赋给多个变量的捷径。

x = y = 1

print x

#1

print y

#1

#效果同上

x = 1

x = y

print x

#1

print y

#1

#注意上面的语句和下面的语句不一定等价:

x = 1

y = 1

#增量赋值

#对于*,/,% 等标准运算符都适用

#增量赋值可以让代码更加紧凑和简练,很多情况下会更易读。

x = 2

x += 1

print x

#3

x *= 2

print x

#6

#对于其他数据类型也适用。

fnoed = 'foo'

fnoed += 'bar'

print fnoed

#foobar

fnoed *= 2

print fnoed

#foobarfoobar

#----------------------------------------------  条件和条件语句  --------------------------------------------

#这就是布尔变量的作用

#下面的值在作为布尔表达式的时候,会被解释器看作假(false)

#False  None  0  ""  ()  []  {}

#其他一切都被解释为真,包括特殊值True

#标准布尔值为0表示假,1表示真

print True == 1

#True

print False == 0

#True

print True + False + 42

#43

#bool转换成布尔值

print bool('I think, therefore I am')

#True

print bool(42)

#True

print bool('')

#False

print bool(0)

#False

#条件执行和if语句

#如果条件(在if和冒号之间的表达式)判定为真,那么后面的语句块就会被执行。如果条件为假,语句块就不会被执行。

#name = raw_input('What is your name? ')

#if name.endswith('Gumby'):               #如果语句结尾是Gumby

#    print 'Hello, Mr. Gumby'

#执行结果:

#What is your name? Gumby

#Hello, Mr. Gumby

#else子句

#name = raw_input('What is your name? ')

#if name.endswith('Gumby'):               #如果语句结尾是Gumby

#    print 'Hello, Mr. Gumby'

#else:

#    print 'Hello, stranger'

#执行结果:

#What is your name? abc

#Hello, stranger

#elif子句

#如果需要检查多个条件,就可以使用elif,它是else if的简写,也是if和else子句的联合使用,也就具有条件的else子句。

#num = input('Enter a number: ')        可以使用int(raw_input('Enter a number: ')来替代

#if num > 0:

#    print 'The number is positive'

#elif num < 0:

#    print 'The number is negative'

#else:

#    print 'The number is zero'

#执行结果:

#Enter a number: 1

#The number is positive

#Enter a number: -1

#The number is negative

#Enter a number: 0

#The number is zero

#嵌套代码块

#if语句里面可以嵌套使用if语句,就像下面这样:

#name = raw_input('What is your name? ')

#if name.endswith('Gumby'):

#    if name.startswith('Mr. '):

#        print 'Hello,Mr. Gumby'

#    elif name.startswith('Mrs. '):

#        print 'Hello,Mrs. Gumby'

#    else:

#        print 'Hello, Gumby'

#else:

#    print 'Hello stranger'

#执行结果:

#What is your name? Mr. Gumby

#Hello,Mr. Gumby

#What is your name? Mrs. Gumby

#Hello,Mrs. Gumby

#What is your name? Gumby

#Hello, Gumby

#What is your name? abc

#Hello stranger

#更复杂的条件

#比较运算符

# x == y         x 等于 y

# x < y          x 小于 y

# x > y          x 大于 y

# x >= y         x 大于等于 y

# x <= y         x 小于等于 y

# x != y         x 不等于 y

# x is y         x 和 y 是同一个对象

# x is not y     x 和 y 是不同的对象

# x in y         x 是 y 容器(列如,序列)的成员

# x not in y     x 不是 y 容器(列如,序列)的成员

#注意如果偶然遇见x <> y 这样的表达式,它的意思其实就是x != y 不建议使用<>运算符,应该尽量避免使用它。

#在python中比较运算符和赋值运算符一样可以连接的---几个运算符可以连在一起使用,比如:

#0 < age < 100

#相等运算符

#如果想要知道两个东西是否相等,应该使用相等运算符,即两个等号==

print "foo" == "foo"

#True

print "foo" == "bar"   #为何用双等于,一个等于是变量赋值,所以用双等于

#False

#is:同一性运算符

#判断是否一个对象,如果值相同,并不一定是一个对象

x = y = [1, 2, 3]                #同一个对象

z = [1, 2, 3]                    #值相同,但和x y 不是一个对象

print x == y

#True

print x == z                    #值相同,所以True

#True

print x is y                    #是一个对象

#True

print x is z                    #不是一个对象,所以False

#False

#另外一个例子:

x = [1, 2, 3]

y = [2, 4]

print x is not y

del x[2]

y[1] = 1

y.reverse()         #反向排序

print x

#[1, 2]

print y

#[1, 2]

print x == y

#True

print x is y

#False

#in 成员资格运算符

#name = raw_input('What is your name? ')

#if 's' in name:

#    print 'Your name contains the letter "s".'

#else:

#    print 'Your name does not contain the letter "s".'

#执行结果:

#What is your name? tom

#Your name does not contain the letter "s".

#What is your name? toms

#Your name contains the letter "s".

#字符串和序列比较

#字符串按照字母顺序排列

print "alpha" < "beta"

#True

#大写的话比较会乱,所以先转换小写再比较

print 'FnOrD'.lower() == 'Fnord'.lower()

#True

#其他序列比较,比较的是序列元素

print [1, 2] < [2, 1]

#True

print [2, [1, 4]] < [2,[1,5]]

#True

#布尔运算符

#多个条件写法:

#number = input('Enter a number between 1 and 10: ')

#if number <= 10:

#    if number >= 1:

#        print 'Great!'

#    else:

#        print 'Wrong!'

#else:

#    print 'Wrong!'

#执行结果:

#Enter a number between 1 and 10: 1

#Great!

#Enter a number between 1 and 10: -1

#Wrong!

#Enter a number between 1 and 10: 11

#Wrong!

#这样写没问题,但方法太笨,简单方法:

#number = input('Enter a number between 1 and 10: ')

#if number <= 10 and number >= 1:             #and运算符就是所谓的布尔运算符,它连接两个布尔值,并且在两者都为真时返回真,否则返回假。与它同类还有两个运算符,or和not。使用这3个运算符就可以随意结合真值。

# 还有更简单的表达式 1<=number<=10

#    print 'Great!'

#else:

#    print 'Wrong!'

#执行结果:

#Enter a number between 1 and 10: 1

#Great!

#Enter a number between 1 and 10: -1

#Wrong!

#Enter a number between 1 and 10: 11

#Wrong!

#短路逻辑惰性求值

#表达式x and y需要两个变量为真才为真,所以如果x为假,表达式就会立刻返回false,而不管y的值。

#表达式x or y,x为真时,它直接返回x值(True),否则返回y值(True)

#这有什么用?它主要是避免了无用的执行代码,可以作为一种技巧使用。假设用户应该输入名字,但也可以选择什么都不输入,这时可以用默认值'<unknown>'。可以用if语句但可以用更简洁的方式:

#name = raw_input('Please enter your name: ') or '<unknown>'

# assert

#断言,要求程序代码中某些条件必须为真,否则报错退出,不再继续后续代码执行。用来测试调试程序代码的辅助条件。

age = 10

assert 0 < age < 100            #条件为真,不报错

#age = -1

#assert 0 < age < 100            #不为真,报错

#Traceback (most recent call last):

#  File "Z:/CloudStation/***.py", line 96, in <module>

#    assert 0 < age < 100

#AssertionError

#age = -1

#assert 0 < age < 100, 'The age must be realistic'

#条件后面添加字符串,可以用来注释解释错误信息。

#Traceback (most recent call last):

#  File "Z:/CloudStation/***.py", line 96, in <module>

#    assert 0 < age < 100, 'The age must be realistic'

#AssertionError: The age must be realistic

#-------------------------------------------  循环  --------------------------------------------------------

#while循环

x = 1

while x <= 100:         #打印1到100

   print x

   x += 1

#name = ''

#while not name:           #如果输入空格也被接受,防止这种情况,可以修改while not name or name.isspace() 或者 while not name.strip()

#    name = raw_input('Please enter your name: ')

#print 'Hello, %s!' % name

#for循环

#为一个集合每个元素执行一个代码块,比如序列,迭代对象

words = ['this', 'is', 'an', 'ex', 'parrot']

for word in words:

   print word

#this

#is

#an

#ex

#parrot

numbers = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

for number in numbers:

   print number

#0

#1

#2

#3

#4

#5

#6

#7

#8

#9

print range(0,10)           #range函数,下限默认为0,不包含上限

#[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

print range(10)             #默认下限为0

#[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

#rang函数一次创建整个序列,xrange一次只创建一个数,当需要一个巨大序列时xrange会更高效

for number in range(1,101):   #打印1到100,它比之前的while更简洁。

   print number

#如果能使用for,就尽量不用while

#循环遍历字典元素

#一个简单的for语句就能遍历字典的所有键,就像遍历访问序列一样

d = {

'x': 1, 'y': 2, 'z': 3}

for key in d:

   print key, 'corresponds to', d[key]

#y corresponds to 2

#x corresponds to 1

#z corresponds to 3

for key, value in d.items():

   print key, 'corresponds to', value

#y corresponds to 2

#x corresponds to 1

#z corresponds to 3

#一些迭代工具

#并行迭代

#程序可以同时迭代两个序列

names = ['anne', 'beth', 'george', 'damon']

ages = [12, 45, 32, 102]

#如果想要打印名字和对应的年龄:

for i in range(len(names)):

   print names[i], 'is', ages[i], 'years old'

#anne is 12 years old

#beth is 45 years old

#george is 32 years old

#damon is 102 years old

#zip函数并行迭代,可以把两个序列"压缩"在一起,然后返回一个元组的列表

print zip(names, ages)

#[('anne', 12), ('beth', 45), ('george', 32), ('damon', 102)]

#现在可以在循环中解包元组

for name, age in zip(names, ages):

   print name, 'is', age, 'years old'

#anne is 12 years old

#beth is 45 years old

#george is 32 years old

#damon is 102 years old

#zip函数也可以作用于任意多的序列,它可以作用不等长的序列,最短的序列用完的时候就会停止。

print zip(range(5),xrange(100000))      #不推荐用range替换xrange,range会计算所有的数字,xrange只计算前5个数字

#[(0, 0), (1, 1), (2, 2), (3, 3), (4, 4)]

#按索引迭代

names = ['anne', 'beth', 'anne', 'damon']

ages = [12, 45, 32, 102]

for name, age in zip(names, ages):

   if 'anne' in name:                                #找到索引进行替换

       index = names.index(name)

       names[index] = '[censored]'

       print names

#['[censored]', 'beth', 'anne', 'damon']

#['[censored]', 'beth', '[censored]', 'damon']

names = ['anne', 'beth', 'anne', 'damon']

ages = [12, 45, 32, 102]

index = 0

for name, age in zip(names, ages):

   if 'anne' in name:                              #更好的方法

       names[index] = '[censored]'

       print names

   index += 1

#['[censored]', 'beth', 'anne', 'damon']

#['[censored]', 'beth', '[censored]', 'damon']

#enumerate函数,可以在提供索引的地方迭代索引--值对

names = ['anne', 'beth', 'anne', 'damon']

ages = [12, 45, 32, 102]

for index, name in enumerate(names):                #更好的方法

   if 'anne' in name:

       names[index] = '[censored]'

       print names

# ['[censored]', 'beth', 'anne', 'damon']

# ['[censored]', 'beth', '[censored]', 'damon']

#翻转和排序迭代

#reversed和sorted。它同列表的reverse和sort方法类似,但作用于任何序列或可迭代对象上,不是原地修改对象,而是返回翻转或排序后的版本:

print sorted([4, 3, 6, 8, 3])

#[3, 3, 4, 6, 8]

print sorted('Hello, world!')

#[' ', '!', ',', 'H', 'd', 'e', 'l', 'l', 'l', 'o', 'o', 'r', 'w']

print list(reversed('Hello, word!'))

#['!', 'd', 'r', 'o', 'w', ' ', ',', 'o', 'l', 'l', 'e', 'H']

print ''.join(reversed('Hello, world!'))

#!dlrow ,olleH

#跳出循环

#break

#结束(跳出)循环可以使用break语句。

from math import sqrt

for n in range(99, 0, -1):          # -1是步长,负数是反向迭代

   root = sqrt(n)

   if root == int(root):

       print n

       break                       #跳出循环

#81

#continue

#continue意思是跳过剩余的循环体,但是不结束循环。当循环体很大而且很复杂的时候,这会很有用。

#for x in seq:

#   if condition1:continue

#   if condition2:continue

#   if condition2:continue

#   do_something()

#   do_something_else()

#   do_anoter_thing()

#   etc()

#很多时候,只要使用if语句就可以了:

#for x in seq:

#   if not (condition1 or condition2 or condition3):

#       do_something()

#       do_something_else()

#       do_anoter_thing()

#       etc()

#尽管continue语句非常有用,它却不是最本质的。应该习惯使用break语句,因为while True语句中会经常用到它。下一节会介绍:

#while True/break习语

while True:             #无限循环

   word = raw_input('Please enter a word: ')

   if not word: break          #满足条件跳出循环

   # 处理word:

   print 'The word was ' + word

#执行结果:

#Please enter a word: first

#The word was first

#Please enter a word: second

#The word was second

#Please enter a word:

#循环中的else子句

#当循环内使用break语句时,通常是因为“找到”了某物或者因为某事“发生”了。在跳出前做一些事情很简单(比如在break上面加入print n),但是有些时候想要在没有跳出之前做些事情。

from math import sqrt

for n in range(99, 81, -1):

   root = sqrt(n)

   if root == int(root):

       print n

       break

else:

   print "Didn't find it"

#Didn't find it

#列表推导式-轻量级循环

#列表推导式是利用其它列表创建新列表的一种方法。

print [x*x for x in range(10)]

#[0, 1, 4, 9, 16, 25, 36, 49, 64, 81]

print [x*x for x in range(10) if x % 3 == 0]

#[0, 9, 36, 81]

print [(x, y) for x in range(3) for y in range(3)]

#[(0, 0), (0, 1), (0, 2), (1, 0), (1, 1), (1, 2), (2, 0), (2, 1), (2, 2)]

#效果同上的for语句写法

result = []

for x in range(3):

   for y in range(3):

       result.append((x,y))

print result

#[(0, 0), (0, 1), (0, 2), (1, 0), (1, 1), (1, 2), (2, 0), (2, 1), (2, 2)]

girls = ['alice', 'bernic', 'clarice']

boys = ['chris', 'arnold', 'bob']

print [b+'+'+g for b in boys for g in girls if b[0] == g[0]]    #比较第一位相同字母的方式

#['chris+clarice', 'arnold+alice', 'bob+bernic']

#上面例子效果不高,因为它会检查每个可能的配对。

#下面例子创建了一个叫做letterGirls的字典,其中每一项都把单字母作为键,以女孩名字组成列表作为值。在字典建立后,列表推导式循环整个男孩集合,并且查找那些和当前男孩名字首字母相同的女孩集合。

#这样列表推导式就不用尝试所有的男孩女孩的组合,检查首字母是否匹配。

girls = ['alice', 'bernic', 'clarice']

boys = ['chris', 'arnold', 'bob']

letterGirls = {}

for girl in girls:

   letterGirls.setdefault(girl[0], []).append(girl)

   print letterGirls               #打印出来,查看循环情况

print [b+'+'+g for b in boys for g in letterGirls[b[0]]]

#{'a': ['alice']}

#{'a': ['alice'], 'b': ['bernic']}

#{'a': ['alice'], 'c': ['clarice'], 'b': ['bernic']}

#['chris+clarice', 'arnold+alice', 'bob+bernic']

#------------------------------------------ 三人行  -----------------------------------------------------

#pass

#什么都没发生,用于代码没写完,暂时顶替

#name = raw_input('Please enter name: ')

#if name == 'Ralph Auldus Mellish':

#    print 'Welcom!'

#elif name == 'Enid':

   #还没完...

#    pass                    #没想到写什么,但可以pass顶替,要不然程序会报错

#elif name == 'Bill Gates':

#    print 'Access Denied'

#del

#删除变量名字,值如果没有python会自动回收

scoundrel = {

'age':42, 'first name': 'Robin', 'last name': 'of Locksley'}

robin = scoundrel

print scoundrel

#{'last name': 'of Locksley', 'first name': 'Robin', 'age': 42}

print robin

#{'last name': 'of Locksley', 'first name': 'Robin', 'age': 42}

scoundrel = None        #当设置scoundrel为None的时候,字典通过robin还是可用

print scoundrel

#None

print robin

#{'last name': 'of Locksley', 'first name': 'Robin', 'age': 42}

robin = None           #当robin也设置为None的时候,字典就漂在内存里面,没有任何名字绑定在它上面,没有任何办法可以获取到它。就会被删除回收。也可以用None之外的其他值。同样会被删除回收。

print robin

#None

#另一种方法是del,它会删除对象引用和名字(但不会删除值)

x = 1

y = x

del x           #x和y指向一个值,删除x变量只是名字,所以y的值还在

print y

#1

#使用exec和eval执行和求值字符串

#执行一个字符串的语句时exec

exec "print 'Hello, world!'"

#exec 需要放入到一个命名空间,但这个空间是用户自己的,并不会影响全局空间

from math import sqrt

scope = {}          #增加scope,起到放置代码字符串命名空间作用的字典

exec 'sqrt = 1' in scope

print sqrt(4)       #没有覆盖sqrt函数

#2.0

print scope['sqrt']     #通过exec赋值的变量sqrt只在它的作用域内有效

#1

#eval

#eval(用于求值)是类似于exec的内建函数。exec语句会执行一系列python语句(不返回值),而eval会计算python表达式(以字符串形式书写),并且返回结果值。

#print eval (raw_input("Enter an arithmetic expression: "))

#Enter an arithmetic expression: 6 + 18 * 2

#42

#跟exec一样,eval也可以使用命名空间

scope = {}

scope['x'] = 2

scope['y'] = 3

print eval('x * y', scope)

#6

scope = {}

exec 'x = 2' in scope

print eval('x * x', scope)

#4

#涉及函数

#chr(n)                                     当传入序号n时,返回n所代表的包含一个字符的字符串,(0 <= n < 256)

#eval(source[, globals[, locals]])          将字符串作为表达式计算,并且返回值

#enumerate(seq)                             产生用于迭代的(索引,值)对

#ord(c)                                     返回单字符字符串的int值

#rang([start,] stop[, step])                创建整数的列表

#reversed(seq)                              产生seq中值的反向版本,用于迭代

#sorted(seq[,cmp][,key][,reverse])          返回seq中值排序后的列表

#xrange([start,]stop[,step])                创造xrange对象用于迭代

#zip(seq1,_seq2...)                         创造用于并行迭代的新序列