Znajdź synchronizację

33

Biorąc pod uwagę ciąg znaków składający się w całości z qs reprezentujących ćwierćnuty esi reprezentujących ósme nuty, wyprowadzaj indeksy ćwierćnuty, które są zsynchronizowane.

Synchronizacja jest złożona, ale na potrzeby tego wyzwania nasza definicja „synchronizacji” będzie bardzo prosta: ćwierćnuta rozpoczynająca się od „off-beat” - to znaczy, uderzenia liczone jako „i” w n / 4 czas.

Może to być alternatywnie zdefiniowane jako dowolna ćwierćnuta poprzedzona nieparzystą liczbą ósmych nut. Na przykład notatki oznaczone *poniżej są uważane za zsynchronizowane, a ich indeksy są również wyświetlane:

eqqeqqeqqe
 **    **
 12    78
Output: 1 2 7 8

Dane wejściowe zawsze będą składały się z całej liczby taktów w czasie 4/4 (ćwierćnuta to ćwierć taktu, a ósma nuta to ósma taktu). (Dane wejściowe również nigdy nie będą puste.) Dane wyjściowe mogą być pojedynczym ciągiem znaków z elementami oddzielonymi dowolnym separatorem, który nie zawiera liczb lub tablicy / listy / itp. Dane wyjściowe mogą być oparte na 1 (tzn. Pierwszy indeks to 1 zamiast 0), jeśli chcesz, i może być również w dowolnej bazie liczbowej (jednostkowa, dziesiętna itp.).

Ponieważ jest to , wygrywa najkrótszy kod w bajtach.

Przypadki testowe:

In                        Out
-----------------------------------------------
eqqqe                     1 2 3
qeqeq                     2
qqqeqqeeeeqeqeqeqqeqqeqq  4 5 10 14 19 20
eeeeeqeeqeeqqqqeqeqeeqe   5 8 11 12 13 14 18 21
qqqq                      <none>
eeeeeeee                  <none>
Klamka
źródło
1
Czy wyjście może być oparte na 1?
Luis Mendo,
1
Czy możesz zrobić działający przykład, aby pokazać, jak działają indeksy?
Peter Taylor
1
@LuisMendo Pewnie, jeśli spowoduje to skrócenie kodu.
Klamka
@PeterTaylor Dobra, czy jest coś takiego, o czym myślałeś?
Klamka
Czy dane wejściowe mogą być ciągiem zawierającym znaki cudzysłowu? 'eqqqe'zamiasteqqqe
Luis Mendo

Odpowiedzi:

12

Galaretka , 12 9 bajtów

=“e”µ<^\O

Jako program powyższy kod wymaga cudzysłowu wokół danych wejściowych. Ponieważ nie jest to dozwolone, jest to przesyłanie funkcji. Dane wyjściowe są oparte na 1. Wypróbuj online!

Jak to działa

=“e”µ<^\O    Monadic link. Argument: s (string)

=“e”         Check each character for equality with 'e'. Yields a Boolean array.
    µ        Start a new, monadic chain.
      ^\     Compute the array of partial reductions by XOR, i. e., the parities
             of all prefixes of the Boolean array.
     <       Check if the Booleans are strictly smaller than the parities.
             A truthy outcome indicates an off-beat quarter note.
        O    Yield all indices of 1's.

Aktualizacja

Powyższy kod nie działa już w najnowszej wersji Jelly, ponieważ potrzebujemy znaku e , ale “e”daje ciąg znaków. Naprawa, która oszczędza bajt, w sumie 8 bajtów .

=”eµ<^\O

Działa to jako pełny program. Wypróbuj online!

Dennis
źródło
7

Ruby, 46

i=e=0
gets.bytes{|n|e^=n
e&4|n>114&&p(i)
i+=1}

Wejście do standardowego wejścia. Wyjście na standardowe wyjście, oddzielone znakiem nowej linii.

Skomentował

i=e=0               #i keeps index, e keeps track of 8ths.
gets.bytes{|n|      #iterate through bytes in the input
e^=n                #xor e with input. We're interested in the 4's bit, which is only affected by ascii e, not ascii q
e&4|n>114&&p(i)     #e&4 evaluates to 4 or 0. OR with n and if the value is greater than ascii code for q, print index
i+=1}               #increment index
Level River St
źródło
6

JavaScript ES7, 50 48 bajtów

Krótko mówiąc o JS, jeśli mnie pytasz. [for...of]Składnia, w zasadzie połączona mapa i filtr, jest przydatna w przypadku tego wyzwania.

s=>[for(c of(i=f=0,s))if(++i&&c>'e'?f%2:f++&0)i]

Definiuje anonimową funkcję, która generuje tablicę 1-indeksowaną.

Testowy fragment kodu

Korzysta z niezgolfowanej wersji kodu ES7'd.

a = function(s) {   // Create a function a that takes in a parameter s and does these things:
  var r = [],       // Set variable r to an empty array,
  i = 0, f = 0;     // i to 0, and f to 0.
  for(c of s) {     // For each character c in s:
    i++;            //  Increment i by 1.
    if(             //  If
      c == 'q' ?    //   if c == 'q',
      f%2 === 1 :   //    f is even; otherwise,
      f++ && false) //    increment f and don't execute this:
      r.push(i);    //   Add i to the end of r.
  } return r;       // Return r.
}
<input type="text" value="eqqqe" id=O />
<button onclick="P.innerHTML='['+a(O.value)+']'">Try it</button>
<p id=P />

ETHprodukcje
źródło
3
Bardzo dobre wytłumaczenie! A także świetny przykład użycia nowego [dla ... z] ES7]
Aᴄʜᴇʀᴏɴғᴀɪʟ
Czy zatem potrzebujemy nowego pytania „Wskazówki dotyczące gry w ECMAScript 7”?
Neil
@Neil Próbowałem zaktualizować post ES6 do ES6 / 7, ale OP wycofał edycję. Tymczasem jest to: codegolf.stackexchange.com/a/61489/42545
ETHproductions
5

J, 20 19 17 bajtów

=&'e'(I.@:<~:/\@)

Dzięki randomra za uratowanie bajtu i Dennisowi za uratowanie dwóch. Jest to bezimienny czasownik monadyczny, używany w następujący sposób:

  f =: =&'e'(I.@:<~:/\@)
  f 'eqqqe'
1 2 3

Wypróbuj tutaj.

Wyjaśnienie

=&'e'(I.@:<~:/\@)
=&'e'               Replace every 'e' with 1, other chars with 0
     (         @)   Apply the verb in parentheses to the resulting 0-1 vector
           ~:/\     Cumulative reduce with XOR (parity of 'e'-chars to the left)
          <         Element-wise less-than with original vector
      I.@:          Positions of 1s in that vector
Zgarb
źródło
5

GNU grep, 3 + 17 = 20 3 + 15 = 18 bajtów

Program wymaga opcji boP. Kod jest

q(?!(q|eq*e)*$)

Zapisz jako synco, a następnie uruchom jako grep -boPf synco.

Po separatorze wyjściowym :qnastępuje nowa linia. Np. Wyjściem eqqqejest

1:q
2:q
3:q

Znaczenia flag to:

  • P: Użyj wyrażeń regularnych PCRE.
  • o: Oznacza to, że wypisuje tylko tę część wiersza, która pasuje do wyrażenia regularnego, ale nie dlatego jest to ważne. ojest używany, ponieważ pozwala na wielokrotne dopasowanie w linii.
  • b: Wydrukuj przesunięcie w bajtach początku każdego dopasowania od początku pliku.

Wzór sprawdza, czy po ćwierćnecie nie ma parzystej liczby ósemek.

feersum
źródło
Czy sam w sobie grepkwalifikuje się jako język? Niezależnie od tego +1 za świetną odpowiedź
Digital Trauma
@DigitalTrauma Nie rozumiem, dlaczego nie ... Może używać wyrażeń regularnych PCRE, więc powinien być co najmniej kompletny Turing i może wykonywać kod z pliku, jak pokazano tutaj.
feersum
Miałem wrażenie, że nie udowodniono, że PCRE jest kompletny w Turingu. Niezależnie od tego, twój wyraz twarzy spełnia ten wymóg, więc zgadzam się z tym, ale mogą istnieć inni z skargami z przyczyn teoretycznych.
Cyfrowy uraz
@DigitalTrauma Huh, wygląda na to, że mylę się co do kompletności Turinga.
feersum
5

MATL , 12 14 16 bajtów

j101=tYs2\<f

Podziękowania dla Dennisa za usunięcie 2 bajtów (i za hosting MATL-a na jego niesamowitej platformie online!)

Używa bieżącej wersji (9.3.0) języka / kompilatora.

Wejście i wyjście odbywa się poprzez standardowe wejście i standardowe wyjście. Wynik jest oparty na 1.

Przykład :

>> matl j101=tYs2\<f
> eeeeeqeeqeeqqqqeqeqeeqe
6  9 12 13 14 15 19 22

Lub wypróbuj online!

Wyjaśnienie

j             % input string
101=          % vector that equals 1 at 'e' characters and 0 otherwise
t             % duplicate
Ys2\          % cumulative sum modulo 2
<             % detect where first vector is 0 and second is 1
f             % find (1-based) indices of nonzero values
Luis Mendo
źródło
3

Python 2, 94 85 79 75 66 bajtów

EDYCJA: Dzięki Doorknob i Alex A.

EDYCJA: Dzięki Alex A.

EDYCJA: Teraz używa input (), więc wejście musi być ciągiem z cudzysłowami.

EDYCJA: Dzięki Zgarb za zalecenie, abym użył wyliczenia.

Po prostu liczy liczbę e, a jeśli q, sprawdź, czy liczba e jest nieparzysta, a następnie wydrukuj indeks.

e=0
for j,k in enumerate(input()):
 if"q">k:e+=1
 elif e%2:print j

Wypróbuj tutaj

TanMath
źródło
Możesz zamienić drugi na „ if ...tylko”, elseaby zaoszczędzić 8 bajtów.
Klamka
Możesz również usunąć spację po print1 bajcie
Alex A.
Myślę, że możesz zmienić na else: if e%2:sprawiedliwy elif e%2:.
Alex A.
Możesz zapisać jeszcze jeden bajt, zaznaczając i[j]<"q"zamiast i[j]=="e".
Alex A.,
2
@TanMath Poprosiłem Doorknob, ponieważ zaoszczędzi mi to 2 bajty na wprowadzenie danych z cudzysłowami. Ale nie da się tego zrobić
Luis Mendo,
3

Haskell, 58 51 bajtów

f x=[i|(i,'q')<-zip[0..]x,odd$sum[1|'e'<-take i x]]

Przykład użycia: f "eeeeeqeeqeeqqqqeqeqeeqe"-> [5,8,11,12,13,14,18,21].

Przejrzyj listę i wypisz bieżący indeks idla każdego znaku, 'q'jeśli przed nim jest nieparzysta liczba 'e's.

nimi
źródło
2

Minkolang 0,15 , 28 bajtów

(o"q"=7&z1+$z8!z2%,2&iN$I$).

Wypróbuj tutaj.

Wyjaśnienie

(                        Open while loop
 o                       Read in character from input
  "q"                    Push the character "q"
     =                   1 if the top two items on stack are equal, 0 otherwise
      7&                 Pop and jump 7 spaces if truthy

        z                Push register value on stack
         1+              Add one
           $z            Pop top of stack and store in register
             8!          Jump eight spaces

        z                Push register value on stack
         2%              Modulo by 2
           ,             boolean not
            2&           Pop and jump two spaces if truthy
              i          Push loop counter
               N         Output as number

                $I       Push length of input
                  $).    Close while loop when top of stack is 0 and stop.
El'endia Starman
źródło
2

C (funkcja), 65

Dzięki @Dennis za dodatkowe gry w golfa!

i,n;f(char*m){for(i=n=0;*m;i++)*m++&4?++n:n%2?printf("%d ",i):0;}
Cyfrowa trauma
źródło
1
Myślę, że i,n;f(char*m){for(i=n=0;*m;m++,i++)*m&4?++n:n%2?printf("%d ",i):0;}powinien działać.
Dennis
2

Python 3, 109 95 80 90 88 76 68 67 66 64 bajtów

Liczy liczbę qsi esi dodaje indeks prądu, qjeśli liczba poprzedzających es jest nieparzysta.

Edycja: Teraz wypisuje listę indeksów s, które poprzedzają je qnieparzystą liczbą es. Osiem bajtów zaoszczędzonych dzięki Klamce i dwa kolejne dzięki Feersum .

lambda s:[x for x,m in enumerate(s)if("e"<m)*s[:x].count("e")%2]

Nie golfowany:

def f(s):
    c = []
    for index, item in enumerate(s):
        if item == "q":
            if s[:index].count("e")%2 == 1:
                c.append(index)
    return c
Sherlock9
źródło
1
Czy nie mógłbyś uczynić z tego lambda, aby niepotrzebne były oświadczenia inputi print?
Klamka
Powinien być krótszy w użyciu enumerateniż range(len(....
feersum
2

JavaScript ES6, 63 60 58 bajtów

x=>[...x].map((e,i)=>e>'e'?n%2&&a.push(i):n++,a=[],n=0)&&a

Anonimowa funkcja generująca tablicę. Dziękujemy użytkownikowi 81655 za zapisanie dwóch bajtów. Oto wersja bez golfa, która wykorzystuje lepiej obsługiwaną składnię.

f=function(x) {
  a=[] // Indeces of syncopated notes
  n=0 // Number of e's encountered so far
  x.split('').map(function(e,i) { // For each letter...
    e>'e'? // If the letter is q...
      n%2&& // ...and the number of e's is odd...
        a.push(i): // ...add the current index to the array
      n++ // Otherwise, it is e so increment the counter
  })
  return a
}

run=function(){document.getElementById('output').textContent=f(document.getElementById('input').value)};document.getElementById('run').onclick=run;run()
<input type="text" id="input" value="qqqeqqeeeeqeqeqeqqeqqeqq" /><button id="run">Run</button><br />
<samp id="output"></samp>

NinjaBearMonkey
źródło
0

Mathematica, 76 bajtów

Flatten[Range[#+1,#2-1]&@@@StringPosition[#,"e"~~"q"..~~"e",Overlaps->1<0]]&

Zauważyłem coś interesującego. Wszystkie zsynchronizowane części są formalne eqqq..qqe, więc po prostu je wykrywam i podam wskaźniki qs.

LegionMammal978
źródło
0

Japt, 29 23 21 bajtów

Już nie konkuruje!

0+U ¬®¥'e} å^ ä© m© f

Wypróbuj online!

Jak to działa

         // Implicit: U = input string, e.g.    "eqqeqeq"
0+U      // Add a 0 to the beginning.           "0eqqeqeq"
¬        // Split into chars.                   ['0,'e,'q,'q,'e,'q,'e,'q]
®¥'e}    // Map each item X to (X == 'e).       [F, T, F, F, T, F, T, F]
å^       // Cumulative reduce by XOR'ing.       [0, 1, 1, 1, 0, 0, 1, 1]
ä©       // Map each consecutive pair with &&.  [0, 1, 1, 0, 0, 0, 1]
m©       // Map each item with &&. This performs (item && index):
         //                                     [0, 1, 2, 0, 0, 0, 6]
f        // Filter out the falsy items.         [   1, 2,          6]
         // Implicit output                     [1,2,6]

Wersja niekonkurencyjna, 18 bajtów

U¬m¥'e å^ ä©0 m© f

Wypróbuj online!

ETHprodukcje
źródło
0

Befunge, 43 bajty

:~:0\`#@_5%2/:99p1++\>2%#<9#\9#.g#:*#\_\1+\

Wypróbuj online!

Wyjaśnienie

Zaczynamy od dwóch domyślnych zer na stosie: numer nuty i liczba uderzeń.

:               Make a duplicate of the beat count.
~               Read a character from stdin.
:0\`#@_         Exit if it's less than zero (i.e. end-of-file).
5%2/            Take the ASCII value mod 5, div 2, translating q to 1 and e to 0.
:99p            Save a copy in memory for later use.
1+              Add 1, so q maps to 2 and e to 1.
+               Then add that number to our beat count.
\               Get the original beat count that we duplicated at the start.
2%              Mod 2 to check if it's an off-beat.
99g*            Multiply with the previously saved note number (1 for q, 0 for e).
_               Essentially testing if it's a quarter note on an off-beat.
       \.:\     If true, we go turn back left, get the beat count, and output it.
         >2     Then push 2 onto the stack, and turn right again.
2%              That 2 modulo 2 is just zero.
99g*            Then multiplied by the saved note number is still zero.
_               And thus we branch right on the second pass.
\1+\            Finally we increment the note number and wrap around to the start again.
James Holderness
źródło