Помните, что vowelese!

25

вход

Строка печатных символов ASCII, например:

This is an example string.

Выход

Для каждого согласного ( BCDFGHJKLMNPQRSTVWXYZbcdfghjklmnpqrstvwxyz), за которым не следует гласный ( AEIOUaeiou), добавьте последний гласный перед ним в нижнем регистре.
Согласные перед первым гласным остаются как есть :

Thisi isi ana examapale seterinigi.

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

AN EXAMPLE WITH A LOT UPPERCASE (plus some lowercase)
=> ANa EXAMaPaLE WITiHi A LOTo UPuPEReCASE (pelusu some lowerecase)

And here comes a **TEST** case with 10% symbols/numbers(#)!
=> Anada here comese a **TESeTe** case witihi 10% siyimiboloso/numuberese(#)!

This is an example string.
=> Thisi isi ana examapale seterinigi.

abcdefghijklmnopqrstuvwxyz
=> abacadefegehijikiliminopoqorosotuvuwuxuyuzu

A pnm bnn
=> A panama banana

Tell me if you need more test cases!
=> Telele me ifi you neede more tesete casese!

счет

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

Wastl
источник
Итак, должны ли вставленные гласные всегда быть строчными, а текст - прописными и строчными?
Эрик Outgolfer
Может ли вывод быть в форме списка / массива?
Натан Диммер
@EriktheOutgolfer Да, я не хотел, чтобы прописные буквы были там, где нам нужны строчные буквы, но это усложнило бы задачу, если бы пришлось проверять регистр соседних букв
это было
11
Ешьте здоровых детей, попробуйте A pnm bnn!
Стьюи Гриффин
4
Кто-нибудь еще думает, «Как итальянцы» должны идти где-то в названии?
Артелий

Ответы:

14

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

i`(?<=([aeiou]).*?[^\W\d_aeiou])(?![aeiou])
$l$1

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

Нил
источник
8

JavaScript (ES6), 108 105 байт

(Сохранено 3 байта благодаря @Shaggy.)

f=s=>(t=(s+=' ').replace(/[aeiou]|[a-z][^aeiou]/ig,r=>r[1]?r[0]+v.toLowerCase()+r[1]:v=r,v=''))!=s?f(t):s

В поисках гласных или согласные без следующих гласных:

/[aeiou]|[a-z][^aeiou]/ig

(Нам не нужно явно искать согласные, потому что гласные исключаются на основе /[aeiou]|....)

В нем хранятся гласные v, и в них vвставлены согласные без следующих гласных :

r[1]?r[0]+v.toLowerCase()+r[1]:v=r

(Если r[1] существует, мы сопоставляем согласный плюс не гласный.)

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

Рик Хичкок
источник
1
Это даже лучше. Мне действительно нужно взглянуть на регулярное выражение
Луис Фелипе Де Иисус Муньос
+1 умная идея использовать замену по карте + объединение
Downgoat
Исходя из вашей (почти) рабочей версии без рекурсии: s=>s.replace(/[aeiou][^a-z]*([a-z](?![aeiou]))+/gi,s=>s.replace(/(?!^)./g,a=>a+s[0].toLowerCase()))кажется, у меня не может быть проблем с последовательностями не-букв
Downgoat
Рекурсия, безусловно, упрощает вещи здесь.
Рик Хичкок
(s+=' ')следует сохранить несколько байтов.
Лохматый
4

Стандартный ML , 225 223 байта

str o Char.toLower;fun?c=String.isSubstring(it c)"aeiou"fun g(x,l)$d=(fn l=>if Char.isAlpha$andalso not(?d)then if? $then(it$,l)else(x,l^x)else(x,l))(l^str$)fun f$(c::d::r)=f(g$c d)(d::r)|f$[c]= #2(g$c c);f("","")o explode;

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

Менее гольф:

val lower = str o Char.toLower

fun isVowel c = String.isSubstring (lower c) "aeiou"

(* c is the current char, d is the next char, x is the last vowel and l the accumulator 
   for the resulting string *)
fun g (x,l) c d = 
    if Char.isAlpha c andalso not (isVowel d)
    then if isVowel c 
         then (lower c, l^str c)
         else (x, l^str c^x)
    else (x, l^str c)

fun f t (c::d::r) = f (g t c d) (d::r)
  | f t [c] = #2(g t c #"d")

val h = f ("","") o explode;

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

Laikoni
источник
вау, ML Golf выглядит действительно интересно! Я люблю itи использование имени $переменной.
Линн
@ Линн Я написал совет о переименовании идентификатора некоторое время назад и планировал написать о нем itтоже, но пока не удосужился сделать это.
Лайкони
4

sed 4.2.2 , 64 байта

:
s/(([aeiou])[^a-z]*[b-df-hj-np-tv-z])([^aeiou]|$)/\1\l\2\3/I
t

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

KernelPanic
источник
Честно говоря, моя цель здесь состоит в том, чтобы просто попытаться побить Perl на пару байтов. Посмотрим, будет ли это так :)
KernelPanic
4

Perl 5, 68 67 59 байт

perl -pe '$v="[aeiou])";1while s/($v[^a-z]*[b-z]\K(?<!$v(?!$v/\L$1/i'

Вот отличный пример полезности \K , и я не могу поверить, что не знал об этой функции до того, как Дом Хастингс указал на нее.

Я не смог получить правильное поведение с помощью только использования s///g, поэтому реальный цикл кажется необходимым. (Возможно, правильное использование косвенного утверждения могло бы работать без явного while- но я не нашел его.)

Хлебница
источник
Хороший подход! Не смог придумать ничего лучшего, но сумел получить 6 байтов: попробуйте онлайн!
Дом Гастингс
1
@DomHastings: еще короче (до 58 байт) с учетом [aeiou])переменной: попробуйте онлайн!
ShadowRanger
3

JavaScript ES6, 115 байт

Экономит 8 байт благодаря @ETHProductions

s=>[x="",...s].map((i,j)=>(r=/[aeiou]/i).test(i)?x=i:/[a-z]/i.test(i)&&!r.test(s[j]||1)?i+x.toLowerCase():i).join``

Мне удалось надуть это больше в процессе игры в гольф O_o, но это также исправляет ошибку

s=>[x="",...s].map(             // Create a new array with x storing last vowel
                                // This also offsets indexes by one so rel to original str refers to next char
   (i,j)=>                      // Going through each char...
      (r=/[aeiou]/i).test(i)?   // If it's vowel, store it in x
          x=i:
      /[a-z]/i.test(i)          // If a letter (thats not a vowel excluded by above)
         &&!r.test(s[j]||1)?    // Test if next char is *not* vowel
         i+x.toLowerCase():i    // If it isn't, then add the most recent vowel after
    ).join``                    // Combine back to string
Downgoat
источник
@RickHitchcock, ох, совсем забыл про окончание
чарса
1
@RickHitchcock исправлено
Downgoat
Ах, да, спасибо за гольф @ETHproductions
Downgoat
3

JavaScript, 88 82 байта

Сделано с одним регулярным выражением:

Оригинальная версия (88 байт):

s=>s.replace(/(?<=([aeiou]).*?(?![aeiou])[a-z])(?=[^aeiou]|$)/gi,(_,c)=>c.toLowerCase())

Обновленная версия (82 байта) после просмотра регулярного выражения Нила :

s=>s.replace(/(?<=([aeiou]).*?[^\W\d_aeiou])(?![aeiou])/gi,(_,c)=>c.toLowerCase())

var tests = {
  "AN EXAMPLE WITH A LOT UPPERCASE (plus some lowercase)":
    "ANa EXAMaPaLE WITiHi A LOTo UPuPEReCASE (pelusu some lowerecase)",
  "And here comes a **TEST** case with 10% symbols/numbers(#)!":
    "Anada here comese a **TESeTe** case witihi 10% siyimiboloso/numuberese(#)!",
  "This is an example string.":
     "Thisi isi ana examapale seterinigi.",
  "abcdefghijklmnopqrstuvwxyz":
    "abacadefegehijikiliminopoqorosotuvuwuxuyuzu",
  "A pnm bnn":
     "A panama banana",
  "Tell me if you need more test cases!":
     "Telele me ifi you neede more tesete casese!"
};

for ( test in tests )
{
  var result = (s=>s.replace(/(?<=([aeiou]).*?[^\W\d_aeiou])(?![aeiou])/gi,(_,c)=>c.toLowerCase()))(test);
  console.log( result === tests[test], result );
}

mt0
источник
3

Japt -P , 28 байт

ó@\ctX ©\VtYÃËè\v ?P=D:D¬qPv

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

Распаковано и как это работает

UóXY{\ctX &&\VtY} mD{Dè\v ?P=D:Dq qPv

UóXY{           }  Split the string between any two chars that don't satisfy...
     \ctX &&\VtY     The first char is a consonant and the second is a non-vowel
mD{                And map...
   Dè\v              If this item is a vowel...
       ?P=D            Assign it to P and return as-is
           :Dq qPv     Otherwise, split the item into chars and join with P lowercased
                       (P starts with "", so beginning consonants are not affected)

-P                 Join with ""

óФункция побеждает любого рода регулярных выражений.

фонтанчик для питья
источник
Хороший, ты меня побил: D.
Волшебная урна осьминога
Прекрасно сделано - с этим у меня была сильная головная боль!
Мохнатый
2

Perl 6 ,  75 73 71  69 байт

{({S:i/.*(<[aeiou]>).*<-[\W\d_aeiou]><()><![aeiou]>/$0.lc()/}...*eq*).tail}

Попытайся

{({S:i{.*(<[aeiou]>).*<-[\W\d_aeiou]><()><![aeiou]>}=$0.lc}...*eq*).tail}

Попытайся

{({S:i{.*(<[aeiou]>).*<:L-[_aeiou]><()><![aeiou]>}=$0.lc}...*eq*).tail}

Попытайся

{({S:i{.*(<[aeiou]>).*<:L-[_aeiou]><(<![aeiou]>}=$0.lc}...*eq*).tail}

Попытайся

Expanded:

{  # bare block lambda with implicit parameter $_

  (
    # generate a sequence

    {  # code block used to generate the values

      S               # substitute (not in-place)
      :i              # :ignorecase
      {

          .*              # start at end of string

          ( <[aeiou]> )   # store the previous vowel in $0

          .*

          <:L - [_aeiou]> # letter other than a vowel

          <(              # ignore everything before this

                          # this is where `$0.lc` gets inserted

          # )>            # ignore everything after this

          <![aeiou]>      # not a vowel (zero width lookahead)

      } = $0.lc       # replace with lowercase of the earlier vowel
    }

    ...    # keep generating until:

    * eq * # there are two equal strings (no changes)

  ).tail   # get the last value
}
Брэд Гилберт b2gills
источник
2

Python 3 , 125 байт

lambda s,v='[^aeiouAEIOU':sub(f'(?<={v}\W\d])(?={v}]|$)',lambda m:sub(f'{v}]','',s[:m.end()])[-1:].lower(),s)
from re import*

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

Python 3.6 позволяет нам (ab) использовать f-строки для повторного использования нашего набора гласных (и для еще четырех сохраненных символов - начала инвертированного класса символов регулярного выражения) дешево ( fпрефикс в каждой строке, затем, {v}при необходимости, вместо '+v+'вам нужно с конкатенацией, или[^aeiouAEIOU вы вставите буквально.

Регулярное выражение, которое не соответствует ни одному символу, а только позиции, позволяет избежать проблем с неперекрывающимися совпадениями, которые требуются обычным регулярным выражениям, и устраняет необходимость обратной ссылки на любую часть соответствия; все, для чего мы используем объект соответствия, - это получить индекс слайса, который мы используем для поиска предыдущего гласного.

Частично де-гольф, это было бы что-то вроде:

import re

def get_last_vowel(string):
    '''
    Returns the lowercase version of the last vowel in a string if
    the string contains any vowels, otherwise, return the empty string
    '''
    try:
        *restvowels, lastvowel = re.sub(r'[^aeiouAEIOU]', '', string)
    except ValueError:
        lastvowel = ''  # No vowels in string
    return lastvowel.lower()

def rememebere_tehe_vowelese(string):
    '''Inserts the lowercased last vowel seen after any consonant not followed by a vowel'''
    return re.sub(r'(?<=[^aeiouAEIOU\W\d])(?=[^aeiouAEIOU]|$)',
                  lambda match: get_last_vowel(string[:match.end()]),
                  string)
ShadowRanger
источник
2

TSQL, 500 байт

 CREATE TABLE i (i CHAR(999)); INSERT i VALUES ('The rain in Spain stays mainly in the plain')
 DECLARE @w CHAR(999)=(SELECT i FROM i),@r VARCHAR(999)='';WITH d(n,c,i,q)AS(SELECT n,SUBSTRING(@w,n,1),CHARINDEX(SUBSTRING(@w,n,1),'AEIOUaeiou'),CHARINDEX(SUBSTRING(@w,n,1),'BCDFGHJKLMNPQRSTVWXYZbcdfghjklmnpqrstvwxyz')FROM(SELECT DISTINCT number n FROM master..[spt_values]WHERE number BETWEEN 1 AND LEN(@w))D)SELECT @r=@r+f.c+LOWER(COALESCE(CASE WHEN f.q<>0 AND COALESCE(d2.i,0)=0 THEN SUBSTRING(@w,(SELECT MAX(n)FROM d WHERE i<>0 AND n<f.n),1)END,''))FROM d f LEFT JOIN d d2 ON f.n=d2.n-1 SELECT @r

Таблица iиспользуется для ввода

Ян Дрозен
источник
2
Предполагается, что входные данные присутствуют в определенной переменной, как правило, не допускается . Может ли это быть адаптировано, чтобы быть функцией вместо этого?
Лайкони
Решение @Laikoni обновлено в соответствии с заданными правилами
Ян Дрозен
2

SWI-Пролог, 593 байта

a(S,D):-atom_chars(S,D).
g(_,[],_,-1).
g(E,[E|_],R,R).
g(E,[_|T],I,R):-N is I+1,g(E,T,N,R).
c(A,E):-g(E,A,0,R),R > -1.
v(X):-a('AEIOUaeiou',X).
c(X):-a('BCDFGHJKLMNPQRSTVWXYZbcdfghjklmnpqrstvwxyz',X).
d([_],_,R,R).
d([H|T],_,I,R):-v(V),c(V,H),!,d(T,H,I,[H|R]).
d([H,N|T],V,I,R):-c(C),c(C,H),v(W),c(W,N),!,d([N|T],V,I,[H|R]).
d([H,N|T],V,I,R):-c(C),c(C,H),v(W),\+c(W,N),string_lower(V,LV),!,d([N|T],V,I,[LV,H|R]).
d([H|T],V,I,R):-!,d(T,V,I,[H|R]).
r([],Z,Z).
r([H|T],Z,A):-r(T,Z,[H|A]).
r(S,D):-r(S,D,[]).
m(X,R):-a(X,O),r(O,P),r([''|P],Q),d(Q,'',I,[]),r(I,J,[]),atomic_list_concat(J,R).

Используются только встроенные предикаты (без использования регулярных выражений или библиотеки управления списками).

Использование:

?- m('A pnm bnn').
'A panama banana'
true .
Ян Дрозен
источник
2

Хаскелл , 142 130 байт

""&
import Data.Char
v=(`elem`"aeiouAEIOU")
s&(x:y:z)|v y=x:s&(y:z)
s&(x:y)|v x=x:[toLower x]&y|isAlpha x=x:s++s&y|1>0=x:s&y
_&x=x

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

Начальное ""&является частичным применением(&) функции, определенной позже, и расположено так странно, что TIO подсчитывает байты ""&, но не учитывает байты, которые в полной программе потребовались бы для присвоения этого любому именованному значению.


Менее гольф:

import Data.Char (isAlpha, toLower)

vowel :: Char -> Bool
vowel = (`elem`"aeiouAEIOU")

replace :: String -> String
replace = go "" -- start by carrying no extra vowel
  where go _ "" = ""
        -- special case for "anything followed by vowel" so later cases can ignore next character
        go s (x:y:more) | vowel y = x : go s (y:more)
        go s (x:xs) | vowel x = x : go [toLower x] xs -- update the vowel we're carrying
                    | isAlpha x = x : s ++ go s xs -- non-vowel letter not followed by a vowel
                    | otherwise = x : go s xs -- some non-letter junk, just include it and carry on

На самом деле должен быть способ сделать это более кратко с помощью сгиба вместо рекурсии, но я не мог понять это.

amalloy
источник
Вот очень хакерский способ определения заголовка, который fне появляется в теле: Попробуйте онлайн!
Лайкони
В нем есть два ненужных пробела, которые v = (вы можете определить gкак инфиксный оператор .
Лайкони
Установка базового регистра g _""=""в последнюю позицию экономит байт: g _ x=x(два байта, если вы переключитесь на инфикс, как предлагает Лайкони).
Ними
В соответствии с нашими соглашениями вам нужно добавить круглые скобки, ""&чтобы сделать его функцией.
Лайкони
1

05AB1E , 34 байта

vyžMylåil©1V}žPylåžM¹N>èå_Y&&i®«}J

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


Я забираю это назад, я могу сбрить только 3 байта от этого уродства ... Я думаю, что я мог бы сбрить логическое значение, но ДОЛЖНО быть 3 случая. 1 для гласных. 1 для согласных. 1 для случая, когда существует цифра / символ.


v                                 # For each...
 y                                # Push current element.
  žM                              # Push lower-case vowels (aeiou).
    ylå                           # Lower-case current element is vowel?
       i©1V}                      # If so, put it in register, set Y to 1.
            žP                    # Push lower-case consonants (b...z)
              ylå                 # Is current char a consonant?
                 žM¹N>èå_         # Push vowels again, is input[N+1] NOT a vowel? 
                         Y        # Did we ever set Y as 1?
                          &&      # All 3 previous conditions true?
                            i®«}  # Concat the current vowel to the current char.
                                J # Join the whole stack.
                                  # '}' isn't needed here, b/c it's implied.
                                  # Implicit return.
Урна волшебного осьминога
источник
0

Powershell, 104 байта

на основе регулярного выражения Нейла .

[regex]::Replace($args,'(?i)(?<=([aeiou]).*?[^\W\d_aeiou])(?![aeiou])',{"$($args.Groups[1])".ToLower()})

сохранить как get-rememebere.ps1. Скрипт для тестирования:

$test = @"
AN EXAMPLE WITH A LOT UPPERCASE (plus some lowercase)
And here comes a **TEST** case with 10% symbols/numbers(#)!
This is an example string.
abcdefghijklmnopqrstuvwxyz
A pnm bnn
Tell me if you need more test cases!
"@

$expected = @"
ANa EXAMaPaLE WITiHi A LOTo UPuPEReCASE (pelusu some lowerecase)
Anada here comese a **TESeTe** case witihi 10% siyimiboloso/numuberese(#)!
Thisi isi ana examapale seterinigi.
abacadefegehijikiliminopoqorosotuvuwuxuyuzu
A panama banana
Telele me ifi you neede more tesete casese!
"@

$result = .\get-rememebere.ps1 $test
$result -eq $expected
$result
Mazzy
источник
1
Разве это не только фрагмент? Я имею в виду, в PowerShell есть вход, поэтому вы не можете предполагать, что вход находится в $t. Соответствующее сообщение мета: codegolf.meta.stackexchange.com/a/8731/78123
wastl
0

Красный , 276 байт

func[s][v: charset t:"AEIOUaeiou"c: charset 
u:"BCDFGHJKLMNPQRSTVWXYZbcdfghjklmnpqrstvwxyz"b:
parse s[collect[any keep[thru c opt v]keep thru end]]p:""foreach
c b[either find t e: last c: to-string c[p: e][parse c[any[copy p v
| skip]]if find u e[append c lowercase p]]prin c]]

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

Удобочитаемый:

f: func [ s ] [
   v: charset t: "AEIOUaeiou"
   c: charset u: "BCDFGHJKLMNPQRSTVWXYZbcdfghjklmnpqrstvwxyz"
   b: parse s [
       collect [ any keep [ thru c opt v ]
       keep thru end ]
   ]
   p: "" 
   foreach c b [
       e: last c: to-string c
       either find t e [ p: e ][
           parse c [ any [ copy p v | skip ] ]
           if find u e [ append c lowercase p ]
       ]
       prin c
   ]
]
Гален Иванов
источник
0

YaBasic , 180 байт

Полная программа, которая принимает вход от STDIN и выводит на STDOUT

Line Input""s$
x$="AEIOUaeiou"
For i=1To Len(s$)
c$=Mid$(s$,i,1)
?c$;
If InStr(x$,c$)Then
v$=c$
Else
a=Asc(Upper$(c$))
If a>64And a<91And!InStr(x$,Mid$(s$,i+1,1))Then?v$;Fi
Fi
Next

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

Тейлор Скотт
источник