Нарисуйте «Cool S»


38

Введение

Мы все знаем крутой S (также известный как Супермен S, Stüssy S, Super S, Skater S, Pointy S, Graffiti S и т. Д.): Миллиарды школьников по всему миру нарисовали этот S и сразу почувствовали гордость за себя. Если вы забыли или у вас было совсем не крутое детство , вот изображение этой классной буквы S:

Учитывая масштабный коэффициент в nкачестве входных данных (где 1n20 ), выведите Cool S в ASCII art.

Как нарисовать это

Со страницы Википедии на Cool S:

Выход

Cool S когда n= 1 это:

   ^
  / \
 /   \
/     \
|  |  |
|  |  |
\  \  /
 \  \/
 /\  \
/  \  \
|  |  |
|  |  |
\     /
 \   /
  \ /
   v

А для разных значений nвы просто увеличиваете время вывода n. Например, n= 2:

     ^  
    / \
   /   \
  /     \
 /       \
/         \
|    |    |
|    |    |
|    |    |
|    |    |
\    \    /
 \    \  /
  \    \/
  /\    \
 /  \    \
/    \    \
|    |    |
|    |    |
|    |    |
|    |    |
\         /
 \       /
  \     /
   \   /
    \ /
     v

Обратите внимание, что вертикальные секции в два раза длиннее, а расстояние между вертикальными линиями в два раза шире.

И когда n= 3:

       ^
      / \
     /   \
    /     \
   /       \
  /         \
 /           \
/             \
|      |      |
|      |      |
|      |      |
|      |      |
|      |      |
|      |      |
\      \      /
 \      \    /
  \      \  /
   \      \/
   /\      \
  /  \      \
 /    \      \
/      \      \
|      |      |
|      |      |
|      |      |
|      |      |
|      |      |
|      |      |
\             /
 \           /
  \         /
   \       /
    \     /
     \   /
      \ /
       v

Примечание. Хотя это и не обязательно, ваш код также может поддерживатьn= 0:

 ^
/ \
\\/
/\\
\ /
 v

выигрыш

Самая короткая программа в байтах побеждает.



Парень из 90-х, строящий ASCII, хочет предложить / \ вместо ^ в качестве подсказки. Так выглядит чище, плюс он сохраняет тот же наклон склона :)
Флатер

Единственная проблема @Flater в том, что / \ использует два символа, поэтому центральная вертикальная линия должна быть смещена, что делает ее очень неопрятной
Beta Decay

@BetaDecay: выглядит хорошо при N = 2 и N = 3 (поскольку сохраняет точечную симметрию), но я согласен с N = 1. Также есть вариант перевернутой Λ
буквы

2
@JacobGarby: Мой аргумент был стилистическим, а не гольфистским :)
Флатер

Ответы:


14

Древесный уголь , 58 53 47 43 41 байт

Nθ≔⊕⊗θδ↗θ/⊗θ↘δ^‖B↓‖M← vMδ⁰⊗θ↗⊕θM⁰δ↗θ/⊗θ⟲T

Попробуйте онлайн!

Я просто хотел попробовать другой подход, он рисует внешнюю сторону с помощью отражений (спасибо Нейлу за расширение идеи), а затем рисует внутреннюю часть. Поскольку Charcoal имеет :Leftнаправление по умолчанию для рисования линий, я максимально использую это направление, чтобы сэкономить несколько байтов, рисуя S горизонтально, например:

     /----\    /----\     
    /      \  /      \    
   /        \/        \   
  /         /          \  
 /         /            \ 
v     ----/    /----     ^
 \            /         / 
  \          /         /  
   \        /\        /   
    \      /  \      /    
     \----/    \----/     

И тогда мне просто нужно повернуть холст на 90 градусов против часовой стрелки.


Вы можете быть на что-то там ... 22 байта дает вам все снаружи ...
Нил

@ Нил, это было не совсем так, ваша идея нуждалась в небольшом исправлении, но действительно это было большое улучшение!
Чарли

Да, я допустил аналогичную ошибку в своем исходном сообщении, потому что не правильно проверял эффект масштабирования.
Нил

Кто-то сказал Rotate? Это дает мне представление ...
Нил

@ Нил, эй, у тебя там довольно хорошее улучшение! :-)
Чарли

13

Python 3 , 255 249 248 209 байт

-6 байт благодаря Кевину Круйссену

-1 байт благодаря Кевину Круйссену

-39 байтов благодаря Роду и Джо Кингу

n=int(input())
m=2*n
a,b,q,c,l='\ \n/|'
f=m*b
s=q+q.join([f[d:]+c+b*2*d+b+a+f[d:]for d in range(m+1)]+[l+f+l+f+l]*m+[d*b+a+f+a+f[d*2:]+c+d*b for d in range(n)]+[n*b+a+f+a+c+n*b])
print(f,'^'+s+q+s[::-1]+f,'v')

Попробуйте онлайн!

Теперь он обрабатывает n = 0.


И o+~dто, m-dи другое может быть и range(o)может быть range(m+1), а затем вы можете удалить, o=m+1\nчтобы сохранить 6 байтов. Хороший ответ, хотя +1 от меня.
Кевин Круйссен

1
Да, и еще один байт, изменив p(s)\np(s[::-1])на p(s+q+s[::-1]): 248 байтов
Кевин Круйссен

Вы можете сэкономить 6 байт, если вы используете один print, и больше 4, удалив []из join([...]), что составляет 238 байт
Род

Вы также можете сохранить q.joinв переменной, чтобы сохранить байт
Род

217 . Присоединились все q.joins и пара других вещей
Джо Кинг,

13

Древесный уголь , 47 42 41 байт

Fv^«↓⊗θ↘⊗⊕θ←↓⊗θ↙⊕⊗θ↖ι↖⊕⊗θ→↑⊗θ↗⊕θMθ⁺⊗θ⊕θ⟲⁴

Попробуйте онлайн! Ссылка на подробную версию кода. Объяснение: Рисует следующие линии по порядку:

   ^
  / \
 /   \
/     \
|  1  |
|  1  |
\  2  /
 \  2/
 8\  2
8  \  2
7  |  3
7  9  3
6     4
 6   4
  6 4
   5

Где 5текущий символ строки v^. В конце первого цикла курсор помещается в точку 9. Затем весь холст вращается, так что можно нарисовать другую половину Cool S. (Холст фактически поворачивается дважды, но это только деталь реализации.)

Древесный уголь не поддерживает, RotateCopy(:Up, 4)но если бы это было так, это работало бы на 33 байта:

↖^↖⊕⊗θ→↑⊗θ↗⊕θ‖BM↓↙⊗θ→↓⊗θ⟲C↑⁴J⁰¦⁰v

@BetaDecay К сожалению об этом. В любом случае, у меня тоже был неправильный счетчик байтов ...
Нил

Хорошо, он тоже получает n = 0
Beta Decay

6

Холст , 36 32 29 байт

«|*‼l├/L1^╋;╶╵\∔∔│α╶«├:╵╋:↔↕∔

Попробуй это здесь!

Много манипуляций со стеком. (устаревшее) объяснение:

«|*                                an array of input*2 "|"s
   ‼                               cast to a 2D object (needed because bug)
    :                              duplicate that (saved for the center line)
     l├                            height+2
       /                           create a diagonal that long
        L1^╋                       and in it, at (width; 1) insert "^"
            ;∔                     append the vertical bars
                               ^
                              /
          so far done:       / 
                            /  
                            |  
                            |  
              ⁸╵                   input+1
                \                  antidiagonal with that size
                 ∔                 appended to the above
                  │                mirror horizontally
                              ^
                             / \
                            /   \
                           /     \
                current:   |     |
                           |     |
                           \     /
                            \   /                                                       |
                   α               get the 2nd to last popped thing - the antidiagonal  |
                    └∔             append it to the vertical line copied way before:    \
                      ⁸«├          input/2 + 2                                            \
                         :╵        duplicate + 1
                           ╋       at (input/2 + 2; input/2 + 3) in the big part insert  ^
                            :↔↕∔   mirror a copy vertically & horizontally and append that to the original


3

C (gcc) , 379 353 344 334 байта

Я использовал пару #defines для исключения подвыражений и несколько глобальных переменных для связи между внутренними функциями. Основной цикл идет {0,1,2,3,3,2,1,0} для построения S.

Спасибо Джонатану Фреху за предложения.

#define z(a,b...)printf("%*c%*c%*c\n"+a,b);}
#define y(a){for(i=~-a*t;v*i<v*a*!t+t;i+=v)
i,n,p,r,t,u,v;a(){z(6,r+2,94+t*24)b()y(-~r)z(3,-i-~r,47+u,i*2+2,92-u)c()y(r)z(0,~r,124,~r,124,~r,124)d()y(-~n)z(0,i+1,92-u,2*(n-t*i)+1,92,2*(n-!t*i)+1,47+u)(*x[])()={a,b,c,d};f(s){r=2*s;for(p=0;p<8;x[7*t-p++*(2*t-1)](n=s))t=p>3,v=2*!t-1,u=t*45;}

Попробуйте онлайн!


w -r-1может быть в гольф w~r.
Джонатан Фрех

Хотя тогда вставка на один байт короче .
Джонатан Фрех



3

C (gcc) , 260 254 байта

-6 байт благодаря потолку .

f(n){int s=2*n++,t=s+1,I[]={1,t,s,n,n,s,t,1},A[]={s,1,1,1,2*t,1,t,t,1,t,1,n,t,t,1,t,t,1,1,1,t,s,1,1},x;for(s=8;s--;)for(n=0;n<I[s];n++,puts(""))for(t=3;t--;)x=s*3+t,printf("%*c",n*("AAAA?BAAAAC@?ABAAACA@AAA"[x]-65)+A[x],"w!!!0]}}}]]00]]}}}]!0_!!"[x]-1);}

Попробуйте онлайн!

Спуститься

Мы можем разделить форму на части:

 ^           Top cap
/ \          Top slope
|||          Sides
\\/          Twist, part 1
/\\          Twist, part 2
|||          Sides
\ /          Bottom slope
 v           Bottom cap

Каждая часть может быть описана числом строк, тремя символами и тремя связями с определенными значениями, которые определяют ширину поля в каждой строке.

Первой итерацией стало:

#define g(x,s,A,B,C)for(i=0;i<x;i++)printf("%*c%*c%*c\n",A,*s,B,s[1],C,s[2]);
f(n)
{
    int s=2*n++,t=s+1,i;

    g(1,  "  ^",  1,      1,  t-1)
    g(t, "/ \\",t-i,      1,2*i+1)
    g(s,  "|||",  1,      t,    t)
    g(n,"\\\\/",i+1,      t,t-2*i)
    g(n,"/\\\\",n-i,  2*i+1,    t)
    g(s,  "|||",  1,      t,    t)
    g(t, "\\/ ",i+1,2*t-2*i,    1)
    g(1,  "  v",  1,      1,  t-1)
}

Вызовы к g()макросу очень похожи на то, что таблицу можно построить и зациклить. Ширина поля иногда связана с индексом счетчика, а иногда нет. Мы можем обобщить поле ширины, чтобы быть F * i + A, где F - некоторый фактор, чтобы умножиться i, и A - некоторое значение, чтобы добавить к ширине. Таким образом, последняя ширина четвертого вызова выше будет -2 * i + t, например.

Таким образом мы получаем:

f(n){int s=2*n++,t=s+1,         s = size of "side" parts, t = size of top and bottom slopes
I[]={1,t,s,n,n,s,t,1},          The number of lines per part.
A[]={...},x;                    A[] holds the values to add to each field-width.
for(s=8;s--;)                   Loop through the parts.
for(n=0;n<I[s];n++,puts(""))    I[s] decides how many lines to the part. Ends with newline.
for(t=3;t--;)                   Go through the three chars of each line.
x=s*3+t,                        Calculate offset.
printf("%*c",                   Print the char.
n*("..."[x]-65)+A[x],           Build field-width. The string holds the index factor, A[]
                                holds the offset part.
"..."[x]-1);}                   The char itself is grabbed from the string.
                                Shifted by 1 to eliminated double backspaces.

В итоге он был не намного короче затянутой версии g()вызывающей, но короче - короче.


@ceilingcat Приветствия.
Гастропнер

@ceilingcat Неопределенный порядок вычисления аргументов функции заставляет меня задуматься.
Гастропнер

2

Java, 435 байт

Сама функция занимает 435 байт. Безусловно, есть место для улучшения, «высокого уровня» путем анализа правил о том, где разместить какого персонажа (в конце S - точечно-симметричный), и «низкого уровня», путем классического игры в гольф (возможно, вытаскивая другую переменную или объединяя две из- forпетли). Но это первый выстрел с этим довольно негольфичным языком:

import static java.util.Arrays.*;
import static java.lang.System.*;

public class CoolS
{
    public static void main(String[] args)
    {
        print(1);
        print(2);
        print(3);
    }
    static void print(int n){int i,r,d=3+6*n,w=3+n*4,h=6+n*10,m=n+n,v=w/2,k=h-1,j=w-1;char t[],S='/',B='\\',P='|',s[][]=new char[h][w];for(char x[]:s)fill(x,' ');s[0][v]='^';s[k][v]='v';for(i=0;i<1+m;i++){r=i+1;t=s[r];t[v-r]=S;t[v+r]=B;t=s[k-r];t[v-r]=B;t[v+r]=S;}for(i=0;i<m;i++){r=2+m+i;t=s[r];t[0]=t[v]=t[j]=P;t=s[k-r];t[0]=t[v]=t[j]=P;}for(i=0;i<1+n;i++){r=2+m+m+i;t=s[r];t[i]=t[i+1+m]=B;t[j-i]=S;t=s[d-i];t[i]=S;t[v-i]=t[j-i]=B;}for(char x[]:s)out.println(x);}
}

Всем привет. Боюсь, импорт является частью подсчета байтов, поэтому ваш текущий ответ на самом деле 478 байтов . Вы можете, однако, сыграть в гольф до (по совпадению) ваших текущих 435 байтов с некоторыми основными вещами для игры в гольф.
Кевин Круйссен

Вы можете использовать немного больше до 405 байтов , удалив некоторые переменные и используя t=...немного меньше, чтобы сэкономить байты. Если у вас есть какие-либо вопросы о внесенных мною изменениях, дайте мне знать. :)
Кевин Круйссен

Спасибо @KevinCruijssen, к сожалению, в настоящее время я не могу тратить здесь больше времени - это было просто развлекательным занятием, и учитывая «многословность» Java, во всяком случае, не серьезного конкурента ;-) Подумайте о добавлении своего решения, хотя - тогда мы по крайней мере, есть некоторые
внутриъязыковые

2

PHP , 378 374 378 377 376 335 331 328 328 байт

-3 байта, благодаря manatwork

-4 байта, использовал str_pad вместо str_repeat

-41 байт, благодаря предложениям manatworks

-1 байт, объединены два приращения в + = 2

-1 байт, убрал лишнее \

-4 байта, повторяя один раз.Забыл, мне нужно было передать строку в функцию, так что это больше байтов

Работает и для n = 0.

function s($b){return str_pad($w,$b);}echo s($i=1+$a=2*$argv[1]).'^
';for(;$i;$j++,$y=$z.$y)echo$z=s(--$i).'/'.s(++$j).'\
';for(;$k<$a;$k++)$x.='|'.s($a).'|'.s($a).'|
';echo$x;for(;$l<=$a/2;)echo s($m++).$c='\\',s($a).$c.s($a-$l++*2).'/
';for(;$m;$n+=2)echo s(--$m).'/'.s($n).$c.s($a).'\
';echo$x.strtr($y,'/\\','\/').s($a+1).v;

Попробуйте онлайн!


1
Поскольку объявление функции довольно дорого, и вы используете t () только в два раза, было бы короче без него . Если кроме 9 уведомлений вы также получаете 1 предупреждение, вы можете удалить цитаты 'v'в финале echo.
Манатворк

1
Вы можете использовать одну петлю для верхней и нижней наклонных частей. Инициализация $ a и $ i может быть сжата путем их перемещения при первом использовании.
manatwork

1
О, и $i>0 и $m>0может быть написано просто как $iи $m.
manatwork

1
С замыкающими пробелами , как в некоторых других решениях.
Манатворк

1
Вы также можете переместить объявление $ c к его первому использованию. Просто измените .конкатенацию после ,. Попробуйте онлайн!
Манатворк

1

Python 3 , 321 307 байт

Спасибо @EsolangingFruit за сохранение 14 байт

n=int(input())
b,f='\/'
c,l=4*n+3,10*n+6
r=range
h=c//2
L=[c*[' ']for _ in r(l)]
L[0][h],L[-1][h]='^v'
for i in r(h):a=L[h-i];a[i],a[c+~i]=f,b
for i in r(2*n):L[h-~i][0::h]='|'*3
for i in r(n+1):a=L[h+h+i];a[c+~i],a[i:c-1:h]=f,b*2
for i in r(1,l//2):L[l+~i]=L[i][::-1]
print('\n'.join(''.join(i)for i in L))

Попробуйте онлайн!

Python 2 , 303 байта

n=int(input())
b,f='\/'
c,l,r=4*n+3,10*n+6,range
h=c/2
L=[c*[' ']for _ in r(l)]
L[0][h],L[-1][h]='^v'
for i in r(h):a=L[h-i];a[i],a[c+~i]=f,b
for i in r(2*n):L[h-~i][0::h]='|'*3
for i in r(n+1):a=L[h+h+i];a[c+~i],a[i:c-1:h]=f,b*2
for i in r(1,l/2):L[l+~1]=L[i][::-1]
print'\n'.join(''.join(i)for i in L)

Попробуйте онлайн!


Вы можете заменить '\\','/'на второй строке, *'\/'чтобы сохранить три байта.
Esolanging Fruit


Благодарность! @EsolangingFruit! Я не знал о битовых операциях в Python. Кроме того, было бы сэкономить несколько байтов для использования Python2 из-за деления и круглых скобок вprint
Pétur

В Python 2 input()автоматически eval()выводится строка, поэтому вы также можете пропустить int()вызов.
Esolanging Fruit

Для Python 3 вы можете изменить последнюю строку на for l in L:print(*l,sep="")(я не думаю, что в Python 2 есть эквивалент).
Esolanging Fruit
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.