Буфер защиты от переполнения


23

Задний план

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

обзор

У вас есть буфер только для вставки, размер которого увеличивается по мере добавления к нему элементов. Буфер индексируется нулем, а также индексируется по модулю его текущего размера. Специальное правило для этого вызова таково:

  • Чтобы вставить элемент с индексом я означает вычислить J , j = i % buffer.length()и вставить новый элемент после элемента в списке.

Единственный особый случай - если буфер пуст, так как арифметика по модулю ноль не работает. Таким образом, если буфер в настоящее время пуст, новый элемент будет иметь индекс 0 .

Если в буфере есть только один элемент, то вы всегда вставляете после 0-го элемента. Это всего лишь один пример общего случая.

Если буфер содержит 6 элементов: [4, 9, 14, 8, 5, 2]и вам предлагается вставить новый элемент 10с индексом 15 , вы обнаружите это 15 % 6 == 3, а затем вставите новый 10после 8индекса с 3, что даст результирующий буфер [4, 9, 14, 8, 10, 5, 2]

проблема

Напишите функцию или программу, которая получает упорядоченный список положительных целых чисел и индексов положительных целых чисел, в которые их можно вставить.

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

Выведите упорядоченный список целых чисел, которые находятся в буфере после всех указанных вставок.

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

Рекомендации по вводу

Вы можете взять входные списки, как считаете нужным. Примеры:

  • Список пар: [ [1,1], [2,4], [3,9], [4,16], [5,25]...]
  • Список предметов и индексный список: [1, 2, 3, 4, 5...], [1, 4, 9, 16, 25]
  • Уплощенная: [1, 1, 2, 4, 3, 9, 4, 16, 5, 25 ...]
  • и т.п.

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

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

Случай квадратов сверху:

[(1, 1), (2, 4), (3, 9), (4, 16), (5, 25), (6, 36), (7, 49), (8, 64)] -> [1, 2, 8, 7, 6, 5, 4, 3]

Я генерировал это случайно:

[(11, 9), (13, 14)] -> [11, 13]
[(1, 18), (11, 7), (3, 35), (16, 22)] -> [1, 11, 16, 3]
[(3, 16), (16, 37), (0, 28), (18, 24)] -> [3, 18, 0, 16]
[(7, 26), (8, 20), (11, 39), (1, 23), (17, 27)] -> [7, 8, 11, 1, 17]
[(15, 35), (17, 7), (16, 15), (1, 13), (2, 6), (11, 34)] -> [15, 17, 1, 2, 16, 11]
[(2, 13), (1, 20), (16, 25), (8, 21), (5, 2), (16, 37), (3, 0)] -> [2, 3, 8, 1, 16, 5, 16]
[(6, 20), (15, 15), (12, 26), (10, 27), (17, 13), (7, 18), (4, 16)] -> [6, 10, 17, 12, 7, 4, 15]
[(18, 9), (5, 34), (15, 4), (12, 29), (2, 5), (7, 0), (7, 10), (16, 38)] -> [18, 7, 15, 2, 16, 5, 7, 12]
[(0, 12), (12, 0), (4, 16), (15, 12), (6, 28), (8, 10), (11, 24), (0, 25)] -> [0, 11, 8, 6, 15, 0, 4, 12]
[(6, 12), (14, 13), (10, 33), (11, 35), (1, 3), (0, 28), (15, 27), (8, 10), (1, 2)] -> [6, 14, 10, 1, 11, 8, 15, 0, 1]
[(2, 29), (19, 30), (18, 17), (13, 3), (0, 21), (19, 19), (11, 13), (12, 31), (3, 25)] -> [2, 13, 3, 11, 0, 12, 19, 18, 19]

Реализация ссылок на Python3

def f(inputs):
    # `inputs` is a list of pairs
    buff = []
    for item, index in inputs:
        if len(buff) == 0:
            buff.insert(0, item)
        else:
            insert_after = index % len(buff)
            buff.insert(insert_after+1, item)
    return buff

Можно ли вводить в обратном порядке?
FlipTack

Да, я думаю, что ввод может быть более или менее гибким, как вы хотите.
турбулентность

Ответы:


4

MATL , 24 22 байта

"N?@2)yn\Q:&)@1)wv}@1)

Ввод - это матрица (с ;разделителем строк), содержащая значения в первой строке и индексы во второй.

Выходные данные - это массив столбцов, отображаемый в виде чисел, разделенных символами новой строки.

Попробуйте онлайн! Или проверьте все тестовые случаи , каждый результат отображается в одной строке.

объяснение

"          % Input matrix (implicit). For each column 
  N        %   Number of elements in the stack
  ?        %   If nonzero (true for all iterations but the first)
    @2)    %     Push second element of current column: new index
    yn     %     Duplicate current buffer; push its number of elements
    \      %     Modulo
    Q      %     Add 1
    :&)    %     Split buffer at that point. This gives two pieces, one
           %     of which may be empty
    @1)    %     Push first element of current column: new value
    wv     %     Swap; concatenate all stack. This places the new value
           %     between the two pieces of the buffer
  }        %   Else (this is executed only in the first iteration)
    @1)    %     Push first element of current column: new value
           %   End (implicit)
           % End (implicit)
           % Display (implicit)

8

Perl, 37 байт

35 байтов кода + 2 байта для -lpфлагов.

splice@F,1+<>%(@F||1),0,$_}{$_="@F"

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

Реализация довольно проста, spliceвставка в массив @Fпо индексу 1+<>%(@F||1)(обратите внимание, что @F||1обрабатывает случай, когда массив пуст).

Несколько слов о (казалось бы) непревзойденных фигурных скобках }{(потому что у меня был комментарий по этому поводу, и я думаю, что это довольно странно для людей, которые не знают Perl), и это довольно распространенная уловка в Perl-гольфах:
-pфлаг окружает код с (примерно) while(<>){ CODE } continue { print }, ( continueвыполняется после каждой итерации). Так что с теми , кто не имеет себе равных }{ , я изменяю свой код на while(<>) { CODE}{ } continue { print }. Таким образом, он создает пустой блок сразу после моего кода (но это не проблема), и continueвыполняется только один раз, после while(то есть, когда все входные данные были прочитаны).


3
Это }{сводит меня с ума ...
ETHproductions

@ETHproductions Я привык к этому, но я люблю показывать это другим людям, они всегда думают, что что-то не так! :) (Недостатком является то, что это портит мой отступ Emacs ..)
Dada

1
Это }{напоминает мне об этой иллюзии
Луис Мендо

Да, я сделал. :-)
Деннис

5

ES6 (Javascript), 58,57,53, 50 байтов

Golfed

a=>a.map((e,i)=>b.splice(1+e[1]%i,0,e[0]),b=[])&&b

Принимает массив пар индекс-значение в качестве входных данных.

редактирует

  • Используйте &&для возврата значения, -1 байт
  • Удалено |0(поскольку сплайс, видимо, может хорошо обрабатывать NaN), -2 байта
  • Сделал b=[]второй «аргумент» для map () , -2 байта (Thx @ETHproductions!)
  • Заменена длина b.plext () index (i), -3 байта (Thx @Patrick Roberts!)

Тест

F=a=>a.map((e,i)=>b.splice(1+e[1]%i,0,e[0]),b=[])&&b

F([[11, 9], [13, 14]])
[ 11, 13 ]

F([[2, 29], [19, 30], [18, 17], [13, 3], [0, 21], [19, 19], [11, 13], [12, 31], [3, 25]])
[ 2, 13, 3, 11, 0, 12, 19, 18, 19 ]

1
Хорошо, я должен был попробовать нерекурсивные подходы. Я думаю, что вы можете сделатьa=>a.map(e=>...,b=[])&&b
ETHproductions

2
Вы можете вычесть 3 байта, изменяя e=>к (e,i)=>и используя iвместоb.length
Патрик Робертс

@PatrickRoberts Это хорошая идея! Спасибо !
Цеппелин

5

Haskell , 70 69 байтов

b!(x,i)|b==[]=[x]|j<-1+i`mod`length b=take j b++x:drop j b
foldl(!)[]

Попробуйте онлайн! Использование: foldl(!)[] [(1,5),(2,4),(3,7)]. Сохранено один байт благодаря @nimi!

Объяснение:

b!(x,i)                         -- b!(x,i) inserts x into list b at position i+1
 | b==[] = [x]                  -- if b is empty return the list with element x
 | j <- 1 + i `mod` length b    -- otherwise compute the overflow-save insertion index j
     = take j b ++ x : drop j b -- and return the first j elements of b + x + the rest of b
foldl(!)[]                      -- given a list [(1,2),(3,5),...], insert each element with function ! into the initially empty buffer

Решение без вычисления модуля: (90 байт)

f h t(x,-1)=h++x:t
f h[]p=f[]h p
f h(c:t)(x,i)=f(h++[c])t(x,i-1)
g((x,_):r)=foldl(f[])[x]r

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


j<-1+i`mod`length bсохраняет байт.
Ними


4

Python 2 , 62 60 байт

Принимает ввод в виде списка пар, печатает результат. Редактировать: Outgolfed Деннис

b=[]
for x,y in input():b.insert(1+y%(len(b)or 1),x)
print b

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

Это довольно просто - переберите ввод, вставьте элементы в правильное место, а затем напечатайте результат. Решение о том, в какой индекс вставить, делается 1+y%(len(b)or 1). Это стандартный способ модульной индексации с or 1обработкой края пустого списка.


3

JavaScript (ES6), 60 байт

f=([[q,r],...a],z=[])=>z.splice(r%z.length+1,0,q)+a?f(a,z):z

Тестовый фрагмент


2

V , 38 40 35 байт

Этот ответ отклоняет определение списка и обычно не является языком, который вы бы использовали для манипулирования списком, но я хотел использовать тот, [count]/{regex}который недавно добавил в V. Ввод принимается как [index] [num] [index] [num] ...и возвращается как [num] [num] [num].

Í ¨ä«© ½/¯ÜdÜ+òhea ±^
Hdd2xdG@"
X

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

Hexdump для 2 скрытых персонажей:

00000000: cd20 a8e4 aba9 20bd 2faf dc64 dc2b f268  . .... ./..d.+.h
00000010: 6561 20b1 161b 5e0a 4864 6432 7864 4740  ea ...^.Hdd2xdG@
00000020: 220a 58                                  ".X

объяснение

Код для dG@"форматирования всех \d+ \d+пар, так что список 1 2 3 4 5 6 будет выглядеть как

a 2^[^3/\d\+
hea 4^[^5/\d\+
hea 6^[^

и затем dG@"выполняет все это как V-код, как показано ниже:

a 2^[                 | insert " 2" and return to command mode
     ^                | LOOP: go to the first number
      3/\d\+          | find the 3rd number (0 indexed)
h                     | move one character left
 e                    | go to the end of the next word
  a 4^[               | append " 4" and return to command mode
       ^5/\d\+        | basically the same as LOOP on, just with different numbers


Ах, спасибо, ты этого не знал. Я
приведу

2

PHP, 72 92 байта

for($b=[];++$i<$argc;)array_splice($b,$b?$argv[$i]%count($b)+1:0,0,$argv[++$i]);print_r($b);

принимает ввод, сплющенный из аргументов командной строки. Беги с -nr.


Я вполне уверен, что этот ответ недействителен: я получил Fatal error: Uncaught DivisionByZeroError: Modulo by zero, исправил это, затем попытался 1 1 1 2 1 3и получил [1=>null]как вывод вместо[1,3,2]
user59178

Это все еще переписывает, j+1а не вставляет после j, нет? 18 1 7 11 35 3 22 16=> [1,11,16]а не[1,11,16,3]
user59178

@ user59178: О, я пропустил insertключевое слово. Благодарность; исправлено.
Тит

2

Java 7, 125 124 байта

import java.util.*;List z(int[]a){List r=new Stack();for(int i=0,q=a.length/2;i<q;)r.add(i<1?0:a[i+q]%i+1,a[i++]);return r;}

Принимает плоский список значений, за которым следуют индексы. Для теста квадраты вход будетnew int[] {1, 2, 3, 4, 5, 6, 7, 8, 1, 4, 9, 16, 25, 36, 49, 64}

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


1

Mathematica, 62 байта

Fold[Insert[#,#2[[1]],Mod[Last@#2,Tr[1^#]]+2/.0/0->-1]&,{},#]&

Чистая функция с первым аргументом #должна быть списком пар. Начиная с пустого списка {}, оставляем Foldсписок ввода #со следующей функцией:

Insert[                                            Insert
       #,                                          into the first argument
         #2[[1]],                                  the first element of the second argument
                 Mod[                              at the position given by the modulus of
                     Last@#2,                      the second element of the second argument
                             Tr[1^#]               with respect to the length of the first argument
                                    ]+2            plus 2 (plus 1 to account for 1-indexing, plus 1 because we are inserting after that position)
                                       /.          then replace
                                         0/0       Indeterminate
                                            ->     with
                                              -1   negative 1
                                                ]& End of function

1

Perl 6 , 51 байт

{my @a;.map:{splice @a,(@a??$^b%@a+1!!0),0,$^a};@a}

Принимает сплющенный ввод.


1

Clojure, 87 байт

#(reduce(fn[r[v i]](let[[b e](split-at(+(mod i(max(count r)1))1)r)](concat b[v]e)))[]%)
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.