Rysowanie w ukośnikach

23

Napisz program, który pobiera prostokątny blok tekstu złożonego z X„i .”, na przykład A:

......
..XX..
.X..X.
.XXXX.
.X..X.
.X..X.

Wyjście reprezentacji tej siatki obróconej o 45 stopni w kierunku przeciwnym do ruchu wskazówek zegara poprzez narysowanie ukośnika - do przodu lub do tyłu w zależności od kontekstu - wszędzie na Xgranicy a .lub boku siatki (ze spacjami wypełniającymi resztę):

 /\/\
/ /\ \
\/ /  \
/\/ /\ \
\  /  \/
 \ \
  \ \
   \/

Liczba końcowych i wiodących spacji (i nowych linii) nie ma znaczenia, dopóki kształt Xliter na wejściu jest utrzymywany przez ukośniki. Można przycinać dodatkowe wiersze lub kolumny ..

Do We / Wy możesz użyć dowolnej kombinacji parametrów stdin / stdout / files / wiersza poleceń. Na przykład skrypt może przyjąć nazwę pliku X.wzoru i przepisać plik ukośnikiem. Albo twój skrypt może pobierać X.wzorzec od standardowej linii po linii (naciskając dpo zakończeniu) i wyświetlać wynik na standardowej linii .

Dane wejściowe mogą być dowolnie duże, ale można założyć, że są zawsze dobrze sformatowane.

Żadne znaki oprócz / \i znaki nowej linii nie powinny znajdować się w wynikach.

Punktacja

Najkrótszy kod w bajtach wygrywa. Użyj https://mothereff.in/byte-counter jako licznika bajtów, jeśli używasz znaków spoza ASCII.

Bonus: Minus 50 bajtów, jeśli możesz zmienić (zamienić, nie usunąć) jeden znak w swoim programie, dzięki czemu siatka ukośnika zostanie obrócona o 45 stopni zgodnie z ruchem wskazówek zegara , np .:

   /\/\
  / /\ \
 /  \ \/
/ /\ \/\
\/  \  /
    / /
   / /
   \/

Próbki wejściowe

XXXXXXXXX
X.......X
X.XXXXX.X
X.X...X.X
X.X.X.X.X
X.X...X.X
X.XXXXX.X
X.......X
XXXXXXXXX

XXX...XXX....XXX...XXX
X..X..X..X..X.....X...
XXX...XXX...X.....X.XX
X.....X.....X.....X..X
X.....X......XXX...XXX

XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX...X
X...X...............X...............X...........X...................X...X
X...X...XXXXXXXXX...X...XXXXX...XXXXXXXXX...XXXXX...XXXXX...XXXXX...X...X
X...............X.......X...X...........X...........X...X...X.......X...X
XXXXXXXXX...X...XXXXXXXXX...XXXXXXXXX...XXXXX...X...X...X...XXXXXXXXX...X
X.......X...X...............X...........X...X...X...X...X...........X...X
X...X...XXXXXXXXXXXXX...X...X...XXXXXXXXX...XXXXX...X...XXXXXXXXX...X...X
X...X...............X...X...X.......X...........X...........X.......X...X
X...XXXXXXXXXXXXX...XXXXX...XXXXX...X...XXXXX...XXXXXXXXX...X...XXXXX...X
X...........X.......X...X.......X...X.......X...........X...X...........X
X...XXXXX...XXXXX...X...XXXXX...X...XXXXXXXXX...X...X...X...XXXXXXXXXXXXX
X.......X.......X...X...X.......X.......X.......X...X...X.......X.......X
XXXXXXXXXXXXX...X...X...X...XXXXXXXXX...X...XXXXX...X...XXXXX...XXXXX...X
X...........X...X...........X.......X...X.......X...X.......X...........X
X...XXXXX...X...XXXXXXXXX...XXXXX...X...XXXXX...XXXXX...XXXXXXXXXXXXX...X
X...X.......X...........X...........X.......X...X...X...............X...X
X...X...XXXXXXXXX...X...XXXXX...XXXXXXXXX...X...X...XXXXXXXXXXXXX...X...X
X...X...........X...X...X...X...X...........X...............X...X.......X
X...XXXXXXXXX...X...X...X...XXXXX...XXXXXXXXX...XXXXXXXXX...X...XXXXXXXXX
X...X.......X...X...X...........X...........X...X.......X...............X
X...X...XXXXX...XXXXX...XXXXX...XXXXXXXXX...XXXXX...X...XXXXXXXXX...X...X
X...X...................X...........X...............X...............X...X
X...XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

( źródło labiryntu )

Powiązane pytania:
Narysuj krzywą Hilberta za pomocą ukośników
Wyrównaj wyrównany poziomo, prostokątny kształt ASCII wzdłuż przekątnej

Hobby Calvina
źródło
Czy zastąpienie pojedynczego znaku spacją (lub spacją do symbolu) jest dopuszczalne?
Przywróć Monikę - ζ--
@hexafraction Sure
Calvin's Hobbies
Potrzebujemy kogoś, kto odpowie na to pytanie w itflabtijtslwi . Wtedy mielibyśmy „Rysowanie ukośników w ukośnikach”.
Justin

Odpowiedzi:

10

Python 2 - 236, 228, 226, 221, 250, 248 246 - 50 = 196

s=input().split()
s=zip(*s[::-1])
X=len(s[0])+2
d=('.',)
B=[d*X]
s=B+[d+tuple(l)+d for l in s]+B
Y=len(s)
for x in range(X,-Y,-1):print' '*abs(x)+''.join(' \\'[s[y-1][x+y-1]!=s[y-1][x+y]]+' /'[s[y-1][x+y]!=s[y][x+y]]for y in range(1,Y)if 0<x+y<X)

Dodałem opcjonalną funkcję bonusową, która zmienia kierunek z zgodnego z ruchem wskazówek zegara na przeciwny. Zasadniczo druga linia po prostu obraca wejście o 90 stopni. Aby przełączać się między obiema opcjami, zamień s=zip(...)na coś podobnego S=zip(...)(np. Przypisz do zmiennej, która nie będzie ponownie używana).

.XX.X....X...XX.X.X
X...X...X.X.X...X.X
.X..X...XXX..X..XXX
..X.X...X.X...X.X.X
XX..XXX.X.X.XX..X.X

                 /\           /\/\ 
                 \ \          \/\ \ 
               /\ \ \         /\ \/  
               \ \/  \     /\ \/   /\ 
             /\ \  /\ \    \ \/\  / /  
            / /  \ \ \/     \/\/ / /    
            \/    \ \           / /      
            /\/\/\ \/          / /        
            \/\/\/             \ \    /\/\ 
        /\/\    /\              \ \  / /\/  
        \/\ \  / /               \/ /  \/\   
        /\/  \ \/                  / /\  /    
        \  /\ \                    \/ / / /\/\ 
         \ \ \/                      / /  \/\ \ 
   /\     \ \                        \/   /\ \/  
   \ \     \/                          /\ \/   /\ 
 /\ \ \  /\                            \ \/\  / /  
/ /  \ \/ /                             \/\/ / / /\ 
\/    \  /                                  /  \/ /
/\/\/\ \/                                  / /\  /
\/\/\/                                     \/ / /
    /\                                       / /
   / /                                       \/
   \/
Falko
źródło
Można zaoszczędzić 5bytes przez sandwiching sz Bo s=B+s+B.
BeetDemGuise
1
@BeetDemGuise: Dzięki! To o wiele łatwiejsze! :)
Falko,
7

MATLAB - 286-50 = 236

Zminimalizowane:

S=input('');x=rot90(reshape(input('','s'),S),1);S=size(x);D=sum(S);[r,c]=ind2sub(S,find(x==88));A=zeros(D);f=@(o,p)sub2ind([D D],r+c+o-1,c-r+S(1)+p);A(f(1,1))=1;t=f(0,0);A(t)=A(t)+1;A(f(1,0))=3;t=f(0,1);A(t)=A(t)+3;fprintf('%s',char([(1-mod(A,2))*32+(A==1)*47+(A==3)*92,ones(D,1)*13]'))

Rozszerzony:

S = input('');
x = rot90( reshape( input('','s'), S ), 1 );
S = size( x );
D = sum( S );
[r,c] = ind2sub( S, find( x==88 ) );
A = zeros( D );
f = @(o,p) sub2ind( [D D], r+c+o-1, c-r+S(1)+p );

A(f(1,1)) = 1;
t = f(0,0); A(t) = A(t)+1;
A(f(1,0)) = 3;
t = f(0,1); A(t) = A(t)+3;

fprintf( '%s', char( [(1-mod( A, 2 ))*32 + (A==1)*47 + (A==3)*92, ones( D, 1 )*13]' ) )

Kod można dodatkowo zmniejszyć o 6 znaków (w celu powiązania bieżącej potencjalnej szansy) poprzez wyeliminowanie rzutowania na typ char, ale powoduje to ostrzeżenie MATLAB i nie byłem pewien, czy było to dozwolone.

Można go zmniejszyć o dodatkowe 13 znaków, jeśli dane wejściowe muszą mieć format „zaznaczony”, na przykład ['X..';'.X.';'..X'], ale nie sądziłem, że było to dozwolone. Obecnie skrypt akceptuje tylko dwie liczby (wymiary wiersza i kolumny), a następnie odczytuje pojedynczy ciąg znaków siatki.

Premia

Zmieniając 1wywołanie x = rot90( ..., 1 );na a 0, transformacja zmienia się z obrotu w lewo o 45 ° na obrót w lewo o 45 °. W rzeczywistości możliwe są wszystkie możliwe obroty o 45 ° + n · 90 ° poprzez stopniowanie tego parametru od 0do 3.

Przykładowe dane wyjściowe:

XXX...XXX....XXX...XXX
X..X..X..X..X.....X...
XXX...XXX...X.....X.XX
X.....X.....X.....X..X
X.....X......XXX...XXX

                     /\    
                    / /    
                   / / /\  
                   \/ /  \ 
                   /\ \/\ \
                   \ \  / /
               /\   \ \/ / 
              / /    \/\/  
             / /           
             \/            
             /\    /\      
             \ \  / /      
              \ \/ /       
        /\/\   \/\/        
       / /\/               
      / / /\               
      \ \/ /               
       \  /                
        \ \                
  /\/\   \ \               
 / /\/    \/               
/ / /\                     
\ \/ /                     
 \  /                      
  \ \                      
   \ \                     
    \/                     

    /\                     
   /  \                    
  / /\ \                   
 /  \ \/                   
/ /\ \/\                   
\/  \/\/                   
          /\               
         /  \              
        / /\ \             
       /  \ \/             
      / /\ \/\             
      \/  \/\/             

               /\/\        
              / /\ \       
             / /  \ \      
             \/    \/      
             /\            
             \ \           
              \ \    /\/\  
               \/   / /\ \ 
                   / /  \ \
                   \/ /\ \/
                   /\ \ \  
                   \ \/ /  
                    \  /   
                     \/    
COTO
źródło
3

Perl - 409

while(<>){$a[$.]=[/./g]}$l=$#a<($g=@{$a[1]})?$#a:$g;$r=$#a+$g-1;for$d(1..$r){for$e(1..$l){if(($b=$a[$c=($d>$g?$d-$g:0)+$e][$n=-($d-$c+1)])&&$n<0){$h[$d][$e]=$b;}}$h[$d]=[(0)x(($l-{@$h[$d]})/2),@{$h[$d]}];}sub z{"\0"x((reverse(1..$g),(2..$r-$g+1))[int($q++/2)-1]).join'',map{$_ eq'X'?($q%2?'/\\':'\/'):"\0\0"}@_}@i="\0"x2x$l;for$f(@h){$i[$#i]=(z@$f)^$i[$#i];$i[$#i+1]=z@$f}for$j(@i){print$j=~s/[\0s]/ /gr."\n"}

Odczytuje ze standardowego i drukuje na standardowe.

Przykładowe wyniki:

XXX
X..
XXX
..X
XXX

     /\
    / / 
   / / /\
   \ \/  \
    \  /\ \
     \/ / /
       / /
       \/

XXXXXXXXX
X........
X.XXXXXXX
X.X.....X
X.X..X..X
X.XXXX..X
X...X...X
XXXXXXXXX
           /\
          / / 
         / / /\
        / / /  \
       / / / /\ \
      / / / /  \ \
     / / / /    \ \
    / / / / /\   \ \
   / / / /  \ \  / /
   \ \ \ \  / / / /
    \ \ \ \/  \/ /
     \ \ \  /\  /
      \ \ \/ / /
       \ \  / /
        \ \/ /
         \  /
          \/
faubi
źródło