Создать последовательность Recamán


20

Последовательность Recamán ( A005132 ) является математической последовательностью, определяемой как:

A(0) = 0
A(n) = A(n-1) - n if A(n-1) - n > 0 and is new, else
A(n) = A(n-1) + n

Красивая версия LaTex выше (может быть более читабельным):

A(n)={0if n=0A(n1)nif A(n1)n is positive and not already in the sequenceA(n1)+notherwise

Первые несколько терминов 0, 1, 3, 6, 2, 7, 13, 20, 12, 21, 11

Чтобы уточнить, is newозначает ли номер уже в последовательности.

Получив целое число nчерез аргумент функции или STDIN, вернуть первые nчлены последовательности Recamán.


Это задача для игры в гольф, поэтому выигрывает самый короткий код.


Что значит «новый»?
бета-распад

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

Исправил последовательность.
Джеймс Уильямс

1
Можете ли вы добавить первые значения последовательности?
гордый haskeller

Добавлены первые несколько номеров! (И ссылка на его страницу OEIS)
Джеймс Уильямс

Ответы:


9

CJam, 34 33 байта

0ali{_W=_I-__0<4$@#)|@I+@?+}fI1>`

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

Пример запуска

$ cjam <(echo '0ali{_W=_I-__0<4$@#)|@I+@?+}fI1>`') <<< 33
[0 1 3 6 2 7 13 20 12 21 11 22 10 23 9 24 8 25 43 62 42 63 41 18 42 17 43 16 44 15 45 14 46]

Как это устроено

0ali                               " Push S := [ 0 ] and read an integer N from STDIN.    ";
    {                      }fI     " For each I in [ 0 ... (N - 1) ]:                     ";
     _W=                           "   X := S[-1].                                        ";
        _I-                        "   Y := X - I                                         ";
            _0<                    "   A := (Y < 0)                                       ";
           _   4$@#)               "   B := (Y ∊ S)                                       ";
                     @I+           "   Z := X + I                                         ";
                    |   @?         "   C := (A || B) ? Z : Y                              ";
                          +        "   S += [C]                                           ";
                              1>`  " Push str(S[1:]).                                     ";

Какие изменения вы сделали?
Сохам Чоудхури

Мой первый подход добавлял отрицательные числа к последовательности, поэтому мне не нужно было явно проверять, если A(i) - i > 0. Тем не менее, я не добавил достаточно чисел для небольших значений n. Теперь я просто делаю именно то, что говорит спецификация.
Деннис

33 против 45. Так близко и еще так далеко. :)
Инго Бюрк

Ого, комментируйте без e#Cjam ... вкусная вишня.
Хром

8

Хаскелл, 74

l=0:0#1
a§v|a<0||a`elem`r v=v|1<2=0-v
a#b=a+(a-bb:l!!b#(b+1)
r=(`take`l)

Пример использования:

λ> r 20
[0,1,3,6,2,7,13,20,12,21,11,22,10,23,9,24,8,25,43,62]

6

Рубин, 71 70 байт

f=->n{a=[0];(n-1).times{|i|a+=[[b=a[-1]-i-1]-a!=[]&&b>0?b:b+2*i+2]};a}

Очень "дословная" реализация определения.


5

Python 2, 78 75 73 69 байт

Престижность xnor и flornquake
Теперь почти на 10 байт короче, чем первоначальный ответ

m=p,=0,
exec"p+=1;k=m[-1]-p;m+=k+2*p*(k*(k>0)in m),;"*input()
print m

Вы можете сократить [k,k+2*p][bool]до k+2*p*(bool).
xnor

@xnor Спасибо, сохранил 3 байта.
Маркуз

Кроме того, k in m or k<0может быть, k*(k>=0)in mтак как, если k<0продукт 0, который находится в m.
xnor

@xnor Блестящий! Еще раз спасибо
Маркуз

Вы можете написать -1вместо p-1. Изменить: Вы также можете сделать mкортеж и написать m=0,и m+=k+2*p*(k*(k>0)in m),.
землетрясение

4

Golfscript (41 45 )

Попробуйте это онлайн здесь :

(,1,\{:~1$=~)-:^1<\.^?)!!@|^\{~)2*+}*+}/

объяснение

Это для оригинального 45-байтового решения, но оно по-прежнему почти такое же:

(,              # push array [0 .. n-1]
[0]\            # push sequence elements as [0] and reverse stack
{               # foreach element in [0 .. n-1] do:
  :m;           # store current element in m and discard
  .m=           # get the previous sequence element
  m)-:^         # subtract the current index from it and store in ^
  0>            # is that number greater than 0?
  \.^?)!        # is that number new to our sequence?
  @&            # logically and both checks
  {^}           # if true, push ^
  {^m)2*+}      # otherwise, add the index twice and push
  if
  +             # add new element to our sequence
}/
`               # make output pretty

Правка №1: Спасибо Деннису за то, что он сбрил 4 байта.


4

постоянный ток , 46 байтов

sn[z+z+d0r:a]sF0[pz-d1>Fd;a0<Fddd:azln!<M]dsMx

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

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

Я действительно горжусь этим - он побеждает все, что не является специальным языком для игры в гольф, и демонстрирует три моих любимых трюка с гольфом:

  • Размер стека, используемый в качестве индексной переменной
  • Рефакторинг «если A, то B иначе C» в «безусловно C, и если A, то D», где C и D объединяются, чтобы сделать B
  • малоиспользуемая функция массива произвольного доступа для решения ограничения уникальности

объяснение

sn             Stores the input in register n
[z+z+0r:a]sF   Defines the macro F, which: 
    z+z+         adds twice the stack size/index variable
    0r:a         resets the "uniqueness" flag to 0 in the array a
               In context, F is the "D" in my description above, 
               changing A(z-1)-z to A(z-1)+z
0              The main loop starts with the previous sequence member on top of 
               the stack and total stack depth equal to the next index. 
               Pushing a zero accomplishes both of these things.
[              Start of the main loop M
  p               Print the previous sequence member, with newline (no pop)
  z-             Calculate A(z-1)-z
  d1>F           If that's nonpositive, (F)ix it to be A(z-1)+z
  d;a            a is my array of flags to see if we've hit this value before
  0<F            If we have, (F)ix it! (nonzero = flag, since ;a is zero by
                 default, and also zero if we just (F)ixed and therefore 
                 don't care about uniqueness right now)
  ddd            Make one copy to keep and two to eat
  :a             Flag this entry as "used" in the uniqueness array a
  zln!<M         If our "index variable" is n or less, repeat!
]dsMx          End of main loop - store it and execute

это дико, я понятия не имел, dc даже существовал
наденьте яркий

3

JavaScript - 81 80 79 70

Благодарность edc65 за помощь в экономии 9 байт

f=n=>{for(a=[x=i=0];++i<n;)a[i]=x+=x>i&a.indexOf(x-i)<0?-i:i;return a}

-9: g = n => {для (a = [x = i = 0]; ++ i <n;) a [i] = x + = x> i & a.indexOf (xi) <0? -I: i ; вернуть a
edc65

@ edc65 Grazie mille :)
Уильям Барбоза

3

JavaScript, ES6, 74 69 символов

Запустите приведенный ниже код в последней веб-консоли Firefox.

G=n=>(i=>{for(r=[t=0];++i<n;)r[i]=t+=i>t|~r.indexOf(t-i)?i:-i})(0)||r

Попробую в гольф это позже.

Пример использования:

G(11) -> 0,1,3,6,2,7,13,20,12,21,11

3

МАТЛАБ, 83 78 байт

Сохранить ниже как f.m(73 байта)

A=0;for i=1:n-1 b=A(i)-i;A(i+1)=b+2*i;if b>0&&~any(A==b) A(i+1)=b;end;end

Запуск из командного окна (5 байт)

n=9;f

Если вышеупомянутое не является законным, тогда это требует 90 байтов.

function A=f(n) 
A=0;for i=1:n-1 b=A(i)-i;A(i+1)=b+2*i;if b>0&&~any(A==b) A(i+1)=b;end;end

3

R: 96 символов

Golfed:

A=function(s,n,m,i){if(m==n){return(s)}else{t=i-m;if(t%in%s||t<0){t=i+m};s=c(s,t);A(s,n,m+1,t)}}

Ungolfed:

A = function(s,n,m,i) {
    if(m==n){return(s)}
    else{
        t=i-m
        if(t%in%s||t<0){t=i+m}
        s=c(s,t)
        A(s,n,m+1,t)
    }
}

Пробный прогон:

> An(0,34,1)
[1]   0   1   3   6   2   7  13  20  12  21  11  22  10  23   9  24   8
[18]  25  43  62  42  63  41  18  42  17  43  16  44  15  45  14  46  79


3

Perl 6 , 62 57 байт

{(0, {$ - @ + @ * 2 * ($ !> @ || $ - @ ∈ @ ) с учетом @ [* -1]} ... *) [^ $ ]}

{(0,{($!=@_[*-1])+@_-@_*2*($!>@_&&$!-@_∉@_)}...*)[^$_]}

-5 байт благодаря Джо Кингу

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


это удивительно ... это буквально похоже на то, как мой кот прошелся по клавиатуре.
Дон Ярко

3

05AB1E , 19 байтов

¾ˆG¯¤N-DŠD0›*åN·*+ˆ

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

объяснение

¾ˆ                    # Initialize the global list with 0
  G                   # for N in [1, input-1] do:
   ¯                  # push the global list
    ¤N-               # subtract N from the last item in the list
       D              # duplicate
        Š             # move the copy down 2 spots on the stack
         D            # duplicate again
          0›          # check if it is positive
            *         # multiply, turning negative results to zero
             å        # is the result already present in the list?
              N·*     # multiply by N*2
                 +    # add to the result
                  ˆ   # add this to the list

Как это работает?
lirtosiast

@lirtosiast: Прошло много времени с тех пор, как я выполнил эту задачу, так что это лучшее объяснение, которое я могу сделать в короткие сроки. Надеюсь, этого достаточно.
Эминья

3

K (ок) , 53 байта

Решение:

{$[y>c:#x;o[x,(r;*|x+c)(r in x)|0>r:*|x-c;y];x]}[,0;]

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

Объяснение:

Рекурсивное решение.

{$[y>c:#x;o[x,(r;*|x+c)(r in x)|0>r:*|x-c;y];x]}[,0;] / the solution
{                                              }[,0;] / lambda with first arg set as list containing 0
 $[      ;                                  ; ]       / if[condition;true;false]
       #x                                             / length of x
     c:                                               / save as c
   y>                                                 / y greater than? (ie have we produced enough results?)
                                             x        / return x if we are done
          o[                             ;y]          / recurse with new x and existing y
                                      x-c             / subtract c from x
                                    *|                / reverse first, aka last
                                  r:                  / save result as r
                                0>                    / 0 greater than?
                               |                      / or
                       (      )                       / do together
                        r in x                        / r in x?
              ( ;     )                               / use result to index into this 2-item list
                   x+c                                / add c to x
                 *|                                   / reverse first, aka last 
               r                                      / result
            x,                                        / append to x

2

Ява, 144

int[]f(int n){int[]a=new int[n];a[0]=0;int i,j,k,m;for(i=0;i<n-1;){k=a[i++]-i;m=0;for(j=0;j<i;)if(k==a[j++])m=1;a[i]=m<1&k>0?k:k+2*i;}return a;}

2

Луа - 141 135 139 135

function s(n)a,b={1},{[0]=0}for i=1,n do k=b[i-1]-i c=k+i+i if(k>0)and(a[k]==nil)then b[i],a[k]=k,1 else b[i],a[c]=c,1 end end return b end

читаемая версия:

function s(n)
a,b={1},{[0]=0}
for i=1,n do 
   k=b[i-1]-i 
   c=k+i+i
   if (k>0) and (a[k]==nil) then 
      b[i],a[k]=k,1 
   else 
      b[i],a[c]=c,1
   end 
end 
return b 
end

Я использую 2 таблицы, первая называется a, и она построена так, что a [i] = 1, если i уже появился в последовательности, в противном случае - nil , тогда как вторая таблица фактически содержит последовательность


Ваша последовательность должна начинаться с 0, хотя
Уильям Барбоза

1
Вы правы, я не очень внимательно посмотрел на вопрос и предположил, что в mathworld он имеет такое же определение (начиная с 1), я думаю, что это больше не будет стоить персонажа, я проверю и исправлю его позже, Пишу с телефона сейчас!

2

Python, 73

def f(x,t=0):
 if x:t=f(x-1);t+=2*x*(t*(t>0)in map(f,range(x)))
 return t

Редактировать 1: Благодаря подсказкам @ xnor на другой ответ Python! (Я только что понял, что оба выглядят очень похоже.)

Редактировать 2: Еще раз спасибо, @xnor.


Это дает бесконечный цикл. Вам нужен какой-то поток управления, чтобы f(x)он не всегда сразу вызывал f(x-1).
xnor

@xnor исправил код.
Сохам Чоудхури

1
Кажется, это возвращает n-й член, а не первый n член.
Деннис

Некоторые незначительные сохранения: t=0может быть необязательным параметром для fи t=t+может быть t+=.
xnor

2

Groovy: 122 118 111 символов

Golfed:

m=args[0] as int
a=[0]
(1..m-1).each{n->b=a[n-1];x=b-n;(x>0&!(x in a))?a[n]=x:(a[n]=b+n)}
a.each{print "$it "}

Ungolfed:

m = args[0] as int
a = [0]
(1..m-1).each { n->
    b = a[n-1]
    x = b-n
    ( x>0 & !(x in a) ) ? a[n] = x : (a[n] = b+n) 
}
a.each{print "$it "}

Пробный прогон:

bash$ groovy Rec.groovy 14
0 1 3 6 2 7 13 20 12 21 11 22 10 23

2

Clojure: 174 символа

Golfed:

(defn f[m a](let[n(count a)b(last a)x(- b n)y(if(and(> x 0)(not(.contains a x)))x(+ b n))](if(= m n)a(f m(conj a y)))))(println(f(read-string(first *command-line-args*))[0]))

Ungolfed:

(defn f[m a]
  (let [n (count a) 
        b (last a) 
        x (- b n) 
        y (if (and (> x 0) (not (.contains a x))) x (+ b n)) ]
    (if (= m n) a (f m (conj a y))) ) )

(println (f (read-string (first *command-line-args*)) [0]) )

Образец прогона:

bash$ java -jar clojure-1.6.0.jar rec.clj 14 
[0 1 3 6 2 7 13 20 12 21 11 22 10 23]

1
Я предлагаю вам не читать из STDIN, а вместо этого просто взять целочисленный аргумент функции :) Кроме того, вы не получаете никаких преимуществ от определения yв letформе, вы можете использовать выражение непосредственно там, где необходимо значение.
NikoNyrh

2

Mathcad, 54 "байта"

введите описание изображения здесь


С точки зрения пользователя, Mathcad фактически представляет собой двухмерную доску с выражениями, оцениваемыми слева направо и сверху вниз. Mathcad не поддерживает традиционный ввод текста, но вместо этого использует комбинацию текста и специальных клавиш / панели инструментов / пунктов меню для вставки выражения, текста, графика или компонента. Например, введите «:» для ввода оператора определения (отображается на экране как «: =») или «ctl-shft- #» для ввода оператора цикла for (включая заполнители для переменной итерации, значений итерации и одного тела выражение). На изображении выше вы видите именно то, что отображается в пользовательском интерфейсе и как «напечатано» в.

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


Это все хорошо , но то , что есть реальные нажатия клавиш?
Джо Кинг


2

Stax , 19 байт

É╖C8½ΔL▄░▬L+≡ΩSa⌂¼╧

Запустите и отладьте его

Распакованный, размазанный и прокомментированный, это выглядит так. Пока он хранит последовательность в стеке и запоминает A(n - 1)в регистре X. Индекс итерации используется для n. В первый раз это 0, но в этой итерации он генерирует 0 без особых случаев, поэтому нет необходимости настраивать индекс off-by-1.

0X      push 0 to main stack and store it in X register, which will store A(n - 1)
z       push an empty array that will be used to store the sequence
,D      pop input from input stack, execute the rest of the program that many times
  xi-Y  push (x-register - iteration-index) and store it in the Y register
        this is (A(n - 1) - n)
  0>    test if (A(n - 1) - n) is greater than 0 (a)
  ny#   count number of times (A(n - 1) - n) occurs in the sequence so far (b)
  >     test if (a) > (b)
    y   (A(n - 1) - n)
    xi+ A(n - 1) + n
  ?     if/else;  choose between the two values based on the condition
  X     store the result in the X register
  Q     print without popping
  +     append to sequence array

Запустите и отладьте этот


интересный. как это работает?
Дон Яркий

1
@donbright: Добавлены некоторые аннотации и пояснения.
рекурсивный

2

Pyth, 31 байт

VQ=+Y?Y?|>NeYhxY-eYN+eYN-eYNZ)Y

Добро пожаловать на сайт!
Пшеничный волшебник

Добро пожаловать. Ответы, содержащие только код, не приветствуются, поскольку они автоматически помечаются как некачественные. Добавьте объяснение и рассмотрите возможность добавления ссылки на онлайн-переводчика, например: tio.run/##K6gsyfj/PyzQVjvSPtK@xs4vNTKjIlI3NdJPG4hBdJRm5P//hoYA
mbomb007

2

Pyth , 24 байта

tu+G-eG_W|g0J-eGH}JGHQ]0

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

tu+G-eG_W|g0J-eGH}JGHQ]0   Implicit: Q=eval(input())
 u                   Q     Reduce [0-Q)...
                      ]0   ... with initial value G=[0], next value as H:
              eG             Last value of G (sequence so far)
             -  H            Take H from the above
            J                Store in J
          g0J                0 >= J
                 }JG         Is J in G?
         |                   Logical OR of two previous results
       _W           H        If the above is true, negate H, otherwise leave as positive
    -eG                      Subtract the above from last value in G
  +G                         Append the above to G
                           The result of the reduction is the sequence with an extra leading 0
t                          Remove a leading 0, implicit print

1

Powershell (103)

$n=Read-Host;$a=@(0);$n-=1;1..$n|%{$x=$a[-1]-$_;if($x-gt0-and!($a-like$x)){$a+=$x}else{$a+=$x+2*$_}};$a

Еще одна «дословная» реализация здесь. Удивительно читабельно для PowerShell тоже.

Последовательность хранится в массиве $ a и распечатывается по одному члену на строку.

Для $ n = 20, если мы запустим оператор, $a-join","мы получим

0,1,3,6,2,7,13,20,12,21,11,22,10,23,9,24,8,25,43,62

1

C #: 140 символов

int i,w,t,y;int[]F(int n){var r=new int[n--];for(;i<n;y=0){w=r[i++]-i;for(t=0;y<i&&t<1;)t=w==r[y++]?1:0;r[i]=w>0&&t<1?w:r[i-1]+i;}return r;}

1

C ++: 180 символов (158 без операторов cin и cout)

int a[5000000][2]={0},i,k,l;a[0][0]=0;a[0][1]=1;cin>>k;for(i=1;i<=k;i++){l=a[i-1][0];if(l-i>0&&a[l-i][1]!=1){ a[i][0]=l-i;a[l-i][1]=1;}else{ a[i][0]=l+i;a[l+i][1]=1;}cout<<a[i][0]<<endl;

Добро пожаловать в раздел «Программирование головоломок и обмен стеками Code Golf»! Пожалуйста, отредактируйте количество символов / байтов вашего решения в заголовке, как показано в других ответах здесь. Также, пожалуйста, используйте ваш код (например, удалите пробелы, чтобы уменьшить количество символов), насколько это возможно. Благодарность!
дверная ручка

Конечно, я сделаю это.
Абхай Джайн

1

Mathematica - 81 байт

Fold[#~Append~(#[[-1]]+If[#[[-1]]>#2&&FreeQ[#,#[[-1]]-#2],-#2,#2])&,{0},Range@#]&

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

Fold[#~Append~(#[[-1]]+If[#[[-1]]>#2&&FreeQ[#,#[[-1]]-#2],-#2,#2])&,{0},Range@#]&[30]
{0,1,3,6,2,7,13,20,12,21,11,22,10,23,9,24,8,25,43,62,42,63,41,18,42,17,43,16,44,15,45}

1

PHP , 89 байт

$f=function($n){for(;$i<$n;$s[$r[$i++]=$p=$m]=1)if($s[$m=$p-$i]|0>$m)$m=$p+$i;return$r;};

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

Ungolfed:

$f = function ($n) {
    for (; $i < $n; $s[$r[$i++] = $p = $m] = 1) {
        if ($s[$m = $p - $i] | 0 > $m) {
            $m = $p + $i;
        }
    }

    return $r;
};
  • $r за мой результат
  • $s для отслеживания увиденного
  • $p предыдущее значение
  • $m м экст значение

1

Общий LISP (139 байт)

(defun r(n)(do*(s(i 0(1+ i))(a 0(car s))(b 0(- a i)))((> i n)(nreverse s))(push(cond((= 0 i)0)((and(> b 0)(not(find b s)))b)(t(+ a i)))s)))

Ungolfed:

(defun recaman (n)
  (do*
   (series               ; starts as empty list
    (i 0 (1+ i))         ; index variable
    (last 0 (car s))     ; last number in the series
    (low 0 (- last i)))

   ((> i n)              ; exit condition
    (nreverse series))   ; return value

    (push                ; loop body
     (cond
       ((= 0 i) 0)       ; first pass
       ((and
         (> low 0) (not (find low s)))
        low)
       (t (+ last i)))
     series)))
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.