Геометричний виклик


23

Усі люблять геометрію. То чому б нам не спробувати зашифрувати його? Це завдання передбачає введення літер та цифр та створення фігур залежно від цього.

Вхідні дані

Вхід буде у формі (shapeIdentifier)(size)(inverter).

Але що таке формаІдентифікатор, розмір та інвертор?

Ідентифікатор форми - це ідентифікатор для типу фігури, яку ви будете робити * s. Нижче наведено ідентифікатори форми:

  • s - Майдан
  • t - Трикутник

Розмір буде між ними 1-20 , і це розмір фігури.

Інвертор полягає в тому, чи буде форма перевернута, чи позначається а +чи а -. Зверніть увагу: s3-== (дорівнює), s3+оскільки квадрати симетричні. Однак,t5- ! = (Не дорівнює)t5+ .

Трейлінг пробілу добре у виході, але провідна пробіл - ні.

Вихідні приклади

Input: s3+
Output:
***
***
***

Input: t5+

Output:
  *
 ***
*****

Input: t3-
Output:
***
 *

Спеціальні примітки

Вхід трикутника завжди буде непарним числом, тому трикутники завжди закінчуватимуться на 1 *вгорі.

Розмір трикутника - це розмір основи, якщо інвертор є, +і розмір верху, якщо інвертор -.


3
Як хтось, хто зараз займається геометрією (і навчається на фінал з Геометрії), я можу сказати зі 100% впевненістю: Геометрія - це абсолютно, зовсім не цікаво ... D:
Ешвін Гупта,

Відповіді:


9

Pyth, 40 36 34 32 байт

-1 байт від @isaacg

JstPz_W}\+zjl#m.[J*\*-J*}\tzyd;J

Точка з комою всередині лямбда - це глобальне значення змінної лямбда - особливість, яка зберігає один байт.

                         Implicit: z = input
JstPz                    J = size.
_W }\+z                  Reverse if "+" in z
j l# m                J  Join the nonempty lines in map lambda d:... over range(J)
      .[J            ;   Pad the following with spaces (;) to length J
         *\*               "*", this many times:
            -J*}\tzyd        J if "t" not  in z,
                             otherwise the correct number for a triangle.

Спробуйте тут .

Тестовий набір .


1
Шлях занадто довгий, але бив Japt на 15 байт? Я не можу чекати, щоб побачити, як це буде в гольф :)
ETHproductions

Приємне рішення! Ви можете зберегти байти, замінюючи qez\+з }\+z, тому що +може з'явитися тільки в останній позиції.
isaacg

6

Pyth, 38 байт

JsPtzj?}\szm*\*JJ_W}\-zm.[J*\*hyd;/hJ2

Тестовий набір

В основному так просто, як це виходить. Я б хотів, щоб я міг поєднати трохи логіки для двох форм, але наразі це окремо.


5

JavaScript (ES6), 142 146 147

Редагувати 1 байт збереженого thx @ETHproductions Редагувати 2 байти sve thx @ user81655

i=>([,a,b]=i.match`.(.+)(.)`,Array(l=i<'t'?+a:-~a/2).fill('*'.repeat(a)).map((r,i)=>l-a?(' '.repeat(i=b<'-'?--l:i)+r).slice(0,a-i):r).join`
`)

Тест (запуск у FireFox)

F=i=>(
  [,a,b]=i.match`.(.+)(.)`,
  Array(l=i<'t'?+a:-~a/2).fill('*'.repeat(a))
  .map((r,i)=>l-a?(' '.repeat(i=b<'-'?--l:i)+r).slice(0,a-i):r)
  .join`\n`
)

function test() { O.textContent=F(I.value) }

test()
Input: <input id=I oninput="test()" value="t11-"/>
<pre id=O></pre>


\d-> ., оскільки до і після цього гарантовано буде точно одна нецифрова цифра
ETHproductions

@ETHproductions вірно, дякую
edc65

Приємно. Я думаю, що це оптимальний алгоритм в JS, не можна знайти коротший.
ETHproductions

i.match(/.(.+)(.)/) -> i.match`.(.+)(.)`
користувач81655

@ user81655 приємний натяк, дякую
edc65

5

Python 2, 106 байт

s=raw_input()
n=int(s[1:-1])
for i in[range(1,n+1,2),n*[n]][s<'t'][::2*('+'in s)-1]:print('*'*i).center(n)

Вихід є ідеальним прямокутником, кожен рядок оброблений пробілами, які я припускаю, що це нормально, грунтуючись на коментарях в ОП.

Примітка. Я все ще ніколи не впевнений, чи inputдозволено в Python 2 для таких проблем ...


4

Japt, 62 60 55 52 51 байт

V=Us1 n;U<'t?Vo ç*pV):0oV2 £S²pY iY'*pV-X})·z2*!Uf-

Спробуйте в Інтернеті!

Перше, що нам потрібно зробити - це зрозуміти, наскільки великою має бути наша форма. Це досить просто:

      // Implicit: U = input string, S = space
V=    // Set variable V to
Us1   // everything after the first char of U,
n;    // converted to a number. This turns e.g. "12+" into 12.

Тепер організуємо форму виводу:

U<'t?      // If U comes before "t" lexicographically (here, if the first char is "s"),
Vo         //  make a list of V items,
ç*pV)      //  and set each item to V asterisks.
:0oV2      // Otherwise, create the range [0, V) with steps of 2 (e.g. 7 -> [0,2,4,6]),
£       }) //  and map each item X and index Y to:
S²pY       //   Repeat 2 spaces Y times. This creates a string of Y*2 spaces.
iY'*pV-X   //   At position Y in this string (right in the middle), insert V-X asterisks.
·          // Join with newlines.

На сьогодні ми подбали про розмір та форму виходу. Залишилося лише обертання. На даний момент трикутники вказані вгору, тому нам потрібно перегорнути їх, якщо третім знаком є +:

!Uf-    // Take the logical not of U.match("-").
        // If U contains "-", this returns false; otherwise, returns true.
2*      // Multiply by two. This converts true to 2, false to 0.
z       // Rotate the list 90° that many times.
        // Altogether, this turns the shape by 180° if necessary.

І з неявним результатом наша робота тут робиться. :-)


4

Пітон 2, 235 193 167 157 байт

Оновлення:

Здійснено значну оптимізацію за допомогою розуміння списку та str.center (). У мене є відчуття, що я можу зробити ще тхо, пізніше я погляну на це.

Оновлення 2

Збережено 10 байт із пропозиціями Шерлока9. Дуже дякую! :)

d=raw_input()
x=int(d[1:-1])
o="\n".join("*"*x for i in range(x))if d<"t"else"\n".join(("*"*i).center(x)for i in range(x,0,-2))
print o[::-1]if"+"in d else o

Стара відповідь

d=raw_input()
x=int(d[1:-1])
if "s" in d:
 for y in range(x):
    o+="*"*x+"\n"
 o=o[:-1]
else:
 b=0
 while x+1:
    o+=" "*b+"*"*x+" "*b+"\n"
    x-=2
    b+=1
 o=o[:-1]
 if d[-1]=="+":
    o=o[::-1]
print o

Досить прямолінійний підхід. Введення рядка на рядок у рядку, який я виводя в кінці кінця. Трикутники завжди малюються перевернутими і, якщо потрібно, перевертаються. Те, що ви можете помножити рядок за допомогою Integer, врятувало мені багато байтів!

Я спробую трохи пізніше пограти в гольф, буду тим часом вдячний для пропозицій, оскільки я ще не так сильно володію цим.

редагувати: Гольфували це багато за допомогою коментарів та крадіжки з розрахунком розміру з одного з інших пітон-відповідей. Я думаю, що це найбільше, що я можу зробити з цим алгоритмом.


Як ти порахував? Під час використання wcце дає мені кількість байтів 235. Чи помиляюсь я?
ბიმო

1
Це дійсно 235 байт. Порада з гольфу: Використовуйте вкладки замість двох пробілів, що є дійсним для Python 2 і буде голити 5 байт.
Дверна ручка

Також вам не потрібно користуватися raw_input, використовуючи, inputви заощаджуєте 4 байти. Крім того, вам не потрібні дужки у другому рядку, і якщо зовсім не використовувати змінну x(використовуючи if"s"in d), ви заощадите ще 9 байт.
ბიმო

2
@DenkerAffe при підрахунку у вікні віднімайте 1 байт для кожного нового рядка - нові рядки - 2 байти у Windows, але 1 байт в інших середовищах
edc65

1
По-перше, ви можете зняти []дужки в кожному з joinвикликів функцій. По-друге, if d<"t"elseкоротше і працює тому, що "s3+"<"t"<"t3+"в Python. По-третє, else"\n".joinі .center(x)for. Немає місця. Це не потрібно. По-четверте, print o[::-1]if"+"in d else oде я переставив речі на два байти (один пробіл між ]іif і інший між ifі "+".
Sherlock9

3

JavaScript, 220 байт.

q=s=>+s.slice(1,s.length-1);f=s=>s[0]=="s"?("*".repeat(q(s))+"\n").repeat(q(s)):Array.apply(0,Array(-~(q(s)/2))).map((_,n)=>(s[s.length-1]=="-"?~~(q(s)/2)-n:n)).map(n=>(" ".repeat(q(s)/2-n)+"*".repeat(n*2+1))).join("\n")

Бігайте з f(input here)

Спробуйте тут!

У квадратах є нові лінійки, але трикутники - ні. Пояснення:

q=s=>+s.slice(1,s.length-1);                                                                                                                                                                                                 Define a function, q, that takes returns the argument, without the first and last character, casted into an integer.
                            f=s=>                                                                                                                                                                                            Define a function, f, that takes one argument, s. (This is the main function)
                                 s[0]=="s"?                                                                                                                                                                                  If the first character of s is "s" then...
                                           ("*".repeat(q(s))     )                                                                                                                                                           Repeat the "*" character q(s) times.
                                           (                +"\n")                                                                                                                                                           Append a newline to that
                                                                  .repeat(q(s))                                                                                                                                              Repeat that q(s) times.
                                                                               :                                                                                                                                             Else... (the first character of s isn't "s")
                                                                                Array.apply(0,Array(          ))                                                                                                             Create an array of length...
                                                                                Array.apply(0,Array(-~(q(s)/2)))                                                                                                             floor(q(s)/2)+1
                                                                                                                .map((_,n)=>                                   )                                                             Map each element, _ with index n to...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?              ))                                                             If the last element of s is "-" then...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?~~(q(s)/2)-n  ))                                                             floor(q(s)/2)-n
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?            : ))                                                             Else...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?             n))                                                             Just n
                                                                                                                                                                .map(n=>                                        )            Map each element into...
                                                                                                                                                                .map(n=>(" ".repeat(q(s)/2-n)                   )            Repeat " ", q(s)/2-n times.
                                                                                                                                                                .map(n=>(                   )+"*".repeat(n*2+1)))            Append "*", repeated 2n+1 times.
                                                                                                                                                                .map(n=>(" ".repeat(        )+"*".repeat(n*2+1))).join("\n") Join with newlines

Довжина вашого першого рядка - 338 символів. На показ потрібен один монітор і півтора.
isanae


1
Я не натискаю на випадкове посилання на tinyurl, але перевіряю ще раз. У будь-якому випадку, намагайтеся уникати смуг прокрутки в кодових полях, це значно ускладнює читання.
isanae

1
@Loovjo Я думаю, що він має на увазі перший рядок пояснення. Я зазвичай відкладаю своє пояснення, а не цей стиль для відповідей на JavaScript, тому вам не потрібно прокручувати, щоб побачити половину його.
користувач81655

@ user81655 Так, я мав на увазі в поясненні. Тепер я розумію плутанину!
isanae

3

Python 2, 157 132 байт

def f(s):
 S=int(s[1:-1])
 for n in([range(1,S+2,2),range(S,0,-2)]['-'in s],[S]*S)['s'in s]:
  print "{:^{S}}".format('*'*n,S=S)

Перша спроба викликала, що +/- кінець в кінці був необов'язковим, позбавлення від цього дозволило мені поголити купу

Ідея тут полягає в тому, щоб скласти список, який може потрапити на загальний вихід. Найважчою частиною було відокремлення довжини від введення.


Для отримання довжини я використав x=int(d[1]if len(d)<4 else d[1:3]) а d - вхідний рядок. Це на 5 байт коротше вашого рішення. Ви все ще далеко попереду мого відповіді пітону, треба спробувати зрозуміти, що ви там зробили, і наступного разу побили вас! :)
Денкер

1
Насправді x=int(d[1:-1])це набагато коротше, просто побачив це в іншій відповіді пітона.
Денкер

@DenkerAffe, з будь-якої причини я пам’ятаю, що інвертор був необов’язковим, так що це не спрацювало б, але, мабуть, я просто це створив
wnnmaw

2

Сітківка , 102 85 байт

Кількість байтів передбачає, що вихідний код кодується як ISO 8859-1.

\d+
$0$*:¶
^((\w)+):(:+)
$1$2$3$2¶$0
m`s$|:t

)`(.+)¶-(\D*)
-$2¶$1
m`^.

G`.
T`ts:` *

Спробуйте в Інтернеті.

Я спробую трохи пізніше пограти в гольф.


У блокноті ++ сказано, що ваш код - 89 байт, а не 85. Я використав кодування ISO-8859-1 і пішов на редагування> Перетворення EOL> Формат UNIX / Linux, щоб використовувати \nзамість \r\n. Base64 вмісту: XGQrCiQwJCo6wrYKXigoXHcpKyk6KDorKQokMSQyJDMkMsK2JDAKbWBzJHw6dAoKKWAoLispwrYtKFxEKikKLSQywrYkMQptYF4uCgpHYC4KVGB0czpgICo=(пряма копія із блокнота ++). Як не дивно, будь-яке онлайн-рішення дає мені 85 байт ... Гум ...
Ісмаель Мігель

@IsmaelMiguel Повинні бути щось не в тому, як Блокнот ++ рахує . Вони, безумовно, є одним байтом в ISO 8859-1 (зі значенням 182).
Мартін Ендер

2

Серйозно, 54 байти

,#i's=`≈;'**@½≈";#dXdXεj' +"£n`@`≈;'**n`@Iƒ('-=WXa0WXü

Спробуйте це в Інтернеті

,#i                                                    Take input, push chars separately
   's=                                   Iƒ            IF the first char is "s":
                                `      `@                run the quoted function
                                 ≈;'**n                  make list of n strings of n *'s
      `                       `@                       ELSE run the quoted function:
       ≈;                                                make two copies of int n
         '**                                             use one to make string of n *'s
            @½≈                                          cut the other in half (e.g. 5->2)
               "           "£n                           run n/2 times the quoted function:
                ;#                                        copy the string as list of chars
                  dXdX                                    discard the last 2 *'s
                      εj                                  join back into string
                        ' +                               prepend a space
                                           ('-=WX 0WX  IF the third character is "-":
                                                 a       invert the stack
                                                     ü pop and print the entire stack

@Mego: Бачиш це #dXdXεj? STRING SLICING ????


2

ES6, 178 172 159 байт

s=>(p=s.match(/d+|./g),u=n=+p[1],m=n+1>>1,t=' '.repeat(n)+'*'.repeat(n),v=s<'t'?0:p[2]<'-'?(u=m,1):-1,[...Array(s<'t'?n:m)].map(_=>t.substr(u,u,u+=v)).join`
`)

Це працює завдяки цікавому спостереженню, яке я зробив. Якщо ви повторите nпробіли та nзірочки, ви отримаєте (наприклад, для n=5) це:

     *****

Тепер візьміть підрядки з однаковим початком і довжиною:

     |*****| (5)
    | ***| (4)
   |  *| (3)

Ці підрядки - це саме ті рядки, які нам потрібні t5 .

Редагувати: збережено 6 байт завдяки @ edc65.

Редагувати: Збережено 13 байт завдяки приховуванню u+=vтретього аргументу, substrщо дозволяє мені спростити ініціалізацію.


@ThomasKwa Так, після того, як я встановив tкод обробки, виявилося, що wі uстало еквівалентом і це врятувало мені досить байт , щоб взяти мене назад до 178!
Ніл

[,b,c]=s.matchі пізніше s<'t'... слід зберегти кілька байтів (лише Firefox)
edc65

@ edc65 Просто не збереження відповідності sдозволяє мені використовувати, s<'t'який врятував мені 6 байт, дякую.
Ніл

2

MATL , 48 байт

' *'jt4Y2m)U1$l't'Gm?2MQ2/:1L3$)R!P!R'+'Gm?P]]Q)

Використовується поточна версія (10.1.0) мови / компілятора.

Код приймає символи введення в будь-якому порядку: всі s11+, 11s+і навіть1+s1 були б дійсними рядками введення.

EDIT (30 липня 2016 р.): Пов'язаний код замінюється 1L3$)на той, Y)що відповідає останнім змінам мови

Спробуйте в Інтернеті!

Пояснення

' *'        % push string. Will be indexed into to obtain final result
j           % input string
t           % duplicate
4Y2         % predefined literal string '0123456789'
m           % logical index of digits in input string
)           % index into input string to obtain substring with digits
U           % convert to number
1$l         % generate square of ones with that size
't'         % push character 't'
G           % push input string
m           % true if input string contains 't'
?           % if so...
  2M        % push argument of call to function `l`, i.e. square size
  Q2/       % add 1 and divide by 2. Call result T
  :         % generate vector [1, 2, ... T]
  1L        % predefined literal representing Matlab's `:` index
  3$)       % two dimensional index. Transforms square into rectangle
  R         % remove (set to zero) lower-left corner
  !P!       % flip horizontally
  R         % remove lower-left corner. This gives inverted triangle
  '+'       % push character '+'
  G         % push input
  m         % true if input contains '+'
  ?         % if so...
    P       % flip vertically
  ]         % end if
]           % end if
Q           % add 1. This gives array of values 1 and 2
)           % index string ' *' with this array to produce char array
            % implicitly display that char array

1

C, 259 байт

#define x(y);)putchar(y)
#define m(n)for(n=0;n++<
#define T {m(q)i x(32);m(q)s-i*2 x(42);puts("");}
main(q,v,i,s)char**v;{s=atoi(v[1]+1);if(*v[1]=='s')m(i)s*s x(42)&&!(i%s)&&puts("");else if(strchr(v[1],'+'))for(i=s/2+1;i-->0;)T else for(i=-1;i++<s/2+1;)T}

неозорий

main(q,v,i,size)char**v; // neat way of declaring variables
{
    size=atoi(v[1]+1);
    if(*v[1]=='s')
    {
        for(i=0;i++<size*size;)
        {
            putchar(42); // returns 42 (true)
            if(!(i%size))
                puts("");
        }
    }
    else if(strchr(v[1],'+')) // if finds plus sign
    {
        for(i=size/2+1;i-->0;) // iterate the height of the triangle
        {
            for(q=0;q++<i;)putchar(32); // conveniently i is the number os spaces before each line
            for(q=0;q++<size-i*2;) putchar(42);
            puts("");
        }
    }
    else for(i=-1;i++<size/2+1;) // does the same as above but inverted order
    {
        for(q=0;q++<i;)putchar(32);
        for(q=0;q++<size-i*2;)putchar(42);
        puts("");
    }
}

Пропозиції та критика дуже вітаються.


1

Рубі, 99

->s{n=s[1,2].to_i
n.times{|i|d=(s.ord-115)*(s[-1]<=>?,)*(n-1-i*2)
d<1&&puts((?**(n+d)).center(n))}}

Обчислює квадрат або трикутник висоти nта середньої ширини n шляхом вирівнювання нахилу сторін (тому обчислена ширина трикутника становить 2n-1 біля основи, 1 на кінчику.), Але він друкує лише ті рядки, які не перевищуютьn символів.

unolfolf у тестовій програмі

f=->s{                         #take a string as an argument
  n=s[1,2].to_i                #take 2 characters starting at index 1 and convert to a number for the size
  n.times{|i|                  #iterate through n rows    
    d=                         #calculate how many stars "MORE THAN" n we need on a row
    (s.ord-115)*               #ascii code for 1st character of string - 115 : s-->0, t-->1
    (s[-1]<=>?,)*              #compare last character of input with comma character - --> +1 + --> -1
    (n-1-i*2)                  #row number * 2: 0 at centre, positive above it, negative below it
    d<1&&                      #only output if d is nonpositive (i.e we need less than n or exactly n stars)
    puts((?**(n+d)).center(n)) #print n+d stars, centred in a field of n characters padded by whitespace
  }
}

f[gets.chomp]

1

Джолф, 37 байт, неконкурентоспроможний

Я додав функції після опублікування цього виклику, тому це не може вважатися прийняттям. Це закодовано в ISO-8859-7. Спробуйте всі тестові приклади тут .

onFiΒ€ioSgiγ?='sn―sΒ'*―TΒ1'*?='-SZiγγ

Частина 1: розбір рядка

onFiΒ€ioSgi
on          set n to
  Fi         the first entity of i (the shape identifier)
    Β       set Β (beta) to
     €i      the "inside" of i (in this case, the size) as a number
       oS   set S to
         gi  the last entity of i (the inverter)

Частина 2: отримання результату

γ?='sn―sΒ'*―TΒ1'*
γ                 set γ (gamma) to the result of the following expression
 ?='sn             if n is the character s,
      ―sΒ'*         then return a pattern "s" (a square) made with "*"s
           ―TΒ1'*    otherwise, return a pattern "T" (triangle) that is centered and
                     has a scale factor of 1, made with "*"s

Частина 3: інвертування результату

?='-SZiγγ
?='-S     if S is a "-"
     Ziγ   return γ, inverted across its lines
        γ  otherwise, return γ untouched
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.