ランレングスレーサー


18

2つの入力が与えられます:ランニングトラックを定義するランレングスエンコード形式の文字列と、開始するレーンを表す大文字。たとえば、文字列「3a4A6b5B」は「aaaAAAAbbbbbbBBBBB」に展開されます。次に、展開された文字列を使用して、次のようなトラックを作成します。

 A) aaaAAAA
 B) bbbbbbBBBBB

これは2車線のトラックです。小文字は空気を表します。オンエアはできません!大文字は、あなたが走れる道を表します。このチャレンジの目標は、大文字を与えられて、そのレーンでスタートするレーサーがどこまで走れるかを出力することです。レーサーは、直上または直下に道路がある場合、車線を切り替えることができます。また、逆方向への実行も許可されています!この特定のトラックでは、いずれのトラックも位置1の走行可能な道路がないため、文字入力の出力は0です。

例:

入力:「4A5B4c3C」、「A」

このコードは、次のようなトラックに展開されます。

A) AAAA
B) BBBBB
C) ccccCCC

この例の出力は7です。これは、レーンAで始まるランナーがレーンBに移動し、次にレーンCに移動して、7番目の位置で終わる可能性があるためです。

入力:「4A2B3D」、「D」

追跡:

A) AAAA
B) BB
C)
D) DDD

出力は3です。これは、レーンDで始まるランナーがレーンBまたはAに到達する方法がないためです。

入力:「4A4a4A3b6B5C」、「A」

追跡:

A) AAAAaaaaAAAA
B) bbbBBBBBB
C) CCCCC

AのランナーはBに切り替えて、最後にAに戻ることができるため、出力は12です。「C」の最大距離も12です。「B」の場合、0です。

入力:「12M4n10N11O」、「M」

追跡:

M) MMMMMMMMMMMM
N) nnnnNNNNNNNNNN
O) OOOOOOOOOOO

複数桁のランレングスを使用した簡単な例。出力は14です。

入力:「4A5B1b2B4c3C」、「A」

追跡:

A) AAAA
B) BBBBBbBB
C) ccccCCC

出力は8です。これは、AのランナーがBに移動し、次にCに移動してからBに戻ることができるためです(この例についてはFryAmTheEggmanに感謝します)。

入力:「1a2A2a2B1c1C1d3D」、「B」

追跡:

A)aAAaa
B)BB
C)cC
D)dDDD

出力は4です。ランナーは、どちらが先に進むかを確認するために両方のパスをチェックする必要があります。(この例のuser81655に感謝します。)

入力:「2A1b1B2C1D3E」、「A」

追跡:

A) AA
B) bB
C) CC
D) D
E) EEE

出力は3です。あなたは最も遠い目的地に到達するために逆走する必要があります。(もう一度、この例のuser81655に感謝します。)

ノート:

  • トラックの特定の位置に文字がない場合、それも空気としてカウントされます。そのため、入力が「Q」で、車線「Q」に道路が配置されていない場合、出力は0になります
  • 入力には2つの部分があります。最初は、ランレングスでエンコードされた文字列です。2番目は大文字です(文字列またはcharデータ型を使用できます)。読みやすくするために、これらの入力(スペース、改行、タブ、コンマ、セミコロン)の間に適切な区切り文字が必要です。
  • ランレングスでエンコードされた文字列は、常にアルファベット順に要素をリストします
  • レーンの全長が可能な最長の長さは1000です。したがって、可能な最大出力は1000です。

トラックジェネレーター:

最初の答えに敬意を表して、ここにトラックジェネレーターがあります。現在の答えを断つために何かを考え出してみてください!(注:ジェネレーターにエラーメッセージが表示されないからといって、トラックコードが必ずしも有効であるとは限りません。適切な形式については上記の例を参照してください。)

function reset() {
    var t = document.getElementById("track");
    t.innerHTML = "";
    for(var i = 0;i<26;i++) {
      var c = String.fromCharCode(i+65);
      t.innerHTML += "<div><span>"+c+") </span><span id='"+c+"'></span></div>";
      
    }
  }

function rand() {
  var track = "";
  for(var i = 0;i<26;i++) {
  var blocks = Math.floor(Math.random()*4);
  var start = Math.floor(Math.random()*2);
  for(var j = 0;j<blocks;j++) {
    var letter = String.fromCharCode(65+i+32*((start+j)%2));
    var length = Math.floor(Math.random()*4)+1;
    track += length+letter;
  }
  }
  document.getElementById("code").value = track;
}

  function gen() {
  var s = document.getElementById("code").value;
    var check = s.match(/(\d+[A-Za-z])+/);
    if(check == null || check[0]!=s) {
      alert("Invalid Track");
      return false;
    }
    reset();
  var n = s.match(/\d+/g);
    var o = s.match(/[A-Za-z]/g);
    for(var i = 0;i<n.length;i++) {
      var c = o[i].toUpperCase();
      document.getElementById(c).textContent += o[i].repeat(n[i]);
    }
    return true;
    }
<body onload="reset()">
Track: <input type="text" id="code" size="75%" /><input type="submit" onclick="gen()" /><input type="button" value="Random Track" onclick="rand()" /><code id="track"/>
  </body>


3
スイッチの決定と逆走により、現在のトラックよりも迷路のようになりました。P
user81655

テストケースのように、ルートは1つしかありませんか?
RichieAHB

@RichieAHB複数のルートが存在する可能性があります。
geokavel

欠落しているCを処理する複雑4A2B3Dさを取り除くことができるのではないかと疑問に思いましたか?たとえば、0c?そうでない場合、say 1A1Zが与えられたときに予想されますか?レーンBYは存在すると仮定されます(しかし空です)?
ケニー

1
また、逆走は大きな問題です。12M4n10N11O例えば、出力14は、次に偽である:最長経路は、25の長さのために、M0及びC0で終わりから始まり
ケニー

回答:


3

Perlの、231の 219 203 192 189バイト

+1を含む -p

sub f{my($l,$p,$m)=@_;map{$m=$_>$m?$_:$m}f($l,$p+1)+1,f($l-1,$p),f($l+1,$p),f($l,$p-1)-1if$L[$l][$p]&&!$V{$l}{$p}++;$m}s/(\d+)(.)\s*/push@{$L[ord$2&~32]},(0|$2lt'a')x$1;()/ge;$_=0|f(ord,0)

少ないゴルフ:

sub f{                          # this is a recursive function, so we need locals.
    my($l,$p,$m)=@_;            # in: lane, position; local: max path length

    map{
      $m = $_ > $m ? $_ : $m    # update max
    }
    f( $l,   $p+1 )+1,          # same lane, forward
    f( $l-1, $p   ),            # left lane, same pos
    f( $l+1, $p   ),            # right lane, same pos
    f( $l,   $p-1 )-1           # same lane, backtrack
    if
        $L[$l][$p]              # check if there's road here
    && !$V{$l}{$p}++            # and we've not visited this point before.
    ;

    $m                          # return the max
}

s/(\d+)(.)\s*/                  # Parse RLE pattern, strip starting lane separator
  push@{ $L[ord$2&~32] }        # index @L using uppercase ascii-code, access as arrayref
  ,(0|$2lt'a')x$1               # unpack RLE as bitstring
  ;()                           # return empty list for replacement
/gex;                           # (x for ungolfing)
                                # $_ now contains trailing data: the start lane.

$_ =                            # assign output for -p
   0|                           # make sure we print 0 instead of undef/nothing
   f(ord,0)                     # begin calculation at start of current lane

ランニング

上記コードをファイルに保存します(たとえば231.pl)。の形式で入力し(\d+\w)+ *\wます。例:トラック4A5B4c3Cとレーンの入力A

echo 4A5B4c3C A | perl -p 231.pl

TestSuite

(ゴルフではない)

printf "==== Testing %s\n", $file = shift // '231.pl';

sub t{
    my($input,$expect) = @_;
#   $input =~ s/\s//g;
    printf "TEST %-20s -> %-3s: ", $input, $expect;

    $output = `echo $input | perl -p $file`;

    printf "%-3s  %s\n", $output,
    $output == $expect
    ? " PASS"
    : " FAIL: $output != $expect";

}

t("4A5B4c3C A", 7);
t("4A5B4c3C C", 0);
t("4A2B3D D", 3);
t("4A4a4A3b6B5C A", 12);
t("4A4a4A3b6B5C B",  0);
t("4A4a4A3b6B5C C", 12);
t("12M4n10N11O M", 14 );
t("4A5B1b2B4c3C A", 8);
t("1a2A2a2B1c1C1d3D B", 4 );
t("2A1b1B2C1D3E A", 3 );
t("10A9b1B8c2C9D1E11F A", 11);
  • 更新219は、配列インデックスを作り直して12バイトを節約します。
  • 更新203再帰をリファクタリングして16バイトを節約します。
  • 192更新すると、@L=map{[/./g]}@L後処理が不要になり11バイト節約されます。
  • 189更新すると、の代わりにifを使用して3バイトを節約できます。mapfor

これがPerlのものかどうかはわかりませんが、これは高速に実行されます。
geokavel

6

JavaScript(ES6)、298 334バイト

(t,s)=>[a=[],t.match(/\d+(.)(\d+\1)*/gi).map(l=>a[c=l.match`[A-Z]`+"",n=c.charCodeAt(),c==s?i=n:n]=l[r="replace"](/\d+./g,p=>(p.slice(-1)<"a"?"1":"0").repeat(parseInt(p))),i=o=-1),...a.join``,a[i]?a[i]=a[i][r](/^1/,2):0].map(_=>a.map((l,y)=>a[y]=l[r](/1/g,(c,x)=>((a[y-1]||s)[x]|(a[y+1]||s)[x]|l[x-1]|l[x+1])>1?(x>o?o=x:0,2):c)))&&o+1

説明

基本的に、このソリューションはトラックを迷路として扱います。ランナーが到達できるすべてのタイルがどこにあるかを見つけ、見つかったXインデックスの最大値を返します。

最初に行うことは、入力文字列を行の配列にデコードすることです。ただし、文字を使用する代わりに1、大文字をa に、小文字をaに変換し0ます。結果のマップは次のようになります。

11100011
0011100
100111

この後、開始トラックaの最初のタイルを2(既にある場合のみ1)作成し、隣接するタイルaをチェックするすべてのタイルをループします2。に1隣接2している場合、になります2。ランナーが最初の行で開始した場合、上記のマップは次のようになります。

22200011
0022200
100222

aの最高のXインデックス2が結果になります。

私はこれの最初のバージョンを作ったときに非常にマイナーな監視をしました、そしてそれが機能するまでそれをハックするのに36バイトを要しました。*はぁ*

非ゴルフ

(t,s)=>
  [

    // Decode run-length encoded string into an array of track lanes
    a=[],                           // a = array of track line strings, 0 = air, 1 = tiles
    t.match(/\d+(.)(\d+\1)*/gi)     // regex magic that separates pairs by their letter
    .map(l=>                        // for each line of pairs
      a[                            // add the tiles to the array
        c=l.match`[A-Z]`+"",        // c = pair character
        n=c.charCodeAt(),           // n = index of line
        c==s?i=n:n                  // if this line is the starting line, set i
      ]=l[r="replace"](/\d+./g,p=>  // match each pair, p = pair
        (p.slice(-1)<"a"
          ?"1":"0").repeat(         // repeat 0 for air or 1 for ground
            parseInt(p)             // cast of match would return NaN because of the
          )                         //     letter at the end but parseInt works fine
      ),
        i=                          // i = index of starting line, initialise as invalid
          o=-1                      // o = output (max value of x)
    ),

  // Find all positions that are possible for the runner to get to
    ...a.join``,                   // add every letter of the track lines to an array
    a[i]?a[i]=a[i][r](/^1/,2):0    // set the starting tile to 2 if it is already 1
  ].map(_=>                        // loop for the amount of tiles, this is usually way
                                   //     more than necessary but allows for hard to reach
                                   //     tiles to be parsed
    a.map((l,y)=>                  // for each line l at index y
      a[y]=l[r](/1/g,(c,x)=>       // for each character c at index x

        // Replace a 1 with 2 if there is a 2 to above, below, left or right of it
        ((a[y-1]||s)[x]|(a[y+1]||s)[x]|l[x-1]|l[x+1])>1?
          (x>o?o=x:0,2):c          // set o to max value of x for a 2 tile
      )
    )
  )
  &&o+1                            // return o + 1

テスト

おまけ:出力には解析されたマップが含まれます!

var solution = (t,s)=>[a=[],t.match(/\d+(.)(\d+\1)*/gi).map(l=>a[c=l.match`[A-Z]`+"",n=c.charCodeAt(),c==s?i=n:n]=l[r="replace"](/\d+./g,p=>(p.slice(-1)<"a"?"1":"0").repeat(parseInt(p))),i=o=-1),...a.join``,a[i]?a[i]=a[i][r](/^1/,2):0].map(_=>a.map((l,y)=>a[y]=l[r](/1/g,(c,x)=>((a[y-1]||s)[x]|(a[y+1]||s)[x]|l[x-1]|l[x+1])>1?(x>o?o=x:0,2):c)))&&o+1
function generateMap() { var start = 0; a.some((l, i) => l ? start = i : 0); var end = 0; a.map((l, i) => l && i <= 90 ? end = i : 0); for(var output = "", i = start; i < end + 1; i++) output += String.fromCharCode(i) + ") " + (a[i] || "") + "\n"; return output; }
Track = <input type="text" id="track" value="2A1b1B2C1D3E" /><br />
Starting Letter = <input type="text" id="start" value="A" /><br />
<button onclick="result.textContent=solution(track.value,start.value)+'\n\n'+generateMap()">Go</button>
<pre id="result"></pre>

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.