Розбираємо формат словника Bookworm


42

Нещодавно я потурав собі ностальгією у вигляді Bookworm Deluxe:

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

aa
2h
3ed
ing
s
2l
3iis
s
2rdvark
8s
4wolf
7ves

Правила розпакування словника прості:

  1. Прочитайте число на початку рядка та скопіюйте стільки символів з початку попереднього слова. (Якщо немає номера, скопіюйте стільки символів, скільки минулого разу.)

  2. Додайте наступні літери до слова.

Отже, наше перше слово aa, а потім 2h, що означає «копіюємо перші дві букви aaі Append h," формування aah. Потім 3edстає aahed, і оскільки наступний рядок не має числа, ми знову копіюємо 3 символи у форму aahing. Цей процес триває впродовж решти словника. Отримані слова з невеликого введення зразка:

aa
aah
aahed
aahing
aahs
aal
aaliis
aals
aardvark
aardvarks
aardwolf
aardwolves

Ваше завдання полягає в тому, щоб виконати розпакування в якомога менше байтах.

Кожен рядок введення буде містити нуль або більше цифр, що 0-9 супроводжуються однією або кількома малими літерами a-z. Ви можете приймати введення та давати вихід у вигляді списку рядків, або як єдиний рядок зі словами, розділеними будь-яким символом, окрім 0-9/a-z .

Ось ще один невеликий тестовий випадок із кількома крайовими випадками, не висвітленими у прикладі:

abc cba 1de fg hi 0jkl mno abcdefghijk 10l
=> abc cba cde cfg chi jkl mno abcdefghijk abcdefghijl

Ви також можете перевірити свій код у повному словнику: введення , виведення .


Чи є можливість у другому рядку не бути числа? Крім того, чи можна вважати, що жодне число, окрім, 0не матиме провідних 0s?
Ерік Аутгольфер

@EriktheOutgolfer Так, це можливо; Я додав це до тестового випадку. І так, ви можете припустити, що (а також, що число не буде більше довжини попереднього слова).
Дверна ручка

11
Ось такий симпатичний формат стиснення:]
Poke

1
locateПрограма використовує цей тип кодування на колійних іменах.
Дан Д.

Я написав цю програму для власного використання, близько 15 років тому. На жаль, я більше не думаю, що у мене джерело ...
hobbs

Відповіді:


13

Vim, 57 байт

:%s/\a/ &
:%norm +hkyiwjP
:g/\d/norm diw-@"yl+P
:%s/ //g

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


Чи <H<Gзамість останньої заміни працюватиме?
Kritixi Lithos

@cowsquack На жаль, ні. Кожен ввід, який не починається з числа, збільшує кількість провідних пробілів, тому немає жодного способу гарантувати, що <рішення буде невідповідним.
DJMcMayhem

Я думаю, що ви можете зробити :%s/ *замість останньої заміни, щоб зберегти два байти.
CD Dexter

10

JavaScript (ES6),  66 62  61 байт

a=>a.map(p=s=>a=a.slice([,x,y]=/(\d*)(.*)/.exec(s),p=x||p)+y)

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

Прокоментував

a =>                  // a[] = input, re-used to store the previous word
  a.map(p =           // initialize p to a non-numeric value
  s =>                // for each string s in a[]:
    a =               //   update a:
      a.slice(        //     extract the correct prefix from the previous word:
        [, x, y] =    //       load into x and y:
          /(\d*)(.*)/ //         the result of a regular expression which splits the new
          .exec(s),   //         entry into x = leading digits and y = trailing letters
                      //       this array is interpreted as 0 by slice()
        p = x || p    //       update p to x if x is not an empty string; otherwise leave
                      //       it unchanged; use this as the 2nd parameter of slice()
      )               //     end of slice()
      + y             //     append the new suffix
  )                   // end of map()

5

Perl 6 , 50 48 байт

-2 байти завдяки nwellnhof

{my$l;.map:{$!=S[\d*]=substr $!,0,$l [R||]=~$/}}

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

Порт рішення Арнальда . Людина, цей R||трюк був у американських гірках від "я думаю, що це могло бути можливим", до "ні, це неможливо", до "свого роду, можливо, можливо" і, нарешті, "ага!"

Пояснення:

{my$l;.map:{$!=S[\d*]=substr $!,0,$l [R||]=~$/}}
{                                              }  # Anonymous code block
 my$l;    # Declare the variable $l, which is used for the previous number
      .map:{                                  }  # Map the input list to
            $!=              # $! is used to save the previous word
               S[\d*]=       # Substitute the number for
                      substr $!,0    # A substring of the previous word
                                 ,              # With the length of 
                                           ~$0     # The num if it exists
                                  $l [R||]=        # Otherwise the previous num

$l [R||]=~$/Частина приблизно перекладається , $l= ~$/||+$lале ... це має таке ж кількість байтів :(. Спочатку він зберігав байти, використовуючи анонімну змінну, тому my$lпішов, але це не працює, оскільки область тепер є заміною, а не mapблоком коду. Ну добре. У будь-якому випадку Rце зворотний метаоператор, тому він обертає аргументи ||, тому $lзмінній в кінцевому підсумку присвоюється нове число (~$/ ), якщо воно існує, інакше знову.

Це може бути 47 байт, якби Perl 6 не кинув якусь надлишкову помилку компілятора для =~.


5

Рубі , 49 45 43 байт

$0=$_=$0[/.{0#{p=$_[/\d+/]||p}}/]+$_[/\D+/]

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

Пояснення

$0=                                         #Previous word, assign the value of
   $_=                                      #Current word, assign the value of
      $0[/.{0#{              }}/]           #Starting substring of $0 of length p which is
               p=$_[/\d+/]||p               #defined as a number in the start of $_ if any 
                                 +$_[/\D+/] #Plus any remaining non-digits in $_

5

C, 65 57 байт

n;f(){char c[99];while(scanf("%d",&n),gets(c+n))puts(c);}

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

Пояснення:

n;                     /* n is implicitly int, and initialized to zero. */

f() {                  /* the unpacking function. */

    char c[99];        /* we need a buffer to read into, for the longest line in
                          the full dictionary we need 12 + 1 bytes. */

    while(             /* loop while there is input left. */

        scanf("%d",&n) /* Read into n, if the read fails because this line
                          doesn't have a number n's value does not change.
                          scanf's return value is ignored. */

        ,              /* chain expressions with the comma operator. The loop
                          condition is on the right side of the comma. */

        gets(c+n))     /* we read into c starting from cₙ. c₀, c₁.. up to cₙ is
                          the shared prefix of the word we are reading and the
                          previous word. When gets is successful it returns c+n
                          else it will return NULL. When the loop condition is
                          NULL the loop exits. */

        puts(c);}      /* print the unpacked word. */

5

мозковий ебать , 201 байт

,[[[-<+>>>+<<]>-[---<+>]<[[-<]>>]<[-]>>[<<,>>>[-[-<++++++++++>]]++++<[->+<]-[----->-<]<]<]>>>[[>>]+[-<<]>>[[>>]+[<<]>>-]]+[>>]<[-]<[<<]>[->[>>]<+<[<<]>]>[>.>]+[>[-]<,.[->+>+<<]>>----------]<[<<]>-<<<,]

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

Потрібен останній рядок у кінці вводу. Версія без цієї вимоги на 6 байт довше:

мозковий ебать , 207 байт

,[[[-<+>>>+<<]>-[---<+>]<[[-<]>>]<[-]>>[<<,>>>[-[-<++++++++++>]]++++<[->+<]-[----->-<]<]<]>>>[[>>]+[-<<]>>[[>>]+[<<]>>-]]+[>>]<[-]<[<<]>[->[>>]<+<[<<]>]>[>.>]+[>[-]<,[->+>+<<]>>[----------<.<]>>]<[<<]>-<<<,]

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

Обидві версії припускають, що всі цифри суворо менше 255.

Пояснення

Стрічка викладається так:

tempinputcopy 85 0 inputcopy number 1 a 1 a 1 r 1 d 0 w 0 o 0 l 0 f 0 ...

Осередок "число" дорівнює 0, якщо не вводяться цифри, і n + 1, якщо вводиться число n. Введення приймається у комірці з позначкою "85".

,[                     take input and start main loop
 [                     start number input loop
  [-<+>>>+<<]          copy input to tempinputcopy and inputcopy
  >-[---<+>]           put the number 85 in the cell where input was taken
  <[[-<]>>]            test whether input is less than 85; ending position depends on result of comparison
                       (note that digits are 48 through 57 while letters are 97 through 122)
  <[-]>                clean up by zeroing out the cell that didn't already become zero
  >[                   if input was a digit:
   <<,>>               get next input character
   >[-[-<++++++++++>]] multiply current value by 10 and add to current input
   ++++                set number cell to 4 (as part of subtracting 47)
   <[->+<]             add input plus 10*number back to number cell
   -[----->-<]         subtract 51
  <]                   move to cell we would be at if input were a letter
 <]                    move to input cell; this is occupied iff input was a digit

                       part 2: update/output word

 >>>                   move to number cell
 [                     if occupied (number was input):
  [>>]+[-<<]>>         remove existing marker 1s and decrement number cell to true value
  [[>>]+[<<]>>-]       create the correct amount of marker 1s
 ]
 +[>>]<[-]             zero out cell containing next letter from previous word
 <[<<]>                return to inputcopy
 [->[>>]<+<[<<]>]      move input copy to next letter cell
 >[>.>]                output word so far
 +[                    do until newline is read:
  >[-]<                zero out letter cell
  ,.                   input and output next letter or newline
  [->+>+<<]            copy to letter cell and following cell
  >>----------         subtract 10 to compare to newline
 ]
 <[<<]>-               zero out number cell (which was 1 to make copy loop shorter)
 <<<,                  return to input cell and take input
]                      repeat until end of input

4

Python 3.6+, 172 195 156 123 122 121 104 байти

import re
def f(l,n=0,w=""):
 for s in l:t=re.match("\d*",s)[0];n=int(t or n);w=w[:n]+s[len(t):];yield w

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

Пояснення

Я прогинався і використовував регулярні вирази. Це заощадило щонайменше 17 байт. :

t=re.match("\d*",s)[0]

Якщо рядок взагалі не починається з цифри, довжина цього рядка буде 0. Це означає що:

n=int(t or n)

буде, nякщо tпорожньо, int(t)інакше.

w=w[:n]+s[len(t):]

видаляє число, з якого знайдено звичайний вираз s(якщо немає знайденого числа, воно видалить 0символи, залишивши sневченими) та замінить усі, крім перших nсимволів попереднього слова, поточним фрагментом слова; і:

yield w

виводить поточне слово.


4

Haskell, 82 81 байт

tail.map concat.scanl p["",""]
p[n,l]a|[(i,r)]<-reads a=[take i$n++l,r]|1<2=[n,a]

Бере та повертає список рядків.

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

        scanl p["",""]        -- fold function 'p' into the input list starting with
                              -- a list of two empty strings and collect the
                              -- intermediate results in a list
  p [n,l] a                   -- 1st string of the list 'n' is the part taken form the last word
                              -- 2nd string of the list 'l' is the part from the current line
                              -- 'a' is the code from the next line
     |[(i,r)]<-reads a        -- if 'a' can be parsed as an integer 'i' and a string 'r'
       =[take i$n++l,r]       -- go on with the first 'i' chars from the last line (-> 'n' and 'l' concatenated) and the new ending 'r'
     |1<2                     -- if parsing is not possible
       =[n,a]                 -- go on with the previous beginning of the word 'n' and the new end 'a'
                              -- e.g. [         "aa",     "2h",      "3ed",       "ing"       ] 
                              -- ->   [["",""],["","aa"],["aa","h"],["aah","ed"],["aah","ing"]]
  map concat                  -- concatenate each sublist
tail                          -- drop first element. 'scanl' saves the initial value in the list of intermediate results. 

Редагувати: -1 байт завдяки @Nitrodon


1
Всупереч звичній мудрості Haskell для гольфу, ви можете фактично зберегти один байт, не визначаючи хелперну функцію як оператора інфіксації.
Нітродон

@Nitrodon: добре помічений! Дякую!
німі

3

Japt, 19 18 17 байт

Спочатку надихнув рішення JS Arnauld .

;£=¯V=XkB ªV +XoB

Спробуй це

                      :Implicit input of string array U
 £                    :Map each X
   ¯                  :  Slice U to index
      Xk              :    Remove from X
;       B             :     The lowercase alphabet (leaving only the digits or an empty string, which is falsey)
          ªV          :    Logical OR with V (initially 0)
    V=                :    Assign the result to V for the next iteration
             +        :  Append
              Xo      :  Remove everything from X, except
;               B     :   The lowercase alphabet
  =                   :  Reassign the resulting string to U for the next iteration

2

Желе , 16 байт

⁹fØDVo©®⁸ḣ;ḟØDµ\

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

Як це працює

⁹fØDVo©®⁸ḣ;ḟØDµ\  Main link. Argument: A (array of strings)

              µ\  Cumulatively reduce A by the link to the left.
⁹                     Yield the right argument.
  ØD                  Yield "0123456789".
 f                    Filter; keep only digits.
    V                 Eval the result. An empty string yields 0.
     o©               Perform logical OR and copy the result to the register.
       ®              Yield the value in the register (initially 0).
        ⁸ḣ            Head; keep that many character of the left argument.
          ;           Concatenate the result and the right argument.
            ØD        Yield "0123456789".
           ḟ          Filterfalse; keep only non-digits.


1

Сітківка 0,8,2 , 69 байт

+`((\d+).*¶)(\D)
$1$2$3
\d+
$*
+m`^((.)*(.).*¶(?<-2>.)*)(?(2)$)1
$1$3

Спробуйте в Інтернеті! Посилання включає складніші тестові справи. Пояснення:

+`((\d+).*¶)(\D)
$1$2$3

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

\d+
$*

Перетворити число в одинакове.

+m`^((.)*(.).*¶(?<-2>.)*)(?(2)$)1
$1$3

Використовуйте балансуючі групи, щоб замінити всі 1s відповідним буквою з попереднього рядка. (Це виявляється трохи грішніше, ніж заміщення всіх циклів 1.)




1

Groovy , 74 байти

{w="";d=0;it.replaceAll(/(\d*)(.+)/){d=(it[1]?:d)as int;w=w[0..<d]+it[2]}}

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

Пояснення:

{                                                                        }  Closure, sole argument = it
 w="";d=0;                                                                  Initialize variables
          it.replaceAll(/(\d*)(.+)/){                                   }   Replace every line (since this matches every line) and implicitly return. Loop variable is again it
                                     d=(it[1]?:d)as int;                    If a number is matched, set d to the number as an integer, else keep the value
                                                        w=w[0..<d]+it[2]    Set w to the first d characters of w, plus the matched string


0

Perl 5 -p , 45 41 байт

s:\d*:substr($p,0,$l=$&+$l*/^\D/):e;$p=$_

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

Пояснення:

s:\d*:substr($p,0,$l=$&+$l*/^\D/):e;$p=$_ Full program, implicit input
s:   :                           :e;      Replace
  \d*                                       Any number of digits
      substr($p,0,              )           By a prefix of $p (previous result or "")
                  $l=  +                      With a length (assigned to $l) of the sum
                     $&                         of the matched digits
                          *                     and the product
                        $l                        of $l (previous length or 0)
                           /^\D/                  and whether there is no number in the beginning (1 or 0)
                                                (product is $l if no number)
                                    $p=$_ Assign output to $p
                                          Implicit output


0

05AB1E , 20 19 17 байт

õUvyþDõÊi£U}Xyá«=

Спробуйте в Інтернеті або перевірте всі тестові випадки .

Пояснення:

õ                  # Push an empty string ""
 U                 # Pop and store it in variable `X`
v                  # Loop `y` over the (implicit) input-list
 yþ                #  Push `y`, and leave only the digits (let's call it `n`)
   DõÊi  }         #  If it's NOT equal to an empty string "":
       £           #   Pop and push the first `n` characters of the string
        U          #   Pop and store it in variable `X`
          X        #  Push variable `X`
           yá      #  Push `y`, and leave only the letters
             «     #  Merge them together
              =    #  Print it (without popping)

0

Лист звичайний, 181 байт

(do(w(p 0))((not(setf g(read-line t()))))(multiple-value-bind(a b)(parse-integer g :junk-allowed t)(setf p(or a p)w(concatenate'string(subseq w 0 p)(subseq g b)))(format t"~a~%"w)))

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

Безголівки:

(do (w (p 0))   ; w previous word, p previous integer prefix (initialized to 0)
    ((not (setf g (read-line t ()))))   ; read a line into new variable g
                                        ; and if null terminate: 
  (multiple-value-bind (a b)            ; let a, b the current integer prefix
      (parse-integer g :junk-allowed t) ; and the position after the prefix
    (setf p (or a p)                    ; set p to a (if nil (no numeric prefix) to 0)
          w (concatenate 'string        ; set w to the concatenation of prefix
             (subseq w 0 p)             ; characters from the previous word 
             (subseq g b)))             ; and the rest of the current line
    (format t"~a~%"w)))                 ; print the current word

Як завжди, довгі ідентифікатори Common Lisp роблять його не особливо придатним для PPCG.



0

C # (Visual C # Interactive Compiler) , 134 байти

a=>{int l=0,m,n;var p="";return a.Select(s=>{for(m=n=0;s[m]<58;n=n*10+s[m++]-48);return p=p.Substring(0,l=m>0?n:l)+s.Substring(m);});}

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

-9 байт завдяки @ASCIIOnly!

Менше гольфу ...

// a is an input list of strings
a=>{
  // l: last prefix length
  // m: current number of digits
  // n: current prefix length
  int l=0,m,n;
  // previous word
  var p="";
  // run a LINQ select against the input
  // s is the current word
  return a.Select(s=>{
    // nibble digits from start of the
    // current word to build up the
    // current prefix length
    for(m=n=0;
      s[m]<58;
      n=n*10+s[m++]-48);
    // append the prefix from the
    // previous word to the current
    // word and capture values
    // for the next iteration
    return
      p=p.Substring(0,l=m>0?n:l)+
      s.Substring(m);
  });
}


Це досить круто :) я змінив l=n>0?n:lна , l=m>0?n:lтому що це не збирання випадок , коли лінія почала з нуля ( 0jkl). Дякую за пораду!
дата

0

Скала , 226 129 102 байт

Дякуємо @ ASCII лише за їх роботу тут (і за відповідь Groovy).

s=>{var(w,d,r)=("",0,"(\\d*)(.+)".r)
s map(_ match{case r(a,b)=>{if(a>"")d=a.toInt
w=w.take(d)+b}
w})}

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


: | обидва посилання однакові
лише для ASCII

так, редагування. Я не знав, як це вимкнути, і поспішав, тому не змінив те, що робив.
В. Куртуа



Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.