Przełączanie grawitacji

14

Wyzwanie

Otrzymujesz reprezentację ASCII-art znaków na płaszczyźnie jako dane wejściowe dowolną rozsądną metodą. Będzie to zawierać tylko:

  • [a-z]reprezentujących ruchome postacie. Każda litera pojawi się na planszy maksymalnie raz.
  • # reprezentujące nieruchome ściany
  • . reprezentujący pustą przestrzeń

Na przykład:

abcdef.gh#..
.......ij.#.
#..#.......#
...#.#...###
.#.......#q#
.........###

Otrzymujesz również ciąg reprezentujący zmiany grawitacji. Będzie to zawierać tylko:

  • > reprezentujący zmianę grawitacji w prawo
  • < reprezentujący zmianę grawitacji w lewo
  • ^ reprezentujący zmianę grawitacji w górę
  • v reprezentujący zmianę grawitacji w dół

Na przykład:

v>^

Twój program musi symulować kolejno każdą zmianę grawitacji, aż wszystkie postacie przestaną się poruszać (uderzą w ścianę lub inną postać). Postacie, które „spadają z krawędzi mapy” są trwale usuwane, a postacie mogą „ustawiać się” jeden na drugim.

W tym przykładzie, na początku jest grawitacyjnie do dołu (w v), tak, c, e, g, h, i, i jspadają do dolnej części mapy. Wszystkie pozostałe postacie przesuwają się w dół, aż uderzą w ścianę, pozostawiając mapę w ten sposób:

.........#..
a..d......#.
#..#.f.....#
.b.#.#...###
.#.......#q#
.........###

Następnie przechodzimy do grawitacji w prawo ( >), która pozostawia nam to: Zwróć uwagę na astosy obok d.

.........#..
........ad#.
#..#......f#
..b#.#...###
.#.......#q#
.........###

Wreszcie możemy symulować grawitację w górę ( ^), w którym aa bodpadać mapie.

.........#..
.........d#.
#..#......f#
...#.#...###
.#.......#q#
.........###

Twoim zadaniem jest wyprowadzenie pozostałych postaci po przesunięciu grawitacyjnym. Można je podać w dowolnej kolejności. W tym przykładzie można wyprowadzić dowolną permutacjędfq .

Przypadki testowe

Dla następującej mapy:

abcde
.....
##.##
v   =  abde
v>  =  <nothing>

Dla następującej mapy:

######
#....#
abcdef
#.gh..
######
>   = <nothing>
<   = gh
^>  = bcde
v<  = bghef
jrich
źródło
Jak podaje się dane wejściowe? Lista ciągów znaków? Argument funkcji? STDIN?
Leaky Nun
@KennyLau Wszystkie te opcje są w porządku. Wejście i wyjście może być cokolwiek rozsądnego dla twojego języka.
jrich
Nieco spokrewniony.
Martin Ender,

Odpowiedzi:

4

JavaScript (ES6), 251 233 bajtów

(m,d,r=`replace`)=>[...d].map(c=>[...`<^>v`].map(d=>m=[...(m=(c==d?m[r](/[.\w]+/g,s=>s[r](/\./g,``)+s[r](/\w/g,``))[r](/^\w+/gm,s=>s[r](/./g,`.`)):m).split`
`)[0]].map((_,i)=>m.map(s=>s[i]).join``).reverse().join`
`))&&m[r](/\W/g,``)

Edycja: Zapisano 18 bajtów dzięki @WashingtonGuedes.

Działa poprzez czterokrotne obrócenie siatki wejściowej dla każdego znaku kierunkowego, ale w kierunku, w którym znak kierunkowy pasuje do znaku pętli, wykonujemy lewą grawitację. Pseudo kod:

function slideleft(map) {
    map = map.replace(/[.\w+]/g, match=>movedotstoend(match));
    map = map.replace(/^\w+/gm, match=>changetodots(match));
}
function rotate(map) {
    return joinrows(reverse([for each (column of rows(map)[0])
            joinrow([for each (row of rows(map)) row[column]])
           ]));
}
function gravity(map, directions) {
    for each (direction of directions) {
        for each (angle of '<^>v') {
            if (direction == angle) map = slideleft(map);
            map = rotate(map);
        }
    }
    return letters(map);
}
Neil
źródło
3

JavaScript (ES6), 199

Ten sam algorytm odpowiedzi @ Neil. Siatka jest obracana czterokrotnie dla każdego znaku kierunkowego, gdy we właściwej pozycji przesunięcie grawitacyjne w lewo jest stosowane do każdego rzędu.

x=>y=>(x=>{for(d of y){R='';for(e of'^>v<')x=[...x[0]].map((c,i)=>e!=d?x.map(r=>r[i]).join``:x.map(r=>(c=r[i])<'.'?(q+=p+c,p=''):c>'.'&&q?(q+=c,R+=c):p+='.',q=p='')&&q+p).reverse();}})(x.split`
`)||R

F=x=>y=>(x=>{for(d of y){R='';for(e of'^>v<')x=[...x[0]].map((c,i)=>e!=d?x.map(r=>r[i]).join``:x.map(r=>(c=r[i])<'.'?(q+=p+c,p=''):c>'.'&&q?(q+=c,R+=c):p+='.',q=p='')&&q+p).reverse();}})(x.split`
`)||R

// Less golfed

U=(x,y)=>
{
  x = x.split`\n`;
  for(d of y)
  {
    R = '';
    for(e of '^>v<')
      x = [...x[0]].map( 
        (c,i) => e != d
        ? x.map( r => r[i] ).join`` // basic rotation
        : x.map( // rotation with gravity shift
          r=> (c=r[i])<'.' ? (q+=p+c,p='') : c>'.'&&q?(q+=c,R+=c) : p+='.', q=p=''
        ) && q+p
      ).reverse();
  }
  return R
}

console.log=x=>O.textContent+=x+'\n'

;[
  ['abcdef.gh#..\n.......ij.#.\n#..#.......#\n...#.#...###\n.#.......#q#\n.........###',
   [['v>^','dfq']]
  ],
  ['abcde\n.....\n##.##',[['v','abde'],['v>','']]],
  ['######\n#....#\nabcdef\n#.gh..\n######',[['>',''],['<','gh'],['^>','bcde'],['v<','befgh']]]
].forEach(t => {
  var i=t[0]
  console.log(i)
  t[1].forEach(([d,k])=>{
    var r=F(i)(d),ok=[...r].sort().join``==k
    console.log((ok?'OK ':'KO ')+d+' : '+r+(ok?'':' (expected '+k+')'))
  })
  console.log('')
})
<pre id=O></pre>

edc65
źródło
2

Pyth, 143 bajty

(Czy naprawdę potrzebujemy tyle bajtów?)

JhQKeQMe.u:N+"\."H+H"."G;DybVkI}NG=gkN;D'bVlJ=k@JNykVkI}HG=:kH\.).?B)=XJNk;VKIqN\<'J)IqN\v=C_J'J=_CJ)IqN\>=C_CJ'J=C_CJ)IqN\^=CJ'J=CJ;VJVNI}HGpH

Wypróbuj online!

Jak to działa

Definiujemy funkcję, leftktóra wykonuje lewą grawitację.

Następnie inne kierunki są implementowane przez bałagan z tablicą, tak że pożądany kierunek jest w lewo, a następnie wykonaj left.

Algorytm leftjest tutaj:

  • Wykonaj następujące czynności, aż do wystąpienia idempotent:
  • Wymień ".X"się "X.", gdzie Xreprezentuje literę.

Cały program jest podzielony na 6 następujących sekcji:

JhQKeQ
Me.u:N+"\."H+H"."G;
DybVkI}NG=gkN;
D'bVlJ=k@JNykVkI}HG=:kH\.).?B)=XJNk;
VKIqN\<'J)IqN\v=C_J'J=_CJ)IqN\>=C_CJ'J=C_CJ)IqN\^=CJ'J=CJ;
VJVNI}HGpH

Pierwsza sekcja

JhQKeQ     Q auto-initialized to evaluate(input())
JhQ        J = Q[0]
   KeQ     K = Q[len(Q)-1]

Druga sekcja

Me.u:N+"\."H+H"."G;   @memoized
M                 ;   def g(G,H):
  .u             G      repeat_until_idempotent(start:G, as N):
    :Nxxxxxxyyyyy         return N.replace(xxxxxx,yyyyy)
      +"\."H                               "\."+H
            +H"."                                 H+"."

Trzecia sekcja

DybVkI}NG=gkN;    @memoized
Dyb          ;    def y(b):
   Vk               for N in k:     --TAKES GLOBAL VARIABLE k
     I}NG             if N in G:    --G pre-initialized to "abcde...z"
         =gkN           k = g(k,N)

Czwarta sekcja

D'bVlJ=k@JNykVkI}HG=:kH\.).?B)=XJNk;  @memoized
D'b                                ;  def single_quote(b):
   VlJ                       )          for N in range(len(J)): --TAKES GLOBAL VARIABLE J
      =k@JN                               k = J[N] --SETS GLOBAL VARIABLE k
           yk                             y(k) --returns nothing, but MODIFIES GLOBAL VARIABLE k
             Vk                           for H in k:
               I}HG      )                  if H in G:
                   =:kH\.                     k = k.replace(H,".")
                          .?                else:
                            B                 break
                              =XJNk     J[N] = k --MODIFIES GLOBAL VARIABLE J

Część piąta

VKIqN\<'J)IqN\v=C_J'J=_CJ)IqN\>=C_CJ'J=C_CJ)IqN\^=CJ'J=CJ;
VK                                                       ; for N in K:
  IqN\<  )                                                   if N == '<':
       'J                                                      single-quote(J)
          IqN\v          )                                   if N == 'v':
               =C_J                                            J = transpose(reverse(J))
                   'J                                          single-quote(J)
                     =_CJ                                      J = reverse(transpose(J))
                          IqN\>            )                 if N == '>':
                               =C_CJ                           J = transpose(reverse(transpose(J)))
                                    'J                         single-quote(J)
                                      =C_CJ                    J = transpose(reverse(transpose(J)))
                                            IqN\^            if N == '^':
                                                 =CJ           J = transpose(J)
                                                    'J         single-quote(J)
                                                      =CJ      J = transpose(J)

Sekcja szósta

VJVNI}HGpH
VJ           for N in J:
  VN           for H in N:
    I}HG         if H in G: --G = "abc...z"
        pH         print(H)
Leaky Nun
źródło
1

Rubinowy, 306 bajtów

Funkcja anonimowa. Całkiem okrężna technika, którą prawdopodobnie można zoptymalizować.

->s,d{w=[];x=y=0;m={}
s.chars.map{|c|c>?!?(c<?A?c<?.?w<<[x,y]:0:m[c]=[x,y]
x+=1):(x=0;y+=1)}
d.chars.map{|c|q=[c<?=?-1:c<?A?1:0,c>?a?1:c>?A?-1:0];e=1
(n=m.clone.each{|k,v|z=[v[0]+q[0],v[1]+q[1]]
w.index(z)||m.value?(z)?0:m[k]=z}
m.reject!{|k,v|i,j=v;i<0||i>=x||j<0||j>y}
e=(m!=n ?1:p))while e}
m.keys.join}
Wartość tuszu
źródło