算法练习(一)python 数据结构与算法 栈结构 括号匹配问题及二进制转换 学习记录

最近想学习python 脚本与爬虫的编写,先学点数据结构打点基础,看能不能进一步的了解算法思想。

其实我原来对算法还有点排斥,不知道怎么的就想不开想去学了。

目录

括号匹配问题

十进制转二进制


场景描述:在实际计算中 ,我们经常遇见

例子1:    (5+8)*(7+8)/(1+2)
我们通过(   )来改变计算顺序,那么计算机怎么识别这些括号正好能相对呢?
假如 不小心多打了一个( 该怎么识别呢?,上述例子只有三个括号,正常来说不会打错。
例子2:
在Lisp的编程语言有如下语法结构
    (defun square(n)
        (*n n))
它定义了一个名为square的函数,该函数会返回参数n的平方值。如果我们在一次运行中调用了多次
这个函数,那么存在的括号数量非常多,该怎么判断我们所有的括号都正常匹配呢?

算法: 从左到右读取一个括号串,然后判断其中的括号是否匹配。

例如(((()))) √
       ((())()    ×

 那么该如何设计一个算法呢,想我这样的小白基本上就无所适从了

不过,当意识到使用栈结构可以很好的解决这个问题,其实问题就应刃而解了。可以不急着理解思想,先顺着读一遍代码就行

括号匹配问题

from pythonds.basic import Stack

def parChecker(symbolString):     #定义匹配函数
    s=Stack()                     #定义栈结构
    balanced=True                #平衡为Ture
    index=0
    while index<len(symbolString) and balanced:  #进行循环,且循环中,栈平衡为Ture
        symbol=symbolString[index]
        if symbol=="(":                          #匹配到( 就进行进栈操作
            s.push(symbol)
        else:
            if s.isEmpty():                      #如果栈为空的情况下,匹配到右括号就报错
                balanced=False
            else:
                s.pop()                          #栈不为空,也就是存在左括号的情况下一旦
        index+=1                                 #匹配到右括号 就让出栈
    if balanced and s.isEmpty():                 #如果匹配完成后,栈为空且平衡就返回Ture
        return True
    else:
        return False                             #否则False

print(parChecker('(())()'))
print(parChecker('((()))'))

进阶版,可匹配不同的括号(添加了一个matches函数)

from pythonds.basic import Stack


def matches(open, close):
    opens = "([{"
    closers = ")]}"
    return opens.index(open) == closers.index(close)


def parChecker(symbolString):
    s = Stack()         #建栈

    balanced = True     #平衡 首先定义为Ture
    index = 0

    while index < len(symbolString) and balanced:  #当栈处于平衡时,开始循环
        symbol = symbolString[index]               #依次赋值
        if symbol in "([{":                        #当匹配到([{ 进栈
            s.push(symbol)
        else:
            if s.isEmpty():                        #避免接收到的符号是)]},且栈为空的情况
                balanced=False
            else:
                top = s.pop()                      #在栈里存在左括号的情况下,匹配到右括号,出栈
                if not matches(top, symbol):       #没有右括号没有匹配到对应的左括号
                    balanced = False

        index = index + 1
    if balanced and s.isEmpty():                   #如果循环完了之后,栈处于平衡,且为空情况下
        return True                                #返回Ture
    else:
        return False                               #返回False


print(parChecker('((()())'))
print(parChecker('((([[))))'))
print(parChecker('((){})'))

多读两遍其实不难理解

  1. 先建立一个栈结构
  2. 栈里只存左括号
  3. 如果匹配到右括号就进行出栈操作(符合栈先进后出,后进先出的栈特性)

所以,理解栈的这种先进后出,后进先出的反转概念,对以后用栈结构解决实际问题时很有帮助。

以前我一直在想学习数据结构与算法有什么作用,甚至对搞算法有点排斥,不过静下来学之后,发现 数据结构与算法还是作用很大的,其实它是我们解决复杂问题的一个很好的工具。

有兴趣的可以看下面一个例子。

十进制转二进制

十进制转二进制问题

正整数十进制转换二进制:
要点:除二取余,倒序排列
解释:将一个十进制数除以二,得到的商再除以二,依此类推直到商等于一或零时为止,倒取将除得的余数,即换算为二进制数的结果  

例如把52换算成二进制数,计算结果如图:

52除以2得到的余数依次为:0、0、1、0、1、1,倒序排列,所以52对应的二进制数就是110100。 

可以看出我们首先得到的余数反而是放在最后面的,这就符合栈的反转特性,

from pythonds.basic import Stack
def divideBy2(decNumber):
    remstack=Stack()            #建立空栈

    while decNumber >0:         #如果大于0
        rem=decNumber %2        #求余
        remstack.push(rem)      #将余数压入栈
        decNumber=decNumber //2 #将数字除2 继续进入下一轮循环

    binString=""
    while not remstack.isEmpty():   #上述栈不为空的情况下
        binString =binString+str(remstack.pop()) #将得到的余数压出

    return binString
print(divideBy2(251))

 

10进制转其他进制

from pythonds.basic import Stack
def baseConverter(decNumber,base):
    digits='0123456789ABCDEF'

    remstack=Stack()

    while decNumber>0:
        rem=decNumber %base
        remstack.push(rem)
        decNumber=decNumber //base

    newString=""
    while not remstack.isEmpty():
        newString=newString+digits[remstack.pop()]

    return newString

print(baseConverter(2534,5))

参考资料:

python数据结构与算法分析 Bradley N.Miller&David L.Ranum

二进制与十进制间的转换方法(图文教程) - yaoming-007 - 博客园

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值