Сложите строку в треугольник

22

Учитывая строку, длина которой делится на 4, сделайте треугольник, как показано ниже.

Если строка есть abcdefghijkl, то треугольник будет:

   a
  b l
 c   k
defghij

Если строка есть iamastringwithalengthdivisiblebyfour, то треугольник будет:

         i
        a r
       m   u
      a     o
     s       f
    t         y
   r           b
  i             e
 n               l
gwithalengthdivisib

Если строка есть thisrepresentationisnotatriangle, то треугольник будет:

        t
       h e
      i   l
     s     g
    r       n
   e         a
  p           i
 r             r
esentationisnotat

Заметки

  • Строка будет состоять только из символов от aдо z.
  • Пробелы в начале / в конце и новые строки разрешены, если форма не нарушена.
  • Список строк в качестве вывода разрешен.

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

Дрянная Монахиня
источник

Ответы:

7

Древесный уголь , 25 22 21 байт

≔÷Lθ⁴λ↙…θλ→✂θλ±λ↖✂θ±λ

Попробуйте онлайн! Ссылка на подробную версию кода. Просто нарезать строку на три части и печатать их в соответствующих направлениях. Изменить: 3 байта сохранены с помощью целочисленного деления и нарезки. Сохраненный дополнительный байт, используя CycleChopвместо Sliceзаголовка строки. Изменить: Charcoal теперь поддерживает рисование произвольного текста по краю многоугольника, упрощая код до 12 байтов:

GH↙→→↖⊕÷Lθ⁴θ

Попробуйте онлайн! Ссылка на подробную версию кода.

Нил
источник
Что делают с?
Эрик Outgolfer
@EriktheOutgolfer Это новый оператор Slice.
Нил
: | К сожалению, это должно было заставить PolygonHollow сделать это, GH↙→→↖⊕÷Lθ⁴θбудет работать в следующий раз, когда я
только ASCII
6

05AB1E , 23 байта

ćsIg4÷GćsÁćsŠN·<ú«s}».C

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

объяснение

ć                        # extract head of input
 s                       # swap the remaining string to top of stack
  Ig4÷G                  # for N in [1...len(input)/4-1] do:
       ć                 # extract head
        sÁ               # swap remaining string to top of stack and rotate right
          ć              # extract head
           sŠ            # reorder stack as tail, head, remaining
             N·<ú        # prepend N-1 spaces to tail
                 «s      # concatenate with head and swap remaining string to top
                   }     # end loop
                    ».C  # join by newlines and center
Emigna
источник
6

JavaScript (ES6), 119 117 108 105 байт

s=>(l=s.length/4,S=' ',g=([c,...s],p)=>S.repeat(l)+c+(l--?p+s.pop()+`
`+g(s,p?p+S+S:S):s.join``))(s+S,'')

Отформатировано и прокомментировано

s => (                            // given the input string s:
  l = s.length / 4,               // l = length of side edge - 1
  S = ' ',                        // S = space (defining S costs 6 bytes but saves 7)
  g = (                           // g = recursive function which takes:
       [c,                        //   - c = next character
           ...s],                 //   - s = array of remaining characters
                  p) =>           //   - p = middle padding string
    S.repeat(l) + c + (           // append left padding + left character
      l-- ?                       // if side edges are not complete:
        p + s.pop() + '\n' +      //   append middle padding + right character + Line Feed
        g(s, p ? p + S + S : S)   //   and do a recursive call with updated middle padding
      :                           // else:
        s.join``                  //   append all remaining characters and stop recursion
    )                             //   (this is the bottom edge)
  )(s + S, '')                    // initial call to g()

Контрольные примеры

Arnauld
источник
4

C #, 260 байт

namespace System{using static Console;class P{static void Main(){var d=ReadLine();int e=d.Length/4,x=e,y=0,g=0,i=0;Action<int,int>a=(p,q)=>{SetCursorPosition(p,q);Write(d[g++]);};for(;i<e;i++)a(x--,y++);for(i=0;i<e*2;i++)a(x++,y);for(i=0;i<e;i++)a(x--,y--);}}}

Действительно хотел использовать SetCursorPosition.

Ungolfed:

namespace System {
    using static Console;

    class P {
        static void Main() {
            var d = ReadLine();
            int e = d.Length / 4, x = e, y = 0, g = 0, i = 0;
            Action<int, int> a = (p, q) => { SetCursorPosition(p, q); Write(d[g++]); };
            for (; i < e; i++)
                a(x--, y++);
            for (i = 0; i < e * 2; i++)
                a(x++, y);
            for (i = 0; i < e; i++)
                a(x--, y--);
        }
    }
}
LiefdeWen
источник
Простите за мое невежество, но какова цель действия в вашем решении? Это просто меньше байтов, чем функция void?
запутался и использовал
1
@confusedandamused Я привык писать ответы на отдельные функции, поэтому даже не думал ставить функцию нормально, хотя она будет короче.
LiefdeWen
3

Mathematica, 164 байта

(b=Length[c=Characters@#];k=Column[#,Alignment->Center]&;T=Table;k@{#&@@c,k@T[""<>{c[[i+2]],T[" ",2i+1],c[[-i-1]]},{i,0,(a=b/4)-2}],""<>T[c[[i]],{i,a+1,b/2+1+a}]})&


вход

[ "Iamastringwithalengthdivisiblebyfour"]

J42161217
источник
Мы все знаем, что [[1]]можно заменить на #&@@.
user202729
1
Вы все такие умные сладости!
J42161217
Я имел в виду codegolf.stackexchange.com/questions/12900/… .
user202729
Когда вы обнаружите, что делаете @(...), просто сделайте [...]вместо этого. И я не проверял, но вы, вероятно, можете сохранить другой байт, дав Columnимя (или, может быть, даже, Column[#,Alignment->Center]&чтобы избежать q), а затем поместив все оставшиеся переменные в первый аргумент внешнегоColumn (чтобы сохранить окружающие скобки).
Мартин Эндер
3

Python 3 , 120 байт

Первый гольф, подумал, что я мог бы также изучить Python по пути.

a=input()
l=len(a)//4
print(l*" "+a[0])
for i in range(1,l):print((l-i)*" "+a[i]+(2*i-1)*" "+a[4*l-i])
print(a[l:3*l+1])

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

Объяснение:

Первый символ печатается сам после len(a)//4пробелов, затем первый и последний-й iсимволы, начинающиеся со второго, печатаются через 2*i - 1пробел.

Наконец, оставшаяся подстрока печатается.

Fedone
источник
Добро пожаловать в PPCG! Вы можете узнать из этого решения .
Утренняя монахиня
Возможный гольф здесь, чтобы объявить p=print, а затем просто использовать pдля трех, которые printвы используете.
FlipTack
Кроме того, поскольку длина строки всегда делится на четыре, //(деление по полу) можно заменить на /.
FlipTack
Кстати, код, который вы указали, чтобы попробовать онлайн, не совпадает с кодом в вашем ответе.
FlipTack
3

GNU sed , 178 158 132 + 1 = 133 байта

+1 байт за -rфлаг.

s/(.)(.*)(.)/ \1\n\2;\3/
:
s/( *)(.\n.)(.*)(...);(.*)(.)/\1\2\1  \6\n\3;\4\5/m
t
:A
s/(.*\n)( *)(.*);/ \2;\1\2\3/m
tA
s/. (.)$/\1/gm

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

объяснение

В предыдущих версиях я использовал много байтов, связанных с математикой, особыми случаями и очисткой, хотя интуитивно я был уверен, что их можно избежать. С тех пор мне это удалось, в основном.

Предположим, у нас есть вход abcdEFGHIJKLMnop . Буквы EFGHIJKLMбудут в нижней части треугольника, поэтому я использовал их в качестве наглядного пособия.

Сначала мы подготовим ввод, поместив первый символ в отдельной строке (с пробелом) и вставив курсор ( ;) перед последним символом:

s/(.)(.*)(.)/ \1\n\2;\3/

Теперь у нас есть:

 a
bcdEFGHIJKLMno;p

Теперь, в цикле, мы собираемся сделать несколько вещей до последней строки: 1. Скопируйте пробелы из предыдущей строки и вставьте их после первого символа плюс два; 2. Переместите последний символ справа после пробелов, за которым следует новая строка; и 3. Переместите курсор на три символа влево.

:
  s/( *)(.\n.)(.*)(...);(.*)(.)/\1\2\1  \6\n\3;\4\5/m
  t

Вот результат каждой итерации:

 a
b   p
cdEFGHIJKL;Mno

 a
b   p
c     o
dEFGHI;JKLMn

 a
b   p
c     o
d       n
EF;GHIJKLM

Вы можете видеть, как пирамида начинает обретать форму. Вы также можете увидеть, для чего был выбран курсор: на каждой итерации он перемещался влево на три символа, а когда слева уже не осталось трех символов, он прерывал цикл, как раз тогда, когда мы достигли «дна» пирамиды.

Теперь мы собираемся сделать аналогичную операцию, но в обратном порядке. В цикле мы скопируем пробелы от начала строки с курсором до начала предыдущей строки, плюс один, в процессе перемещения курсора к этой строке.

:A
  s/(.*\n)( *)(.*);/ \2;\1\2\3/m
  tA

Вот пара итераций и конечный результат:

 a
b   p
c     o
 ;d       n
EFGHIJKLM

 a
b   p
  ;c     o
 d       n
EFGHIJKLM

...

    ; a
   b   p
  c     o
 d       n
EFGHIJKLM

Мы все закончили, за исключением некоторых дополнительных символов: A ;и дополнительного пробела в первой строке и двух пробелов в «середине» пирамиды в следующих трех строках. Простая замена избавляет от них:

s/. (.)$/\1/gm

Все сделано!

    a
   b p
  c   o
 d     n
EFGHIJKLM
Иордания
источник
2

Python 2 , 100 97 96 байт

  • Якоблав спас 1 байт: целочисленное деление не нужно
a=input()+" "
k=j=len(a)/4
while j:print j*" "+a[0]+(2*(k-j)-1)*" "+a[-1];a=a[1:-1];j-=1
print a

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

Объяснение:

Одна умная вещь, которую я здесь сделал, это заполнение ввода пробелом в конце, так что первые символы соединяются с ним, и это может быть вставлено в цикл (и так как конечные пробелы разрешены)

abcdefghijkl[space]   
To print [0] [-1]            Output=>[spaces]a[another_calculated_spaces(=0 here)][space]
Strip at both ends(a[1:-1])  
bcdefghijkl                
To print [0] [-1]            Output=>[spaces]b[another_calculated_spaces]l
Strip at both ends(a[1:-1])
and so on.

Количество циклов, чтобы следовать связан с len(word)//4. На последнем шаге печатается вся оставшаяся строка (это основание треугольника). Пробелы следуют простой схеме; первый набор пробелов уменьшается на 1, а второй набор пробелов увеличивается на 2.

officialaimm
источник
1
Можете ли вы сбрить байт, не занимаясь целочисленным делением? Так aкак всегда будет кратным 4. //->/
jacoblaw
Спасибо, я удивлен, что он не выдает никаких ошибок даже для [входных данных, длина которых не делится на 4] [ tio.run/…
officialaimm
1
Это потому, что в Python 2 деление по умолчанию является целым числом. Это было сделано в Python 3.
CalculatorFeline
2

C 225 байт

p(c){putchar(c);}S(n){while(n--)p(' ');}main(int c,char**v){int i= strlen(v[1]),n=i/4,r;char*s=v[1],*e=&s[i-1];S(n);p(*s++);p('\n');for (r=1;r<n;r++){S(n-r);p(*s++);S(2*r-1);p(*e--);p('\n');}e++;while (s!=e)p(*s++);p('\n');}

объяснил

p(c){putchar(c);}        // p is alias for putchar
S(n){while(n--)p(' ');}  // S prints n spaces
main(int c,char**v){
    int i= strlen(v[1]), // counter
        n=i/4,           // num rows in figure - 1
        r;               // current row 
    char*s=v[1],         // start char
        *e=&s[i-1];      // end char
    S(n);p(*s++);p('\n');// print first row
    for (r=1;r<n;r++){ 
        S(n-r);p(*s++);S(2*r-1);p(*e--);p('\n'); // print middle rows
    }
    e++;while (s!=e)p(*s++);p('\n'); // print last row
}
Алекс Зефферт
источник
2

C #, 172 байта

int i=0,n=s.Length;var p="";p=new string(' ',n/4)+s[i]+"\r\n";for(i=1;i<n/4;i++){p+=new string(' ',n/4-i)+s[i]+new string(' ',i*2-1)+s[n-i]+"\r\n";}p+=s.Substring(i,n/2+1);

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

Эрлантц Кальво
источник
1

Октава, 87 байт

@(s,x=(n=nnz(s))/4)[[' ';flip(diag(s(1:x))')]' [' ';diag(s(n:-1:n-x+2))];s(x+1:n-x+1)];

* В машине с Windows приведенный выше код дает правильный результат, однако в tio я добавил код для его исправления.

Объяснение:

[' ';flip(diag(s(1:x))')]'        %left side
[' ';diag(s(n:-1:n-x+2))]         %right side
s(x+1:n-x+1)                      %bottom side

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

rahnema1
источник
1

PHP> = 7.1, 122 байта

for(;$i*2<$w=strlen($a=$argn)/2;$e=$a[-++$i])echo str_pad(str_pad($a[$i],$i*2).$e,$w+1," ",2),"
";echo substr($a,$i,$w+1);

PHP Sandbox Online

PHP> = 7.1, 124 байта

for(;$i*2<$w=strlen($a=$argn)/2;$e=$a[-++$i],$s.=$s?"  ":" ")echo str_pad("",$w/2-$i)."$a[$i]$s$e
";echo substr($a,$i,$w+1);

PHP Sandbox Online

Йорг Хюльсерманн
источник
1

AWK , 129 байт

{n=split($0,a,"")
printf"%"(w=n/4+1)"s\n",a[++i]
for(;++i<w;)printf"%"(w-i+1)"s%"2*i-2"s\n",a[i],a[n-i+2]
$0=substr($0,i,i+w-1)}1

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

Я должен подумать, что это может быть немного больше, просто не видя этого.

Роберт Бенсон
источник
1

Сетчатка , 99 байт

^(.)(?=(....)+)
$#2$*  $1¶$#2$* 
( ( *).)(.*)(.)$
$1 $4¶$2$3
+`(( +).¶ ( *).)(.*)(.)$
$1$2  $5¶$3$4

Попробуйте онлайн! Объяснение: Первые два этапа генерируют первые две строки, но после этого специальный регистр не требуется, и каждая последующая строка может быть сгенерирована автоматически:

thisrepresentationisnotatriangle

        t
       hisrepresentationisnotatriangle

        t
       h e
      isrepresentationisnotatriangl

        t
       h e
      i   l
     srepresentationisnotatriang

...

        t
       h e
      i   l
     s     g
    r       n
   e         a
  p           i
 r             r
esentationisnotat
Нил
источник
1

Java 8, 213 байт

s->{int n=s.length()/4,i;String r=s(n)+s.charAt(0)+"\n";for(i=1;i<n;r+=s(n-i)+s.charAt(i)+s(i*2-1)+s.charAt(n*4-i++)+"\n");return r+s.substring(i,n*2+i+1);}String s(int n){String r="";for(;n-->0;r+=" ");return r;}

Объяснение:

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

s->{                           // Method (1) with String parameter and String return-type
  int n=s.length()/4,          //  The length of the input divided by 4
      i;                       //  And an index-integer
  String r=                    //  Result-String which starts as:
           s(n)                //   Trailing spaces
           +s.charAt(0)+"\n";  //   + the first character and a new-line
  for(i=1;i<n;                 //  Loop from `1` to `n`
      r+=                      //   And append the result-String with:
         s(n-i)                //    Trailing spaces
         +s.charAt(i)          //    + the character of the left diagonal line
         +s(i*2-1)             //    + center spaces
         +s.charAt(n*4-i++)    //    + the character of the right diagonal line
         +"\n"                 //    + a new-line
  );                           //  End of loop
  return r                     //  Return the result-String
         +s.substring(i,n*2+i+1);
                               //   + the bottom part of the triangle
}                              // End of method (1)

String s(int n){               // Method (2) with integer parameter and String return-type
  String r="";                 //  Result-String
  for(;n-->0;r+=" ");          //  Append the result-String with `n` spaces
  return r;                    //  Return the result-String
}                              // End of method (2)
Кевин Круйссен
источник