Python 3、後で得点します
from collections import defaultdict
from functools import lru_cache
import sys
NUMERIC_OUTPUT = True
@lru_cache(maxsize=1024)
def to_befunge_num(n):
    # Convert number to Befunge number, using base 9 encoding (non-optimal,
    # but something simple is good for now)
    assert isinstance(n, int) and n >= 0
    if n == 0:
        return "0"
    digits = []
    while n:
        digits.append(n%9)
        n //= 9
    output = [str(digits.pop())]
    while digits:
        output.append("9*")
        d = digits.pop()
        if d:
            output.append(str(d))
            output.append("+")
    output = "".join(output)
    if output.startswith("19*"):
        return "9" + output[3:]
    return output
def translate(program_str):
    if program_str.count("(") != program_str.count(")"):
        exit("Error: number of opening and closing parentheses do not match")
    program = program_str.splitlines()
    row_len = max(len(row) for row in program)
    program = [row.ljust(row_len) for row in program]
    num_stacks = len(program)
    loop_offset = 3
    stack_len_offset = program_str.count("(")*2 + loop_offset
    stack_offset = stack_len_offset + 1
    output = [[1, ["v"]], [1, [">"]]] # (len, [strings]) for each row
    max_len = 1 # Maximum row length so far
    HEADER_ROW = 0
    MAIN_ROW = 1
    FOOTER_ROW = 2
    # Then stack lengths, then loop rows, then stacks
    # Match closing parens with opening parens
    loop_map = {} # {column: (loop num, stack number to check, is_start)}
    loop_stack = []
    loop_num = 0
    for col in range(row_len):
        col_str = "".join(program[stack][col] for stack in range(num_stacks))
        if col_str.count("(") + col_str.count(")") >= 2:
            exit("Error: more than one parenthesis in a column")
        if "(" in col_str:
            stack_num = col_str.index("(")
            loop_map[col] = (loop_num, stack_num, True)
            loop_stack.append((loop_num, stack_num, False))
            loop_num += 1
        elif ")" in col_str:
            if loop_stack:
                loop_map[col] = loop_stack.pop()
            else:
                exit("Error: mismatched parentheses")
    def pad_max(row):
        nonlocal max_len, output
        while len(output) - 1 < row:
            output.append([0, []])
        if output[row][0] < max_len:
            output[row][1].append(" "*(max_len - output[row][0]))
            output[row][0] = max_len
    def write(string, row):
        nonlocal max_len, output
        output[row][1].append(string)
        output[row][0] += len(string)
        max_len = max(output[row][0], max_len)
    def stack_len(stack, put=False):
        return (to_befunge_num(stack) + # x
                str(stack_len_offset) + # y
                "gp"[put])
    def get(stack, offset=0):
        assert offset in [0, 1] # 1 needed for 2-arity ops
        # Check stack length
        write(stack_len(stack) + "1-"*(offset == 1) + ":0`", MAIN_ROW)
        pad_max(HEADER_ROW)
        pad_max(MAIN_ROW)
        pad_max(FOOTER_ROW)
        write(">" + to_befunge_num(stack + stack_offset) + "g", HEADER_ROW)
        write("|", MAIN_ROW)
        write(">$0", FOOTER_ROW)
        pad_max(HEADER_ROW)
        pad_max(MAIN_ROW)
        pad_max(FOOTER_ROW)
        write("v", HEADER_ROW)
        write(">", MAIN_ROW)
        write("^", FOOTER_ROW)
    def put(stack, value=""):
        put_inst = (value +
                    stack_len(stack) +
                    to_befunge_num(stack + stack_offset) +
                    "p")
        post_insts.append(put_inst)
    def pop(stack):
        put(stack, "0")
    def inc_stack_len(stack):
        post_insts.append(stack_len(stack) + "1+")
        post_insts.append(stack_len(stack, put=True))
    def dec_stack_len(stack):
        post_insts.append(stack_len(stack) + ":0`-") # Ensure nonnegativity
        post_insts.append(stack_len(stack, put=True))
    # Technically not necessary to initialise stack lengths per spec, but it makes it
    # more portable and easier to test against other Befunge interpreters
    for stack in range(num_stacks):
        write("0" + stack_len(stack, put=True), MAIN_ROW)
    for col in range(row_len):
        post_insts_all = []
        loop_start = False
        loop_end = False
        if col in loop_map:
            if loop_map[col][2]:
                loop_start = True
            else:
                loop_end = True
        if loop_start:
            loop_row = loop_offset + 2*loop_map[col][0]
            get(loop_map[col][1])
        elif loop_end:
            get(loop_map[col][1])
            write("!", MAIN_ROW)
        for stack in range(num_stacks-1, -1, -1):
            char = program[stack][col]
            post_insts = [] # Executed after the gets in reverse order, i.e. last added first
            if char in " ()":
                continue
            # Pre-inc, post-dec
            elif char.isdigit():
                inc_stack_len(stack)
                put(stack, char)
            elif char == "?":
                inc_stack_len(stack)
                put(stack, "&")
            elif char == "!":
                get(stack)
                post_insts.append(".91+," if NUMERIC_OUTPUT else ",")
                pop(stack)
                dec_stack_len(stack)
            elif char == "#":
                pop(stack)
                dec_stack_len(stack)
            elif char in "+-":
                get(stack, 1)
                get(stack)
                post_insts.append(char)
                pop(stack) # This one first in case of ! or 1!
                post_insts.append(stack_len(stack) + ":1`-:1\\`+") # Ensure >= 1
                post_insts.append(stack_len(stack, put=True))
                put(stack)                
            elif char in "^v":
                offset = -1 if char == "^" else 1
                get((stack + offset) % num_stacks)
                inc_stack_len(stack)
                put(stack)
            else:
                exit("Error: invalid character " + char)
            post_insts_all.append(post_insts)
        while post_insts_all:
            write("".join(post_insts_all.pop()), MAIN_ROW)
        if loop_start or loop_end:
            loop_row = loop_offset + 2*loop_map[col][0]
            pad_max(HEADER_ROW)
            pad_max(MAIN_ROW)
            pad_max(loop_row)
            pad_max(loop_row + 1)
            write(">v", HEADER_ROW)
            write("|>", MAIN_ROW)
            if loop_start:
                write(" ^", loop_row)
                write(">", loop_row + 1)
            else:
                write("<", loop_row)
                write(" ^", loop_row + 1)
    write("@", MAIN_ROW)
    return "\n".join("".join(row) for row_len, row in output)
if __name__ == '__main__':
    if len(sys.argv) < 3:
        exit("Usage: py -3 prefunge.py <input filename> <output filename>")
    with open(sys.argv[1]) as infile:
        with open(sys.argv[2], "w") as outfile:
            outfile.write(translate(infile.read()))
のように実行しますpy -3 prefunge.py <input filename> <output filename>。
私にとっては遅い週だったので、ようやく退屈してこの6か月の質問に取り組むことができました。他に誰も試しなかった理由を尋ねますが、私はまだデバッグの苦痛を感じています(おそらく、私が知っているすべてのバグがまだ残っているでしょう)。
私が使用しての質問は、Befunge-93インタプリタを提供していませんこの1仕様は若干異なります。2つの重要な違いは次のとおりです。
- プログラムの特定の行にcharが存在しない場合、その行に書き込むことはできません。これは、最後に十分な改行を導入するためにEnterを数回押す必要があることを意味します。- NaN出力にs が表示される場合、これが最も可能性の高い原因です。
 
- グリッドセルはゼロに事前初期化されません-便宜上、Befunge出力に事前初期化をいくつか含めましたが、必要ではないので、スコアリングを開始するときにそれを削除することがあります。 
出力プログラムのコアレイアウトは次のとおりです。
v [header row]
> [main row]
  [footer row]
  ---
   |
   | rows for loops (2 per loop)
   |
  ---
  [stack length row]
  ---
   |
   | rows for stack space (1 per voice)
   |
  ---
スタックスペースはプログラムの外側にあるため、以前の改行Enter-spammingコメントがあります。
コアとなるアイデアは、各ボイスにスタックとして機能する行を割り当てることです。これらのスタックを維持するために、各スタックの長さが行に沿ってセルに記録される特別なスタック長の行もあります。その場合、プログラムは多くのgETとpUTになります。たとえば、プロセスを印刷する場合は次のとおりです。
- セルを取得 y = stack_row[stack], x = stack_length[stack]
- 実行.91+,、つまり整数として印刷してから改行を印刷する
- 上記の座標のセルを0に置き換えます(ポップをシミュレートするため)
- デクリメント stack_length[stack]
列の同時評価を実行するには、必要なすべてのセルが読み取られ、セルに書き込まれる前にその値がスタックに保持されます(たとえば、印刷の例では、最初と2番目のステップの間にさらに命令がある場合があります)。
`これは、スタックの長さが負にならないようにするため、およびスタックが空のときに0をプッシュするために使用されます。これははっきりと見える分岐の元ですが、分岐を削除するアイデアがあります。これにより、1行目と3行目から大量の空白が削除されます。
ループの場合、Preludeループは両方の方法でジャンプできるため、次のような構成ではループごとに2行を使用します。
       >v                     >v
(cond) |>  (program)  (cond) !|>
        ^                     <
       >                       ^
現在、これらのループはバイトの大部分を占めていますが、コードボックスに配置することで簡単に見つけることができます。これは、トランスレーターがp正常に動作していることがわかってから行う予定です。
ここではいくつかのための出力例です?(1-^!)つまり、印刷、n-1まで0:
v                        >6gv>v                      >6gv      >6gv                                 >6gv                   >6gv                           >6gv >v
>005p05g1+05p&05g6p05g:0`|  >|>05g1+05p105g6p05g1-:0`|  >05g:0`|  >-005g6p05g:1`-:1\`+05p05g6p05g:0`|  >05g1+05p05g6p05g:0`|  >.91+,005g6p05g:0`-05p05g:0`|  >!|>@
                         >$0^                        >$0^      >$0^                                 >$0^                   >$0^                           >$0^
                              ^                                                                                                                                <
                             >                                                                                                                                  ^
入力の平方:
v                                >8gv      >8gv             >v      >6gv                                   >8gv      >8gv        >7gv      >7gv                                                            >8gv >v      >7gv
>005p015p025p25g1+25p&25g8p25g:0`|  >25g:0`|  >05g1+05p05g6p|>05g:0`|  >15g1+15p15g7p25g1+25p125g8p25g1-:0`|  >25g:0`|  >15g1-:0`|  >15g:0`|  >+015g7p15g:1`-:1\`+15p15g7p-025g8p25g:1`-:1\`+25p25g8p25g:0`|  >!|>15g:0`|  >.91+,015g7p15g:0`-15p@
                                 >$0^      >$0^                     >$0^                                   >$0^      >$0^        >$0^      >$0^                                                            >$0^         >$0^
                                                             ^                                                                                                                                                  <
                                                            >                                                                                                                                                    ^
分割(小さな入力が推奨されます):
v                                                                          >91+gv>v      >94+gv                                                         >95+gv      >95+gv        >93+gv      >93+gv                                                                    >93+gv      >93+gv               >v      >93+gv                                                     >93+gv >v      >92+gv                  >v      >92+gv                                       >92+gv                                       >91+gv                                       >93+gv                     >91+gv                       >92+gv      >92+gv        >91+gv      >91+gv                                                                                      >92+gv >v                        >91+gv      >91+gv                                     >91+gv >v                        >95+gv      >95+gv                                     >95+gv
>009p019p029p039p049p09g1+09p109g91+p29g1+29p&29g93+p39g1+39p&39g94+p09g:0`|    >|>39g:0`|    >009g91+p09g:0`-09p29g1+29p29g93+p49g1+49p149g95+p49g1-:0`|    >49g:0`|    >29g1-:0`|    >29g:0`|    >-029g93+p29g:1`-:1\`+29p29g93+p+049g95+p49g:1`-:1\`+49p49g95+p29g:0`|    >29g:0`|    >19g1+19p19g92+p|>29g:0`|    >09g1+09p109g91+p19g1+19p19g92+p29g1+29p029g93+p29g:0`|    >!|>19g:0`|    >029g93+p29g:0`-29p|>19g:0`|    >09g1+09p09g91+p019g92+p19g:0`-19p19g:0`|    >019g92+p19g:0`-19p29g1+29p29g93+p09g:0`|    >009g91+p09g:0`-09p19g1+19p19g92+p29g:0`|    >19g1+19p19g92+p09g:0`|    >19g1+19p19g92+p19g1-:0`|    >19g:0`|    >09g1-:0`|    >09g:0`|    >-009g91+p09g:1`-:1\`+09p09g91+p+019g92+p19g:1`-:1\`+19p19g92+p029g93+p29g:0`-29p19g:0`|    >!|>09g1+09p109g91+p09g1-:0`|    >09g:0`|    >+009g91+p09g:1`-:1\`+09p09g91+p09g:0`|    >!|>49g1+49p149g95+p49g1-:0`|    >49g:0`|    >-049g95+p49g:1`-:1\`+49p49g95+p49g:0`|    >.91+,049g95+p49g:0`-49p@
                                                                           >$0  ^        >$0  ^                                                         >$0  ^      >$0  ^        >$0  ^      >$0  ^                                                                    >$0  ^      >$0  ^                       >$0  ^                                                     >$0  ^         >$0  ^                          >$0  ^                                       >$0  ^                                       >$0  ^                                       >$0  ^                     >$0  ^                       >$0  ^      >$0  ^        >$0  ^      >$0  ^                                                                                      >$0  ^                           >$0  ^      >$0  ^                                     >$0  ^                           >$0  ^      >$0  ^                                     >$0  ^
                                                                                  ^                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        <
                                                                                 >                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          ^
                                                                                                                                                                                                                                                                                                          ^                                                                        <
                                                                                                                                                                                                                                                                                                         >                                                                          ^
                                                                                                                                                                                                                                                                                                                                                                                                                    ^                                                                                                                                                                                                                                                                                                                                              <
                                                                                                                                                                                                                                                                                                                                                                                                                   >                                                                                                                                                                                                                                                                                                                                                ^
そこに思い浮かぶ他のマイナーの最適化の束が交換するように、まただ07p07gと:07p、私は一度、この一歩を取っています:)