1

私は Python プログラミングの初心者で、現在の課題で問題が発生しました。課題は次のように書かれています...

  • プログラムは、数字が繰り返されていない 4 桁の数字 (または数字のみを含む 4 文字の長さの文字列) を秘密裏に生成する必要があります。
  • 次に、プログラムは、コードの推測を入力するようにユーザーに要求する必要があります。ユーザーの推測は、4 文字の長さで、数字のみで構成され、数字が繰り返されないようにする必要があります。プログラムは、スコアを付ける前に入力が有効であることを確認し、必要に応じてユーザーに入力の再入力を求める必要があります。無効な入力は、暗号解読者の推測回数にはカウントされません。コードは、入力の先頭または末尾の空白を無視 (削除) することができますが、内部の空白を不正な入力としてカウントする必要があります。
  • 各ターンで、ターン番号を印刷し、ユーザーの推測を取得します。入力が有効な場合、ターンの出力はユーザーの推測であり、その後にフィードバックが続きます。フィードバックは 4 文字の文字列です。正しい位置の各桁は「X」です。コード内にあるが正しい位置にない各桁の場合は「O」。「-」その他すべての場合。X と O は一緒にする必要があることに注意してください。フィードバックは、各タイプのペグの数だけであり、どのペグがどこにあるかではありません。
  • すべての推測とフィードバックの履歴を維持し、ユーザーがターンごとに印刷できるようにします。

今は課題の「フィードバック」部分に集中しています。私のコードは現在:

import random

def validateInput():
inputGuess = input("Enter your guess as 4 numbers:")

while True:
    if len(inputGuess) != 4:
       inputGuess = input("Enter your guess as 4 numbers:")
    else:
        numberList = list(inputGuess) ##

        invalidNumbers = False
        for number in numberList:
            if number not in ['1','2','3','4','5','6','7','8','9']:
                invalidNumbers = True 

        if invalidNumbers == True:
            print ("Possible numbers are 1, 2, 3, 4, 5, 6, 7, 8, 9.")
            inputGuess = input("Enter your guess as 4 numbers:") 

        else:
            return numberList

guessesRemaining=10

code=['1','2','3','4']



while guessesRemaining > 0:
     report=[]
     guess=validateInput()
     guessesRemaining -= 1
     if guess[0] == code[0]:
         report.append("X")
     if guess[1] == code[1]:
         report.append("X")
     if guess[2] == code[2]:
         report.append("X")
     if guess[3] == code[3]:
         report.append("X")

     tempCode=sorted(code)
     tempGuess=sorted(guess)

     if tempCode[0]==tempGuess[0]:
         report.append("O")
     if tempCode[1]==tempGuess[1]:
         report.append("O")
     if tempCode[2]==tempGuess[2]:
         report.append("O")
     if tempCode[3]==tempGuess[3]:
         report.append("O")

     report2=report[0:4]
     dash=["-","-","-","-"]
     report3=report2+dash
     report4=report3[0:5]



     print(report4)

たとえば、ユーザーが 1879 と推測し、コードが 1234 の場合、"XO --" を受信しますが、"X---" を受信したいと考えています。また、私のコードを合理化するためのアドバイスは素晴らしいでしょう. 簡単にするために、ここでは「ランダム コード」を [1,2,3,4] だけにしました。

4

3 に答える 3

4

Python 関数 map() を使用すると、問題を非常にエレガントに解決できます。(最初に思ったほどエレガントではありませんが、それでもかなりきれいです。)

guess = "1879" # or [ '1', '8', '7', '9' ]
answer = "1234"

map() は次のように機能します: 最初の引数として関数を指定し、それに続く引数として 1 つ以上のシーケンスを指定します。次に、その関数を取得して、最初の要素の両方に適用し、次に 2 番目の要素の両方に適用します。例えば:

>>> def f(a,b):
>>>  return a + b
>>> map( f, [1,2,3,4], [ 10, 20, 30, 40 ] )

[ 11, 22, 33, 44 ]

これで、"guess" と "answer" という 2 つの文字列ができました。等しい場合は X を返し、そうでない場合は次のように返す関数を作成できます。

>>> def mastermind_hint( a, b ):
>>>   return "X" if a == b else "-"

これだけでは十分ではありません。「O」も入力する必要があります。このためには、「回答」シーケンス全体を一度に使用する必要があります。

>>> def mastermind_hint( a, b ):
>>>   if a == b: return "X"
>>>   if a in answer: return "O"
>>>   return "-"

map を使用すると、それをシーケンスに適用できます。

>>> map( mastermind_hint, guess, answer )
[ "X", "-", "-", "-" ]

さて、ヒントの位置が推測文字の位置に対応しているため、現在、想定よりも多くの情報を提供しています。この情報を隠す簡単な方法は、シーケンスをソートすることです。Python には、これを行う sorted() 関数があります。

>>> sorted( map( mastermind_hint, guess, answer ) )
[ "-", "-", "-", "X" ]

あとは、これを 1 つの文字列に結合するだけです。

>>> "".join( sorted( map( mastermind_hint, guess, answer ) ) )
"---X"
于 2013-03-26T22:28:37.777 に答える
2

私が言わなければならない非常に興味深い宿題。このような宿題ができたらいいのに。

通常、私は宿題に完全な答えを提供しませんが、私は楽しみのためにこれを解決しました。あなたはこれを自分で解決しようとしたので、ここに行きます:

import random

def main():
    print '>> New game started.\n>> Good luck!\n'
    answer = generateAnswer()
    while True:
        userGuess = getUserGuess()
        if userGuess == answer:
            print '>> Congratulations, you won!'
            return
        print '>> The answer you provided is incorrect.\n>> Perhaps this hint will help you: '
        giveFeedback(answer, userGuess)

def generateAnswer():
    digits = [str(x) for x in range(10)]
    answer = ''
    for i in range(4):
        digit = random.sample(digits, 1)[0]
        digits.remove(digit)
        answer += digit
    return answer

def getUserGuess():
    while True:
        guess = raw_input('>> Please enter a 4-digit number: ').strip()
        if len(guess) != 4:
            continue
        guessIsValid = True
        for x in guess:
            if guess.count(x) != 1 or ord(x) not in range(48, 58):
                guessIsValid = False
                break
        if guessIsValid:
            return guess

def giveFeedback(answer, guess):
    for i in range(4):
        if guess[i] == answer[i]:
            print 'X',
            continue
        if guess[i] in answer:
            print 'O',
            continue
        print '-',
    print '\n'

if __name__ == '__main__':
    try:
        main()
    except Exception, e:
        print '>> Fatal error: %s' % e

このコードで答えが見つかることを願っています。

于 2013-03-26T22:44:14.823 に答える
1

これは、読みやすさと簡潔さを向上させるために、より高度な python イディオムを使用する例です。これにより、m 基数の n 桁の答えなど、より一般的な解を作成することも容易になります。

import random
from itertools import permutations   # for "lucky guess" Easter egg

def main(base=10, size=4):
    print 'New game started.\nGood luck!\n'
    answer, turn = list(genAnswer(range(base), size)), 1
    while True:
        hint = ['X' if a == g 
           else '0' if g in answer
           else '-' for a, g in zip(answer, guess(base, size))]
        if set(hint) == set(['X']):
            if turn == 1:
                choices = len(list(permutations(range(base), size)))
                print 'You won, lucky guess out of %d possible choices' % choices
            else:
                print 'Congratulations, you won in %d turns!' % turn
            return
        print '  Hint %d: %s' % (turn, ''.join(hint))
        turn += 1

def genAnswer(digits, size):
    '''Python generator function'''
    for i in range(size):
        choice = random.choice(digits)
        yield str(choice)
        digits.remove(choice)

def guess(base, size):
    '''Uses duck typing for error detection'''
    while True:
        guess = raw_input('>> Guess: ').strip()
        try:
            if int(guess, base) < base**size and len(set(guess)) == size:
                return guess
        except ValueError:
            pass
        print 'Enter %d unique digits from 0 to %d' % (size, base -1)

>>> main(6, 4)
New game started.
Good luck!

>> Guess: 1234
  Hint 1: 0-X-
>> Guess: 12345
Enter 4 unique digits from 0 to 5
>> Guess: 01227
Enter 4 unique digits from 0 to 5
>> Guess: 0123
  Hint 2: XX-0
>> Guess: 0135
Congratulations, you won in 3 turns!
于 2013-07-16T19:36:01.030 に答える