LoginSignup
1
1

More than 3 years have passed since last update.

Python初心者(中学生)の組み合わせ生成が斜め上だった件(再帰処理で組み合わせ生成)

Last updated at Posted at 2019-12-09

数学パズル

Python勉強中のK君数学パズル
これの第二回目。

四桁の数字の間に四則演算を入れて式にして、演算した結果がもとの数字の逆順と一致する条件を探す問題。四則演算は最低1か所入れる。
答えを書いてしまうと、
5931 5*9*31 = 1395
これが答えです。総当たりで調べて条件に一致するものを求めます。

ざっくり、こんなプログラムでしょうね。

op_table = ["","+","-","*","/"]
for number in range(1000,10000):
    for op1 in op_table:
        for op2 in op_table:
            for op3 in op_table:
                if check(number,op1, op2, op3):
                    print(number,op1, op2, op3)

Python初心者作品

def Quinary(n):
    if (int(n/5)):
        return Quinary(int(n/5)) + str(n%5)
    return str(n%5)

def eval_change(a):
    b = list_change(a)
    try:
        return int(eval(b))
    except ZeroDivisionError:
        return -1
    except SyntaxError:
        return -1

def list_change(a):
    b = ""
    for l in a:
        b += l
    return b

table = ["","+","-","*","/"]
for i in range(1000,10000):
    for j in range(1,125):
        a = (Quinary(j).zfill(3))
        b = list(str(i))
        for k in range(3):
            b.insert(k*2+1,table[int(a[k])])
        if str(eval_change(b)) == str(i)[::-1]:
            print(i,list_change(b),"=",int(str(i)[::-1]))

おいおい。K君は、2重ループで解くという道を選びました。
かなり険しい道を選んだ感がありますが、よく解けたね。
Quinaryはネットで調べてもってきたそうです。演算子は5種類だから、5進数を使おうという発想。
この発想はなかった。

このプログラムの肝は、再帰使うことで、組み合わせを生成している所。
わざわざ5進数を持ち出してくるのはいただけないが、多重ループ使わなくても総当たりを生成できるんじゃない?
それにこの方法は、パラメータを変えれば何桁でも実現できる可能性を秘めている。

という訳で、平凡な4重ループ以外の方法でこの問題を解いてみます。

SyntaxErrorを防止

大改造する前に、気になる部分を修正

pythonでは、01 001など0数字の前に0があるとSyntaxErrorとなります。このため、eval("1+001")は、SyntaxErrorとなります。
K君は、このような場合は計算しない作戦をとりました。正解はこのパターンではなかったので、答えが求まりました。
正確には、総当たりになってないので、エラーにならないように(計算式を作る時に、不要な0を追加しないように)してました。
計算式を作った後で、正規表現で0削除する手もありそうですね。

料理開始

演算子の組み合わせ生成するジェネレータを再帰で作成。

gen_sequence

a0=[["a","b"],["A","B","C"]]

def gen_sequence(a):
    for i in a[0]:
        if(len(a)>1):
            for j in gen_sequence(a[1:]):
                yield([i]+j)
        else:
            yield([i])

for x in gen_sequence(a0):
    print(x)
実行結果
['a', 'A']
['a', 'B']
['a', 'C']
['b', 'A']
['b', 'B']
['b', 'C']

演算子の組み合わせをgen_sequenceで生成して、総当たりで条件に合うか調べます。
K君の計算式の文字列を生成する処理は、かなりトリッキーだったので、zipを使った処理に書き直して、関数化しました。

from itertools import zip_longest

op_list=[["","+","-","*","/"]]*3

def gen_sequence(a):
    for i in a[0]:
        if(len(a)>1):
            for j in gen_sequence(a[1:]):
                yield([i]+j)
        else:
            yield([i])

def build_formula(number_str, ops):
    formula = ""
    for num, op in zip_longest(number_str , ops, fillvalue=""):
        if len(formula)>1 and not formula[-2].isdigit() and formula[-1]=="0":
            # 演算子直後の0は削除する
            formula = formula[:-1]
        formula += num + op
    return formula

def eval_string(string):
    try:
        return eval(string)
    except ZeroDivisionError:
        return -1

for i in range(1000,10000):
    number_str = str(i)
    number_str_rev = number_str[::-1]
    for op in gen_sequence(op_list):
        if op ==[""]*3:
            continue
        formula = build_formula(number_str, op)
        if str(eval_string(formula)) == number_str_rev:
            print(i , formula , "=" , number_str_rev)

ループを使わないで総当たりの組み合わせを生成にこだわってみました。
2020/01/01 zip_longest()に変更

gen_sequence関数を作成しましたが、itertools.product(*op_list)がこれと等価でした。
この関数を作る事が目的なので、今回は作りましたが、この機能が必要な時は、itertools.productを使えばいいと思います。

1
1
4

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1