1.如何比较两个代码的快慢
O(1) 一个运算时间基本单位,大致理解为一个基础运算,例如+-*/print等等
时间复杂度
只考虑影响问题关键的,省略那些无关痛痒的
时间复杂度用来估算算法运算的时间。一边来说,时间复杂度越高,运算越慢。(相同运算内容,相同机器)
常见的时间复杂度(按效率排序)
O(1)
确定问题规模n
循环减半过程—->logn(2卫底)
K曾关于n的循环 —> n*k
复杂情况:根据算法执行过程判断
每层多少个参数 有多少层
空间复杂度:
用来评估算法内存占用的大小
表达方式与时间复杂度完全一样
算法使用了几个变量:O(1)
算法使用了长度为n的一位列表:O(n)
算法使用了m行n列的二位列表:O(mn)
宁可占用更多内存,也要更快
递归:
1.调用自身
2.结束条件
为什么打印顺序不同
汉诺塔问题:
递推式:h(x) = h(x-1) + 1
def hanoi(n,a,b,c): # n:盘子个数 abc三个柱子,从a经过b移动到cif n>0:hanoi(n-1,a,c,b) ## n-1个盘子,从a经过c移动到b,因为下一步是最后一个盘子,由a移动到cprint('moving for %s to %s' % (a,c))hanoi(n-1,b,a,c) ## n-1个盘子,从b经过a移动到chanoi(3,'A','B','C')
列表查找问题
- 顺序查找

def linear_search(li,val):for ind,v in enumerate(li):if v == val:return indelse:return None
ps: enumerate() 函数用于将一个可遍历的数据对象(如列表、元组或字符串)组合为一个索引序列,同时列出数据和数据下标,一般用在 for 循环当中。
二分查找 Binary Search



def binary_search(li,val):left =0right = len(li) - 1while left <= right: # 候选区有值mid =(left + right) //2if li[mid] == val:return midelif li[mid] > val: ## 待查找值在mid左边right = mid-1elif li[mid] < val: #待查找值在mid右边left = mid +1else:return None## keypoint: 不断移动左右边界,实现二分查找
时间复杂度: O(logn)
list 中的内置列表查找函数:index()是线性查找,因为列表不一定有序
排序的时间复杂度 大于O(n),所以自己权衡一下
排序



每蹚跳出一个无序区最大的数,跳入有序区
时间复杂度:O(n2), 原地排序
def bubble_sort(li):for i in range(len(li)-1): ## 第i趟for j in range(len(li)-i-1):if li[j] > li[j+1]:li[j],li[j+1] = li[j+1],li[j] ## list内部元组交换import randomli = [random.randint(0,100) for i in range(100)]print(li)bubble_sort(li)print(li)
改进:如果到第x趟时,已经排好了,就停止,不用跑完len(li)-1趟
def bubble_sort_de(li): ## 降序for i in range(len(li)-1):exchange = Falsefor j in range(len(li)-i-1):if li[j] < li[j+1]:li[j], li[j + 1] = li[j + 1], li[j]exchange = Trueprint(li)if not exchange:returnli = [6,5,4,3,2,1,7,8,9]bubble_sort_de(li)print(li)
- 选择排序
没遍历一次,挑一个最大或最小值,排起来
def select_sort_simple(li):li_new=[]for i in range(len(li)):min_val = min(li)li_new.append(min_val)li.remove(min_val)## remove的操作,不是O1,而是On,因为要先找到,同理min,这两个此处是一个Onreturn li_new## 生成了一个新list,废内存,占两个内存## 复杂度 0(n2)li = [3,2,4,1,5,6,8,9]print(select_sort_simple(li))
改进:不新建list增加内存,采用交换
def slect_sort(li):for i in range(len(li)-1):min_loc = i # 无序区第一个数for j in range(i+1,len(li)): ## 能少算一个数就少算一个数if li[j] < li[min_loc]:min_loc = jli[i],li[min_loc] = li[min_loc],li[i]print(li)li = [3,2,4,1,5,6,8,7,9]print(li)slect_sort(li)## 复杂度:两层循环,O(n2)

插入排序:
对有序区不断排序
def insert_sort(li):for i in range(1,len(li)): ## i表示摸到的牌的下标tmp = li[i]j = i -1 ## 手里的牌的细表while j >= 0 and li[j] > tmp: # li[i]下表在变,所以要把值存起来li[j+1] = li[j] ## 手里的牌右移j -= 1 ## 指针左移li[j+1] = tmpprint(li)li = [3,2,4,1,5,6,8,7,9]print(li)insert_sort(li)## 复杂度:两层循环,O(n2)
题外话:电脑一秒大约运算10**7
