Найдите пропущенное число в неограниченной строке

19

Задача состоит в том, чтобы определить недостающее число в строке из неограниченных целых чисел.

Вам дана строка цифр (допустимый ввод будет соответствовать регулярному выражению ^[1-9][0-9]+$). Строка представляет собой последовательность целых чисел. Например, 1234567891011. Все числа в последовательности находятся в диапазоне от 1и 2147483647включительно.

Последовательность представляет собой последовательность чисел, где каждое число на единицу больше, чем его предшественник. Однако эта последовательность может содержать один и только один пропущенный номер из последовательности. Возможно, что данная строка также не содержит пропущенных чисел в последовательности. Строка всегда будет содержать как минимум два числа из последовательности.

Код должен выводить или возвращать пропущенное значение или 0(это 0- не ложное значение) в случае, если пропущенные значения не были найдены.

Ниже приведены допустимые входные данные и их вывод / возврат:

input                         output    actual sequence (for refrence)
123467                        5         1 2 3 4 _ 6 7
911                           10        9 __ 11
123125126                     124       123 ___ 125 126
8632456863245786324598632460  8632458   8632456 8632457 _______ 8632459 8632460  
123                           0         1 2 3
8632456863245786324588632459  0         8632456 8632457 8632458 8632459  

Хотя все это описывается как «строка» в качестве входных данных, если язык способен обрабатывать произвольно большие числа ( dcи mathematica, я смотрю на вас двоих), входные данные могут быть произвольно большим числом вместо строки, если это делает код проще.

Для справки, это было вдохновлено вопросом Programmers.SE: найти пропущенное число в последовательности в строке

Сообщество
источник
4
Вы уверены, что это однозначно?
Мартин Эндер
@ MartinBüttner Я немного подумал об этом и не смог придумать ситуацию, когда последовательность, увеличивающаяся на 1 (это может быть проблемой), имеет неоднозначную ситуацию.
Есть ли в OEIS запись для списка целых чисел, представляющих собой сцепленную последовательность, в которой отсутствует ровно один элемент?
mbomb007
@ mbomb007 Я так не думаю, потому что существует бесконечно много разных списков. И это всего лишь одна большая оле строка. Не уверен, как бы вы это определили. В этом отношении интересным вопросом CS будет «какой язык принимает все эти строки». Это, конечно, не регулярно. Я сомневаюсь в его CF.
1
Я сделал последовательность предметом вызова: codegolf.stackexchange.com/q/73513/34718
mbomb007

Ответы:

5

Haskell, 115 112 байтов

g b|a<-[b!!0..last b]=last$0:[c|c<-a,b==filter(/=c)a]
maximum.map(g.map read.words.concat).mapM(\c->[[c],c:" "])

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

объяснение

Это решение грубой силы: разбить строку на слова всеми возможными способами, разобрать слова в целые числа, посмотреть, отсутствует ли диапазон с одним пропущенным элементом (вернуть этот элемент и т. Д. 0), И взять максимум для всех разбиений. Проверка диапазона с отсутствующим элементом выполняется в вспомогательной функции g, которая берет список bи возвращает единственный элемент в диапазоне [head of b..last of b], которого нет bили 0если он не существует.

g b|                         -- Define g b
    a<-[b!!0..last b]=       -- (with a as the range [head of b..last of b]) as:
    last$0:[...]             --  the last element of this list, or 0 if it's empty:
            c|c<-a,          --   those elements c of a for which
            b==filter(/=c)a  --   removing c from a results in b.
mapM(\c->[[c],c:" "])        -- Main function: Replace each char c in input with "c" or "c "
map(...)                     -- For each resulting list of strings:
  g.map read.words.concat    --  concatenate, split at spaces, parse to list of ints, apply g
maximum                      -- Maximum of results (the missing element, if exists)
Zgarb
источник
2

JavaScript (ES6), 117 байт

s=>eval(`for(i=l=0;s[i];)for(n=s.slice(x=i=m=0,++l);s[i]&&!x|!m;x=s.slice(x?i:i+=(n+"").length).search(++n))m=x?n:m`)

объяснение

Довольно эффективный подход. Завершает мгновенно для всех тестовых случаев.

Получает каждую подстроку от начала входной строки как число nи инициализирует отсутствующее число mкак 0. Затем он многократно удаляет nиз начала строки, увеличивает nи ищет строку для него. Если index of n != 0это проверяет m. Если m == 0установить m = nи продолжить, если нет, пропущено несколько номеров, поэтому прекратите проверку из этой подстроки. Этот процесс продолжается, пока вся строка не будет удалена.

var solution =

s=>
  eval(`                     // use eval to use for loops without writing {} or return
    for(
      i=                     // i = index of next substring the check
      l=0;                   // l = length of initial substring n
      s[i];                  // if it completed successfully i would equal s.length
    )
      for(
        n=s.slice(           // n = current number to search for, initialise to subtring l
          x=                 // x = index of n relative to the end of the previous n
          i=                 // set i to the beginning of the string
          m=0,               // m = missing number, initialise to 0
          ++l                // increment initial substring length
        );
        s[i]&&               // stop if we have successfully reached the end of the string
        !x|!m;               // stop if there are multiple missing numbers
        x=                   // get index of ++n
          s.slice(           // search a substring that starts from the end of the previous
                             //     number so that we avoid matching numbers before here
            x?i:             // if the previous n was missing, don't increment i
            i+=(n+"").length // move i to the end of the previous number
          )
          .search(++n)       // increment n and search the substring for it's index
      )
        m=x?n:m              // if the previous number was missing, set m to it
  `)                         // implicit: return m
<input type="text" id="input" value="8632456863245786324598632460" />
<button onclick="result.textContent=solution(input.value)">Go</button>
<pre id="result"></pre>

user81655
источник
2

JavaScript (ES6) 114

s=>eval("for(d=0,n=-9,z=s;z=z.slice((n+'').length);z.search(++n)?z.search(++n)?n=(z=s).slice(x=0,++d):x=n-1:0);x")  

Менее гольф и объяснил

f=s=>{
  d = 0  // initial digit number, will be increased to 1 at first loop 
  n = -9 // initial value, can not be found
  z = s  // initializa z to the whole input string
  // at each iteration, remove the first chars of z that are 'n' 
  // 'd' instead of 'length' would be shorter, but the length can change passing from 9 to 10 
  for(; z=z.slice((n+'').length); ) 
  {
    ++n; // n is the next number expected in sequence
    if (z.search(n) != 0)
    {
      // number not found at position 0
      // this could be the hole
      // try to find the next number
      ++n;
      if (z.search(n) != 0)
      {
        // nope, this is not the correct sequence, start again
        z = s; // start to look at the whole string again
        x = 0; // maybe I had a candidate result in xm but now must forget it
        ++d;   // try a sequence starting with a number with 1 more digit
        n = z.slice(0,d) // first number of sequence
      }
      else
      {
        // I found a hole, store a result in x but check the rest of the string
        x = n-1
      }
    }
  }      
  return x // if no hole found x is 0
}

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

F=s=>eval("for(d=0,n=-9,z=s;z=z.slice((n+'').length);z.search(++n)?z.search(++n)?n=(z=s).slice(x=0,++d):x=n-1:0);x")

console.log=x=>O.textContent+=x+'\n'

elab=x=>console.log(x+' -> '+F(x))

function test(){ elab(I.value) }

;['123467','911','123125126','8632456863245786324598632460',
  '123','124125127','8632456863245786324588632459']
.forEach(t=>elab(t))
<input id=I><button  onclick='test()'>Try your sequence</button>
<pre id=O></pre>

edc65
источник
2

C 183 168 166 163 байта

n,l,c,d,b[9];main(s,v,p)char**v,*p;{for(;s>1;)for(d=s=0,n=atoi(strncpy(b,p=v[1],++l)),p+=l;*p&&s<2;)p+=memcmp(p,b,c=sprintf(b,"%d",++n))?d=n,s++:c;printf("%d",d);}

Ungolfed

n,l,c,d,b[9];

main(s,v,p)char**v,*p;
{
    /* Start at length 1, counting upwards, while we haven't
       found a proper number of missing numbers (0 or 1) */
    for(;s>1;)
        /* Start at the beginning of the string, convert the
           first l chars to an integer... */
        for(d=s=0,n=atoi(strncpy(b,p=v[1],++l)),p+=l;*p&&s<2;)
            /* If the next number is missing, then skip, otherwise
               move forward in the string.... */
            p+=memcmp(p,b,c=sprintf(b,"%d",++n))?d=n,s++:c;

    printf("%d",d); /* print the missing number */
}
Коул Камерон
источник
2
Как это работает для входов, например, 891112где числа имеют разную длину?
Згарб
@ Zgarb Работает просто отлично. sprintfВызов возвращает длину недостающего числа, независимо от того , это больше , чем предыдущий или нет.
Коул Кэмерон
Хорошо спасибо! Есть +1.
Згарб