Sześcian tekstu ݀


17

Ostatnim razem zrobiłeś kwadrat tekstu , ale czy teraz możesz zrobić sześcian tekstu?

Wyzwanie

Biorąc pod uwagę ciąg, wypisz ciąg w postaci kostki.

Możesz założyć, że ciąg będzie zawsze zawierał 2 znaki lub więcej i będzie zawierał tylko znaki ascii do wydrukowania.

Jak zrobić kostkę tekstową

terrible mspaint skill.png

Przypadki testowe

Input:
Test

Output:
   Test
  e  ss
 s  e e
tseT  T
s  e e
e  ss
Test

Input:
Hello, world!

Output:
            Hello, world!
           e           dd
          l           l l
         l           r  r
        o           o   o
       ,           w    w

     w           ,      ,
    o           o       o
   r           l        l
  l           l         l
 d           e          e
!dlrow ,olleH           H
d           e          e
l           l         l
r           l        l
o           o       o
w           ,      ,

,           w    w
o           o   o
l           r  r
l           l l
e           dd
Hello, world!

Input:
Hi

Output:
 Hi
iHH
Hi

Implementacja referencji w Pythonie

text = raw_input("Enter a string: ")

print " " * (len(text) - 1) + text

spaces = len(text) - 2
_spaces = spaces

for i in range(1, len(text) - 2 + 1):
    print " " * spaces + text[i] + " " * _spaces + text[(i + 1) * -1] + " " * (_spaces - spaces) + text[(i + 1) * -1]
    spaces -= 1

print text[::-1] + " " * _spaces + text[0]

spaces = _spaces - 1

for i in range(1, len(text) - 2 + 1):
    print text[(i + 1) * -1] + " " * _spaces + text[i] + " " * spaces + text[i]
    spaces -= 1

print text

Zasady

  • To jest , więc wygrywa najkrótsza odpowiedź w bajtach! Tiebreaker jest najbardziej pozytywnie oceniany.
  • Standardowe luki są niedozwolone.
  • Końcowe znaki nowej linii i końcowe spacje są dozwolone.

Czy dozwolone są spacje końcowe?
Neil

@Neil Tak. (15 znaków)
acrolith

Ciekawe, co jest z dodatkowymi postaciami w tytule wyzwania?
AdmBorkBork

@TimmyD „Kostka tekstu” ma 14 znaków, tytuł musi mieć co najmniej 15 znaków, więc dodałem małą kropkę. Myślę, że to ten.
acrolith

Ach, okej. Na moim komputerze pojawia się jako znacznie większy krąg w IE, stąd moje pytanie.
AdmBorkBork

Odpowiedzi:


2

Pyth, 78 bajtów

AtBtlQJ_SH
+*dGQVJs[*Nd@Q-GN*dHK@QN*d-HNK;
++_Q*dHhQVJs[@QN*dH@Q-GN*dtN@Q-GN;

Z końcowym znakiem nowej linii. Zainspirowany odpowiedzią Joshua de Haana na Python 3.

Wypróbuj online tutaj!


4

Python 2, 228 223 221 203 199 195 189

t=input()
x=" "
l=len(t)-1
q=l-1
f=range(q,0,-1)
print x*l+t
for i in f:print x*i+t[l-i]+x*q+t[i]+x*(q-i)+t[i]
print t[::-1]+x*q+t[0]
for i in f:print t[i]+x*q+t[l-i]+x*(i-1)+t[l-i]
print t

Python 3, 192 188 182

t=input()
x=" "
l=len(t)-1
q=l-1
p=print
f=range(q,0,-1)
p(x*l+t)
for i in f:p(x*i+t[l-i]+x*q+t[i]+x*(q-i)+t[i])
p(t[::-1]+x*q+t[0])
for i in f:p(t[i]+x*q+t[l-i]+x*(i-1)+t[l-i])
p(t)

To 203 bajty. Możesz także zapisać 4 bajty, zastępując raw_input()je input().
acrolith

Masz rację, dzięki!
Joshua de Haan,

x*(q)? Powinieneś być w stanie usunąć pareny, tak?
Wartość tuszu

Masz rację, głuptasie;) Naprawiam to teraz haha
Joshua de Haan

x*(i-1)->x*~-i
mbomb007

3

Kod maszynowy x86 (IA-32), 126 bajtów

Hexdump:

60 8b f9 57 33 c0 f2 ae 5e 2b fe 4f 87 fa 8d 1c
12 8b c3 48 f6 e3 c6 04 07 00 48 c6 04 07 20 75
f9 8b ea 4d 53 8d 04 2a 50 53 8b c5 f6 e3 8d 44
68 01 50 53 2b c2 8b c8 50 4b 53 55 53 03 c5 50
f7 d3 53 50 53 95 f6 e2 6b c0 04 50 43 53 51 6a
01 4a 52 6a 01 50 6a ff 51 b0 0a 6a 0b 8b dc 59
8b 6c cb fc 88 04 2f 03 2c cb 89 6c cb fc 83 f9
0a 75 01 ac e2 ea 4a 79 e0 83 c4 58 61 c3

To jest trochę długie, więc aby to wyjaśnić, najpierw dam kod C:

void doit(const char* s, char out[])
{
    int n = strlen(s);
    int w = 2 * n;
    int h = w - 1;
    int m = n - 1;

    memset(out, ' ', h * w);
    out[h * w] = 0;

    int offset1 = n + m;
    int offset2 = w * m + 2 * m + 1; // 2 * n * n - 1
    int offset3 = offset2 - n; // 2 * n * n - n - 1
    int offset4 = 4 * n * m; // 4 * n * n - 4 * n

    int offsets[] = {
        offset3, -1,
        offset4, 1,
        m, 1,
        offset3, 1 - w,
        offset4, -w,
        offset2 - 1, -w,
        offset2 - 1, w - 1,
        m, w - 1,
        offset3, w,
        offset2, w,
        offset1, w,
    };

    do
    {
        char c = *s++;
        for (int i = 0; i < 11; ++i)
        {
            if (i == 9)
                c = '\n';
            int offset = offsets[i * 2];
            assert(offset > 0 && offset < w * h);
            out[offset] = c;
            offsets[i * 2] += offsets[i * 2 + 1];
        }
    } while (--n);
}

Oto ndługość ciągu wejściowego.

Wymiary obszaru wyjściowego to 2n(szerokość) przez 2n-1(wysokość). Po pierwsze, wypełnia wszystko spacjami (i dodaje kończący bajt zerowy). Następnie przesuwa się wzdłuż 11 linii prostych w obszarze wyjściowym i wypełnia je tekstem:

  • 2 linie są wypełnione bajtami końca linii (= 10)
  • 9 wierszy jest wypełnionych kolejnymi bajtami ciągu wejściowego

Każda linia jest reprezentowana przez dwie liczby, początkowe przesunięcie i krok. Umieściłem je oba w tablicy offsets, aby dostęp był „łatwy”.

Interesującą częścią jest wypełnienie tablicy. Niewielkie znaczenie ma kolejność wpisów w tablicy; Próbowałem zmienić ich kolejność, aby zminimalizować liczbę konfliktów rejestrów. Ponadto formuły kwadratowe mają pewną swobodę w wyborze sposobu obliczania; Próbowałem zminimalizować liczbę odejmowań (ponieważ dodatki mogą być implementowane przez elastycznyLEA instrukcję).

Źródło zestawu:

    pushad;

    ; // Calculate the length of the input string
    mov edi, ecx;
    push edi;
    xor eax, eax;
    repne scasb;
    pop esi; // esi = input string
    sub edi, esi;
    dec edi;

    ; // Calculate the size of the output area
    xchg edi, edx;  // edx = n
                    // edi = output string
    lea ebx, [edx + edx]; // ebx = w
    mov eax, ebx;
    dec eax; // eax = h
    mul bl; // eax = w * h

    ; // Fill the output string with spaces and zero terminate it
    mov byte ptr [edi + eax], 0;
myfill:
    dec eax;
    mov byte ptr [edi + eax], ' ';
    jnz myfill;

    mov ebp, edx;
    dec ebp; // ebp = m

    ; // Fill the array of offsets
    push ebx; // w
    lea eax, [edx + ebp];
    push eax; // offset1
    push ebx; // w
    mov eax, ebp;
    mul bl;
    lea eax, [eax + 2 * ebp + 1];
    push eax; // offset2
    push ebx; // w
    sub eax, edx;
    mov ecx, eax; // ecx = offset3
    push eax; // offset3
    dec ebx;
    push ebx; // w - 1
    push ebp; // m
    push ebx; // w - 1
    add eax, ebp;
    push eax; // offset2 - 1
    not ebx;
    push ebx; // -w
    push eax; // offset2 - 1
    push ebx; // -w
    xchg eax, ebp; // eax = m
    mul dl;
    imul eax, eax, 4;
    push eax; // offset4
    inc ebx;
    push ebx; // 1 - w
    push ecx; // offset3
    push 1;
    dec edx; // edx = n - 1
    push edx;
    push 1;
    push eax;
    push -1;
    push ecx;

    ; // Use the array of offsets to write stuff to output
myout:
    mov al, '\n';
    push 11;
    mov ebx, esp;
    pop ecx;
myloop:
    mov ebp, [ebx + ecx * 8 - 4];
    mov [edi + ebp], al;
    add ebp, [ebx + ecx * 8];
    mov [ebx + ecx * 8 - 4], ebp;
    cmp ecx, 10;
    jne skip_read;
    lodsb;
skip_read:
    loop myloop;
    dec edx;
    jns myout;

    add esp, 11 * 8;

    popad;
    ret;

Użyłem tu mnożenia bajtów, ograniczając długość ciągu wejściowego do 127. Pozwala to uniknąć zapełniania rejestru edx- produkt jest obliczany wax zamiast tego .

Drobna usterka: podczas wypełniania tablicy długość łańcucha zmniejsza się o 1. Więc dostosowałem warunek wyjścia pętli:

    jns myout

Odlicza się do -1.


3

Ruby, 148 144 bajtów

+1 bajt z nflagi. Wyświetla znaki nowej linii zamiast średników dla czytelności (ta sama funkcjonalność).

S=" "
X=S*s=$_.size-2
puts X+S+I=$_,(r=1..s).map{|i|c=I[~i];S*(s-i+1)+I[i]+X+c+S*~-i+c},I.reverse+X+I[0],r.map{|i|c=I[i];I[~i]+X+c+S*(s-i)+c},I

Biegnij tak. Dane wejściowe to wiersz STDIN, bez końca nowej linii, więc prawdopodobnie należy go potokować z pliku.

ruby -ne 'S=" ";X=S*s=$_.size-2;puts X+S+I=$_,(r=1..s).map{|i|c=I[~i];S*(s-i+1)+I[i]+X+c+S*~-i+c},I.reverse+X+I[0],r.map{|i|c=I[i];I[~i]+X+c+S*(s-i)+c},I'

1

JavaScript, 225 198 bajtów

Zaoszczędź 27 bajtów dzięki @Neil

f=(s,l=s.length-2,d=' ',r='repeat',t=d[r](l))=>[d+t+s,...a=[...Array(l)].map((_,i)=>d[r](l-i)+s[i+1]+t+(p=s[l-i])+d[r](i)+p),[...s].reverse().join``+t+s[0],...a.map(v=>v.trim()).reverse(),s].join`
`
  • [...] zamiast .concat
  • [...] + mapa zamiast dla pętli
  • tylko jedna instrukcja, przenosząc zmienne jako parametry funkcji
  • lepsza inicjalizacja dla l i t

Oryginalna odpowiedź:

f=s=>{l=s.length,d=' ',r='repeat',a=[],t=d[r](l-2)+s;for(i=1;i++<l-1;)a.push(d[r](l-i)+s[i-1]+d[r](l-2)+(p=s[l-i])+d[r](i-2)+p);console.log(d+[t].concat(a,[...t].reverse().join``+s[0],a.map(v=>v.trim()).reverse(),s).join`
`)}

1
Fajny, choć golfowy: (s,l=s.length-2,d=' ',r='repeat',t=d[r](l))=>[d+t+s,...a=[...Array(l)].map((_,i)=>d[r](l-i)+s[i+1]+t+(p=s[l-i])+d[r](i)+p),[...s].reverse().join``+t+s[0],...a.map(v=>v.trim()).reverse(),s].join`\n`(przy użyciu\n ponieważ nie można wstawiać nowych linii w komentarzach).
Neil

0

Java 7, 283 bajtów

void a(String s){int h=s.length(),n=h*2-1,t=n-h,u=n-1;char[][]c=new char[n][n];for(int i=0;i<h;i++){c[0][t+i]=c[i][t-i]=c[t][t-i]=c[t+i][t]=c[t+i][u-i]=c[t-i][t+i]=c[t-i][u]=c[u][i]=c[u-i][0]=s.charAt(i);}for(int y=0;y<n;y++){System.out.println(new String(c[y]).replace('\0',' '));}}

Wypróbuj tutaj!

Nie golfowany:

void a(String s) {
    int length=s.length(),
        n=length*2-1,
        mid=n-length,
        doubleMid=n-1;
    char[][]c=new char[n][n];
    for(int i=0;i<length;i++) {
        c[0][mid+i]= 
        c[i][mid-i]=
        c[mid][mid-i]=
        c[mid+i][mid]=
        c[mid+i][doubleMid-i]=
        c[mid-i][mid+i]=
        c[mid-i][doubleMid]=
        c[doubleMid][i]=
        c[doubleMid-i][0]=s.charAt(i);
    }
    for(int y=0;y<n;y++){
        System.out.println(new String(c[y]).replace('\0',' '));
    }
}
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.