«Ступенька» струна

12

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

Для каждого символа в строке:

  • Если символ является гласным верхнего или нижнего регистра, не включая «y», выведите его, а затем переместите остальную часть строки вверх по столбцу.

  • Если символ является пробелом или символом табуляции, выведите его, затем переместите остальную часть строки вниз на столбец.

  • Если символ не тот, выведите его как обычно.

IO может быть в любом разумном формате. На входе не будет никаких новых строк. Если вы хотите, вы можете удалить любые пробелы.

Если вы решите вернуть строку, а не печатать ее, пожалуйста, включите короткую программу, которая напечатает вашу строку, чтобы ее можно было визуализировать. Это не является обязательным, и не будет идти к вашему счетчику байтов. Это просто удобство для пользователей, которые не понимают гольф или esolangs (как я), чтобы иметь возможность проверить вывод или возиться с кодом.

Образец ввода-вывода:

Вывод для "bcdef ghijkl":

    f    jkl
bcde  ghi

Выход для «Программирование головоломок и Code-Golf»:

                               lf
                            -Go
                  s  nd   de   
         ng   zzle  A   Co       
      mmi   Pu                 
   gra        
Pro

Выходные данные для "Abcdefghijklmnopqrstuvwxyz":

                     vwxyz
               pqrstu
         jklmno
     fghi          
 bcde             
A        

Как обычно, это код-гольф, поэтому выигрывает самый короткий ответ в байтах.

Джеймс
источник
Связанный: codegolf.stackexchange.com/q/55593/31716
Джеймс
Можем ли мы убрать любые пробелы?
orlp
@orlp Так как это вообще не изменит визуальное представление, я не понимаю, почему нет.
Джеймс
Если мы решим вернуть строку, включена ли программа печати в счетчик байтов?
@PeterPeter Смотрите мое последнее редактирование.
Джеймс

Ответы:

2

MATL , 38 37 байт

Oj33<G13Y2m-IL)hYstX<-"@Z"GX@)h]Xh!c!

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

объяснение

Для каждого символа код вычисляет его вертикальное положение, измеренное сверху (0 - самое высокое). Затем он создает выходную строку транспонированной: каждый символ находится на строке с таким количеством пробелов, сколько указывает его вертикальное положение. Затем все строки объединяются в двумерный массив символов, который в итоге транспонируется и отображается.

O       % push a 0
j       % input a string
33<     % array of the same length as the input that contains true for spaces or tabs
G       % push input again
11Y2    % string 'aeiouAEIOU'
m       % array of the same length as the input that contains true for vowels
-       % subtract
IL)     % remove last element
h       % prepend the 0 that is at the bottom of the stack
Ys      % cumulative sum. This gives the vertical position of each char
tX<     % duplicate. Compute minimum
-       % subtract. This sets minimum vertical position to 0
"       % for each vertical position
  @     %   push vertical position of current character
  Z"    %   string with that many spaces
  G     %   push input again
  X@)   %   get the character corresponding to the current iteration index
  h     %   concatenate horizontally
]       % end for each
Xh      % concatenate all lines into a row cell array
!       % transpose into a column cell array
c       % convert into 2D array, padding with spaces if needed
!       % transpose. Implicitly display
Луис Мендо
источник
7

Pyth, 63 байта

V_Q aY?}rN0"aeiou"=hZ?}N"     "=tZZ;Jh.mbYKh.MZYjC.b++*d+JNY*dK_YQ
                         ^^^^^
                         |||||
                         |tabs
                        space

Пробелы в середине на самом деле представляют собой один символ табуляции, но StackExchange отображает его как четыре пробела.

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

Дрянная Монахиня
источник
Я считаю 64 байта.
Конор О'Брайен,
Из-за того, что вкладка отображается здесь как четыре пробела.
Утренняя монахиня
Определенно 64 байта. mothereff.in/...
Нет, @KennyLau означало, что символ табуляции должен быть помещен вместо четырех пробелов. Посмотрите на ссылку онлайн.
Mama Fun Roll
@MamaFunRoll StackExchange автоматически заменяет вкладки на 4 пробела.
orlp
4

Python 2, 141 137 байт

def S(s,l=[0]):
 for c in s:l+=[l[-1]-(c in"aeiouAEIOU")+(c<"!")]
 for h in sorted(set(l)):print"".join([" ",c][i==h]for i,c in zip(l,s))
orlp
источник
Похоже, что этот не спускается по пробелам
Score_Under
@Score_Under Он отлично работает на моей машине. Вы тестируете на Python 2?
orlp
Работает. Я не совсем знаю, как, но я, должно быть, сделал ошибку, вставляя это в первый раз.
Score_Under
3

JavaScript (Firefox 30-57), 151 байт

s=>[...s].map((c,i)=>r[c<'!'?n++:/[AEIOU]/i.test(c)?n--:n][i]=c,n=s.length,r=[for(_ of s+s)[]])&&[for(a of r)if(s=[for(c of a)c||' '].join``)s].join`\n`

Где \nпредставляет буквальный символ новой строки.

Нил
источник
2
С помощью шаблонных строк вы можете поместить новую строку в строку, чтобы вы могли заменить /nна ``
Generic User
1
@GenericUser Количество байтов корректируется, если вы уже это сделали; Я просто не хотел использовать буквальный перевод строки в своем посте.
Нил
1

C, 180 байтов

char s[99];i,j,p[99],m,M;main(c){for(gets(s);c=s[i];j+=strchr("aeiou",c|32)!=0,j-=c<33,m>j?m=j:M<j?M=j:0)p[i++]=j;for(;m<=M;putchar(10),M--)for(i=0;c=s[i];)putchar(M^p[i++]?32:c);}

Ungolfed:

char s[99];i,j,p[99],m,M;
main(c){for(gets(s);c=s[i];
j+=strchr("aeiou",c|32)!=0,j-=c<33,m>j?m=j:M<j?M=j:0)
  //move current height up or down, adjust minimum and maximum height
p[i++]=j;  //record height of character
for(;m<=M;putchar(10),M--)  //from maximum to minimum height
for(i=0;c=s[i];)putchar(M^p[i++]?32:c);}  //print only characters on this height
mIllIbyte
источник
1

Perl, 110 байт (сценарий 108 байт + флаги 2 байта)

$h=0;map{$h{$h}.=' 'x($p-$p{$h}).$_;$p{$h}=++$p;$h+=/[aeiou]/i-/\s/}split//;print for@h{sort{$b<=>$a}keys%h}

Запуск с perl -nl script.pl, вход на стандартный ввод, выход на стандартный вывод.

Deobfuscated

Я переименовал переменные более разумно, сделал код use strictи use warningsсовместимый, и сделал явным многое из того, что делает perl автоматически.

Это просто запустить как perl script.pl, потому что он копирует эффекты -nlфлагов внутри скрипта.

use strict;
use warnings;
use English;

# The effect of -l in perl's flags
$INPUT_RECORD_SEPARATOR = "\n";
$OUTPUT_RECORD_SEPARATOR = "\n";

# These variables are magicked into existence
our $column = 0;
our %line_col = ();
our %lines = ();

# The implicit while-loop is the effect of -n in perl's flags
while (defined(my $line = <>)) {
    # The "chomp" is part of perl's -l flag too
    chomp $line;

    # Here starts the actual script. "$h=0" turns into...
    our $height = 0;
    for my $char (split '', $line) {
        if (!exists $line_col{$height}) {
            # Setting it to 0 is a bit of a white lie, but it might as well be 0.
            # Perl would otherwise have called the value "undef", which is
            # similar to 0 in numeric contexts.
            $line_col{$height} = 0;
        }

        $lines{$height} .= ' ' x ($column - $line_col{$height});
        $lines{$height} .= $char;

        $column++;
        $line_col{$height} = $column;

        $height++ if $char =~ /[aeiou]/i;
        $height-- if $char =~ /\s/;
    }

    # Sort line heights numerically descending (so the greatest is printed first)
    my @heights = sort { $b<=>$a } keys %lines;

    for my $line (@lines{ @heights }) {
        print $line;
    }
}
Score_Under
источник
1

JavaScript (ES6), 133

s=>s.replace(/[^aeiou ]*(.?)/gi,(z,x,c)=>(q=o[r]||'',o[r]=q+=' '.repeat(c-q.length)+z,x<'!'?++r:r?--r:o=[,...o]),o=[],r=0)&&o.join`
`

Меньше гольфа

s=>(
  s.replace(/[^aeiou ]*(.?)/gi,(z,x,c)=>(
    q = o[r] || '',
    o[r] = q += ' '.repeat(c - q.length) + z,
    x == ' ' ? ++r : r ? --r : o = [,...o]
  ), o = [], r = 0),
  o.join`\n`
)

Тестовое задание

f=s=>s.replace(/[^aeiou ]*(.?)/gi,(z,x,c)=>(q=o[r]||'',o[r]=q+=' '.repeat(c-q.length)+z,x<'!'?++r:r?--r:o=[,...o]),o=[],r=0)&&o.join`
`

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

test()
#I { width:90%}
<input id=I oninput='test()' value='Programming Puzzles And Code-Golf'>
<pre id=O>

edc65
источник
0

Haskell (в терминале ANSI), 75 байт

("\27[2J"++).(h=<<)
h ' '="\27[B "
h c|elem c"aeiouAEIOU"=c:"\27[A"
h c=[c]

Пример использования: putStr $ ("\27[2J"++).(h=<<) $ "bcdef ghijkl"

При этом используются управляющие коды ANSI для перемещения курсора вверх и вниз.

Ними
источник
0

C 173 160 156 155 байтов

Изменить: Заимствовал идею использования strchr из @mIllIbyte, чтобы сбрить 13 байтов

Edit2: упрощено сравнение мин / макс, -4 байта

Edit3: c может иметь любое значение для начала -> в main (c) вместо -1 байта

Edit4: добавлено ungolf / объяснение

p,l,j,m;main(c){char b[99],*s=gets(b);for(;j<m+2;p?putchar(c?l?32:c:10):l<j?j=l:l>m?m=l:0,l-=c?(c&=223)&&c-9?!!strchr("AEIOU",c):-1:(p=s=b,l+j++))c=*s++;}

Разгромил и объяснил:

/* declare and initialize these variables to int and 0 */
p,l,j,m;

/* declares main, but also int c */
main(c)
{

  /* we can handle strings of length 98 (+1 for string-terminating 0) */
  /* we declare and initialize s to point to the beginning of the input
     string for the first pass through the for loop */
  char b[99],*s=gets(b);

  /* the for-loop actually contains nested loops, where the inner loops
     behave differently depending on the outer loop parameter p as follows:
     p attains the values false (0) and true (non-null pointer), in this order.

     p == false:
      the inner loop has the parameter s and passes through all the characters
      in the string until the string is exhausted (*s == 0). l is the vertical
      position of the current character relative to the first character
      (l = 0), smaller number = higher up. The purpose here is simply to find
      the range of vertical positions [j, m] present in the string. The
      commands in execution order are:

      -- loop over s --

      // test does not do anything since j <= m by design
      1. j < m+2

      // puts current char in c and increments string counter
      2. c = *s++          

      // ensures that j (m) equals the min (max) of the vertical positions (l)
         encountered so far. At first step j = l = m = 0.
      3. l<j?j=l:l>m?m=l:0 

      // c != 0, this updates the vertical position for the next character
      // c = SPC or C = TAB -> lower (l increases by 1)
      // c = "aeiouAEIOU" -> higher (l decreases by 1)
      4a. l-=(c&=223)&&c-9?!!strchr("AEIOU",c):-1

      -- loop over s ends --

      // c == 0, this resets the string pointer s and puts p = true, and 
      //         thereby initiates the next phase of the algorithm
      //         see rest of the explanation at p == true)
      4b. p=s=b

    p == true:
     now there are two inner loops. The outer of these has the parameter j,
     which ranges from the smallest vertical position+1 (the value of j after
     the p == false pass) to the largest vertical position+1 (m+2 after the
     p == true pass). The innermost loop has the parameter s and passes through
     all characters in the string until the string is exhausted (*s == 0) just
     as in the p == false inner loop. Here l is now the vertical position
     relative to the current position j-1, so that l == 0 when a character is
     at the current level. Such characters are printed as is, whereas
     characters at other levels are replaced by space. The end-of-string
     marker 0 outputs a newline. The commands in execution order are:

      -- loop over j --

      // at first step increments j to point to be one more than the
      // current vertical position. At other steps moves the current position
      // (j-1) one vertical position downwards. Also, at all steps, this
      // biases the vertical position counter l to be zero at the current
      // vertical position (j-1)
      1. l=-j++

      // compare j to stopping criteria, exit if j > m+1
      2. j < m+2

       -- loop over s --

       // puts current char in c and increments string counter
       3. c = *s++          

       // outputs character as follows:
       // c == 0 (end of string), output newline
       // c != 0 (middle of string)
       //  l == 0 (character at current vertcial position), output c
       //  l != 0 (character not at current vertical position), output space
       4. putchar(c?l?32:c:10)

       // c != 0, this updates the vertical position for the next character
       // c = SPC or C = TAB -> lower (l increases by 1)
       // c = "aeiouAEIOU" -> higher (l decreases by 1)
       5a. l-=(c&=223)&&c-9?!!strchr("AEIOU",c):-1

       -- loop over s ends --

      // c == 0, this resets the string pointer s for next loop over s
      //         algorithm (see rest of the explanation at p == true)
      5b. p=s=b

     -- loop over j ends --
  */

  for(;
      j<m+2;
      p?putchar(c?l?32:c:10):
    l<j?j=l:l>m?m=l:0,
      l-=c?(c&=223)&&c-9?!!strchr("AEIOU",c):-1:
       (p=s=b,l+j++))
    c=*s++;
}
Зунга
источник