Ruszamy się spod kontroli, doktorze!

11

Doktor, próbując uciec przed siłami Daleków, postanowił wysłać ich w wir, podróżując spiralnie po różnych kieszeniach przestrzeni.

W zależności od charakteru dostępnej czasoprzestrzeni, Doktor musi wejść do TARDIS kontroluje wysokość i szerokość odcinka przestrzeni oraz punkt wejścia, od którego rozpoczyna się spirala.

Przekrój przestrzeni może być pomyślane jako h x wag siatki wypełnionej kolejnych liczb całkowitych od lewej do prawej, z góry do dołu, począwszy od 1.

Pozycja początkowa jest podana jako rc dla wiersza i kolumny ... Z tego oprogramowania TARDIS musi wyrzucić uporządkowaną listę liczb całkowitych uzyskanych przez spiralizację na zewnątrz w kierunku przeciwnym do ruchu wskazówek zegara z rzędu r kolumny c , zaczynając w górę ...

Twoim zadaniem, jako towarzyszowi Doktora, jest zaprogramowanie TARDIS do przyjmowania czterech liczb w formacie height width row columni określania, który sektor przestrzeni musi podróżować TARDIS, aby pasował do ruchu spiralnego opisanego poniżej ...

Wejście 1

5 5 3 3

(5 x 5 pól, zaczynając od pozycji 3,3)

Wyjście 1

13 8 7 12 17 18 19 14 9 4 3 2 1 6 11 16 21 22 23 24 25 20 15 10 5

Wyjaśnianie wyników

Oryginalna siatka wprowadź opis zdjęcia tutaj

Wygenerowana spirala wprowadź opis zdjęcia tutaj

Wejście 2

2 4 1 2

(Siatka 2 x 4, zaczynając od pozycji 1,2)

Wyjście 2

2 1 5 6 7 3 8 4

Wyjaśnianie wyników

Nieco inna niż spirala musi teraz krążyć wokół siatki, aby wygenerować odpowiedni wynik ...

Oryginalna siatka wprowadź opis zdjęcia tutaj

Wygenerowana spirala wprowadź opis zdjęcia tutaj

Zasady:

  1. To jest gra w golfa, więc najkrótsza długość kodu jest akceptowana.

  2. Powyższych przykładów należy użyć do przetestowania kodu. Jeśli nie zapewnia odpowiedniego wyniku, coś jest nie tak ...

  3. W odpowiedzi należy podać wersje kodu zarówno do gry w golfa, jak i gry w golfa ...

Powodzenia!

WallyWest
źródło
Chciałbym wskazać wam draw.io, w którym można szybko zrobić dość rozsądne rysunki (masz doskonałą czytelność dzięki ręcznie rysowanej wersji ... po prostu nie widzę żadnych czerwonych kółek). Rozważ i.stack.imgur.com/xbLSA.png jako przykład tego, co można zrobić. Zauważ, że ma w nim xml, więc jeśli przejdziesz do draw.io, możesz zaimportować z adresu URL.
Będę o tym pamiętać przy mojej kolejnej potrzebie rysowania, @MichaelT, dziękuję ...
WallyWest
1
Wysyłam odpowiedź z funkcją zwracającą tablicę jako wynik. Czy jest to akceptowane?
edc65
@ edc65 Mate, ty i ja wracamy tutaj na CG, pozwolę na funkcję S (h, w, r, c) lub tym podobne ... :)
WallyWest

Odpowiedzi:

3

JavaScript (ES6) 124 163 177

Edytuj Zupełnie inny sposób, bez potrzeby przechowywania tablicy do przechowywania odwiedzonych komórek. Wykorzystując fakt, że bok spirali rośnie o 1 po każdych 2 obrotach.

// New way
f=(h,w,y,x)=>
  (e=>{
    for(o=[],d=i=t=l=0;l<w*h;i<t?i+=2:[i,d,e]=[1,-e,d,++t])
      o[l]=y*w-w+x,l+=x>0&x<=w&y>0&y<=h,x+=d,y-=e
  })(1)||o


// Golfed
g=(h,w,y,x)=>
  (g=>{
    for(e=n=0;n<h*w;)g[[n%w+1,-~(n/w)]]=++n;
    for(o=[g[[x,y]]],l=d=1;l<n;l+=!!(o[l]=g[[x+=d,y+=e]]))
      g[[x,y]]=0,
      g[[x+e,y-d]]!=0&&([d,e]=[e,-d])
  })([])||o



// Not golfed
u=(h,w,y,x)=>{
  var i,j,dx,dy,kx,ky,o,n,
    g={} // simulate a 2dimensional array using a hashtable with keys in the form 'x,y'

  for(n=i=0; i++<h;) // fill grid (probably better done in a single loop)
    for(j=0; j++<w;)
      g[[j,i]] = ++n;
  o=[g[[x,y]]] // starting point in output
  dx=1, dy=0 // start headed right
  
  for(; !o[w*h-1]; ) // loop until all w*h position are put in output
  {
    g[[x, y]] = 0 // mark current position to avoid reusing
    kx=dy, ky=-dx // try turning left
    if(g[[x+kx, y+ky]] != 0) // check if position marked
    { // found a valid position
      dx=kx, dy=ky // change direction
    }
    x+=dx, y+=dy // move
    k=g[[x, y]] // get current value
    if (k) o.push(k) // put in output list if not 'undefined' (outside grid)
  }
  return o
}

// TEST - In FireFox

out=x=>O.innerHTML+=x+'\n';
[
 [[5,5,3,3],'13 8 7 12 17 18 19 14 9 4 3 2 1 6 11 16 21 22 23 24 25 20 15 10 5'],
 [[2,4,1,2],'2 1 5 6 7 3 8 4']
].forEach(t=>out(t[0] + '\n Result: ' + f(...t[0])+'\n Check:  ' + t[1]))

test=()=>
{
  var r, i=I.value.match(/\d+/g), h=i[0]|0, w=i[1]|0, y=i[2]|0, x=i[3]|0
  if (y>h||x>w) r = 'Invalid input'
  else r = f(h,w,y,x)
  out(i+'\n Reault: ' +r)
}
<pre id=O></pre>
Your test:<input id=I><button onclick="test()">-></button>

edc65
źródło
Niesamowite gra w golfa! Od 300 do 163 ...
Zdejmuję
1
@WallyWest z tym komentarzem popychasz mnie do robienia lepiej. Thnx
edc65
Ładny! Moje rozwiązanie Python było znacznie dłuższe, ale pomyślałem, że wszystko w porządku, używasz lepszej metody. Teraz używasz tego samego i jest jeszcze krótszy ... Mam trochę pracy. :)
randomra
@randomra Nadal chciałbym to zobaczyć ...
WallyWest
2

Python 3, 191

Prawdopodobnie nie jest to świetny wynik, ale oto:

def f(b,a,d,c):
 p,r,l,s,h=c+1j*d,-1j,1,0,0
 for _ in [0]*((a+b)**2):x,y=p.real,p.imag;0<x<a+1and 0<y<b+1and print(int((y-1)*a+x),end=' ');p+=r;s=(s+1)%l;t=s==0;h=(h+t)%2;l+=h<t;r*=(-1j)**t 

Poruszamy się wzdłuż spirali, zwiększając długość boku co sekundę. Jeśli nasza pozycja znajduje się wewnątrz podanej siatki, wypisujemy odpowiedni numer.

Zmienne to:

  • p jest złożoną pozycją
  • xiy są współrzędnymi pozycji
  • r jest kierunkiem
  • s jest pozycją po bieżącej stronie
  • l to aktualna długość boku
  • h jest parzystością liczby porządkowej bieżącej strony
randomra
źródło