Aplique uma onda a uma matriz

24

Sua tarefa hoje é aplicar uma onda a uma matriz de números. Uma onda se parece com isso: [1, 0, -1, 0, 1, 0, -1, 0, 1...]aplicá-la a uma determinada matriz significa somar os primeiros elementos, os segundos elementos etc.

Mais precisamente:

Seu programa ou função receberá uma matriz de números inteiros. Ele deve imprimir ou retornar uma matriz de tamanho igual, 1adicionada ao 1º, 5º, 9º, etc. elemento da matriz original, -1adicionada ao 3º, 7º, 11º, etc. elemento da matriz original e o restante dos elementos deve ser deixado intocado.

A matriz de entrada é garantida para ter pelo menos um elemento.

Casos de teste:

Input                               | Output
[0]                                 | [1]
[-1]                                | [0]
[-4, 3, 0, 1, 7, 9, 8, -2, 11, -88] | [-3, 3, -1, 1, 8, 9, 7, -2, 12, -88]
[0, 0, 0, 0, 0]                     | [1 ,0 ,-1 ,0 ,1]
[1, 1]                              | [2, 1]

Este é o , o código mais curto vence!

Pavel
fonte
Um pouco unexepectedly, muitas soluções estão usando magia número imaginário ...
Pavel
2
Faz muito sentido porque os números imaginários seriam úteis, este é um problema de onda e os números imaginários têm um histórico bem documentado das propriedades polares. Os números imaginários podem ser uma maneira bastante eficiente de calcular senos e cossenos, especialmente para esses tipos de rotações inteiras de um quarto. Math is cool ...
Assistente de trigo
3
@WheatWizard É uma proporção bastante grande, pois a maioria dos idiomas não suporta números imaginários.
Pavel

Respostas:

8

Geléia , 5 bytes

Jı*Ċ+

Experimente online!

Como funciona

Jı*Ċ+  Main link. Argument: A (array)

J      Indices; yield [1, ..., len(A)].
 ı*    Elevate the imaginary unit to the power 1, ..., len(A), yielding
       [0+1i, -1+0i, 0-1i, 1+0i, ...].
   Ċ   Take the imaginary part of each result.
    +  Add the results to the corresponding elements of A.
Dennis
fonte
O mesmo que o Leaky Nun obteve: chat.stackexchange.com/transcript/message/38868472#38868472
Pavel
11
Alguma explicação?
Pureferret
11
@Pureferret a parte imaginária das potências sucessivas de número imaginário i são adicionados a cada elemento
Coeur
@ Cœur é isso 1, 2, 3 ...ou 1, 0, -1, 0 ...?
Pureferret
11
@Pureferret a mesma explicação como a resposta em MATL ou Math.JS ou Mathematica ou R ou ...
Cœur
14

LOGO , 18 bytes

[map[?+sin 90*#]?]

Não há "Experimente online!" link porque todos os intérpretes on-line do LOGO não oferecem suporte à lista de modelos.

Essa é uma lista de modelos (equivalente à função lambda em outros idiomas).

Uso:

pr invoke [map[?+sin 90*#]?] [-4 3 0 1 7 9 8 -2 11 -88]

( invokechama a função, primprime o resultado)

impressões [-3 3 -1 1 8 9 7 -2 12 -88].

Explicação (já bastante compreensível):

 map[?+sin 90*#]?       map a function over all the items of the input
              #         the 1-based index of the element in the input
       sin 90*#         equal to the required wave
     ?                  looping variable
     ?+sin 90*#         add the wave to the input
user202729
fonte
Hah, eu sabia que alguém teria uma resposta senoidal.
ETHproductions
2
@ETHproductions a primeira resposta, no Mathematica, baseou-se no Sine até que ele fosse jogado. A segunda resposta, em R, ainda é usando seno.
Pavel
11
@ Phoenix Estou chocado por não ter notado ...
ETHproductions
@ETHproductions e .... Sine também recebeu uma resposta fora de série. Eu acho que está fazendo a mesma coisa que a resposta do Mathematica.
Pavel
13

Haskell , 26 bytes

zipWith(+)$cycle[1,0,-1,0]

Experimente online! (executa todos os casos de teste)

Explicação:

zipWith(+)$cycle[1,0,-1,0]  -- anonymous tacit function
zipWith(+)                  -- pairwise addition between input list
          $cycle[1,0,-1,0]  -- and an infinitely-cycling "wave" list
Mego
fonte
9

JavaScript (ES6), 28 bytes

a=>a.map((x,i)=>x-(i%4-1)%2)

O cálculo é assim:

i%4  -1  %2
0    -1  -1
1     0   0
2     1   1
3     2   0

O último bit aproveitando o fato de que em JS, um número negativo quando modulado retém seu sinal negativo (ou seja -5 % 3 -> -2, em vez de 1como seria em Python).

ETHproductions
fonte
9

Mathematica, 26 23 22 bytes

Im[I^Range@Tr[1^#]]+#&

Experimente online!(Matemática)

Nota: O link TIO é para a versão de 23 bytes, a versão de 22 bytes não é compatível com matemática.

user202729
fonte
Existe uma solução Mathematica de 19 bytes abaixo (com 4 bytes de inicialização)
user202729
8

MATL , 11 8 bytes

Jyn:^Yj+

Experimente no MATL Online!

Explicação

J     % Push 1j (imaginary unit)
      % STACK; 1j
y     % Implicit input. Duplicate from below
      % STACK: [-4 3 0 1 7 9 8 -2 11 -88], 1j, [-4 3 0 1 7 9 8 -2 11 -88]
n     % Number of elements
      % STACK: [-4 3 0 1 7 9 8 -2 11 -88], 1j, 10
:     % Range
      % STACK: [-4 3 0 1 7 9 8 -2 11 -88], 1j, [1 2 3 4 5 6 7 8 9 10]
^     % Power, element-wise
      % STACK: [-4 3 0 1 7 9 8 -2 11 -88], [1j -1 -1j 1 1j -1 -1j 1 1j -1]
Yj    % Imaginary part
      % STACK: [-4 3 0 1 7 9 8 -2 11 -88], [1 0 -1 0 1 0 -1 0 1 0]
+     % Add, element-wise. Implicit display
      % STACK: [-3 3 -1 1 8 9 7 -2 12 -88]
Luis Mendo
fonte
Hum, você esqueceu de adicionar +a explicação #
caird coinheringaahing
@cairdcoinheringaahing Obrigado, editado
Luis Mendo
3

Gelatina , 16 bytes

-1Jm2$$¦+2Jm4$$¦

Experimente online!

heh, tenho certeza que isso é muito longo

Editar

Eu sei que uma solução de 5 bytes é possível, mas meu wifi parece estar começando a me impedir, então jogarei amanhã. Se alguém postar a pequena solução Jelly antes que eu possa jogar isso, tudo bem para mim; Vou manter isso aqui para referência sobre o quão ruim eu sou em Jelly, outra maneira de fazê-lo. Quero dizer, eu poderia apenas olhar para o link que Phoenix postou nos comentários, mas como ainda estou aprendendo, não quero olhar para a solução até que eu tenha descoberto. Isso pode me custar reputação, mas o aprendizado é para isso que estou aqui :)))

HyperNeutrino
fonte
LeakyNun fez isso em 5 no bate-papo: Spoiler
Pavel
5
oh .__________.
HyperNeutrino
3

Python 2 , 50 42 bytes

Economizou 8 bytes graças a @Sisyphus!

lambda l:map(sum,zip(l,[1,0,-1,0]*len(l)))

Experimente online!

53 bytes

lambda l:[int(x+(1j**i).real)for i,x in enumerate(l)]

Experimente online!

musicman523
fonte
lambda l:map(sum,zip(l,[1,0,-1,0]*len(l)))para Python 2
Sisyphus
Bom, isso economiza 5 bytes no Python 3 e mais 3 no Python 2. Obrigado!
precisa
3

Haskell , 26 bytes

@Mego me derrotou nesta solução

zipWith(+)$cycle[1,0,-1,0]

Experimente online!

É nisso que Haskell é ótimo. Isso declara uma função sem ponto que fecha a entrada com uma lista infinita.

Haskell , 56 bytes

Aqui está uma solução que usa números complexos. Não é muito competitivo por causa da importação, mas nunca é menos legal.

import Data.Complex
zipWith((+).realPart.((0:+1)^))[0..]

Experimente online!

Assistente de Trigo
fonte
2
Eek! Você me ninja por 20 segundos!
Mego 23/07
Não faz sentido ter duas soluções idênticas. Como você aceitou minha melhoria sem atribuição e tornou nossas respostas idênticas, você excluiria a sua?
Mego 23/07
3

Mathematica, 19 bytes

i=1;#+Im[i*=I]&/@#&

Explicação

i=1;#+Im[i*=I]&/@#&
i=1;                 (* set variable i to 1 *)
               /@#   (* iterate through the input: *)
    #+Im[i   ]&      (* add the imaginary component of i... *)
          *=I        (* multiplying i by the imaginary unit each iteration *)

Nota: i=1aparece fora da função, o que é aceitável para este meta consenso .

JungHwan Min
fonte
Mas então a função não é necessariamente reutilizável (se após uma chamada da função itiver um valor diferente de 1)
user202729
@ user202729 o meta consenso que vinculei lida especificamente com esse problema. Não há problema em declarar uma variável global fora de uma função.
JungHwan Min
3

J, 12 bytes

+1 0 _1 0$~#

Experimente online!

Como o operador de forma de J $preenche ciclicamente, quando o modelamos no comprimento #da entrada, ele faz exatamente o que queremos, e podemos apenas adicioná-lo à entrada]

Jonah
fonte
Você pode salvar um byte, soltando o primeiro] (ex. Use um gancho)
Tikkanz
@Tikkanz nice catch. Eu atualizei a postagem.
Jonah
3

C ++, 93 85 83 63 bytes

auto w=[](auto&i){for(int j=0;j<i.size();j+=2)i[j]+=j%4?-1:1;};

-8 bytes, graças a esta resposta , descobri que os parâmetros lambda podem ser autoe você pode passar com o parâmetro correto, ele funcionará

-2 bytes graças a Nevay

-2 bytes graças a Zacharý

Eu removi a vectorinclusão. Você precisará passar como argumento para o contêiner wa que respeite as seguintes condições:

  • Tenha um método chamado size sem argumentos
  • Sobrecarregaram o operador subscrito

Containers STL que respeitem as seguintes condições forem array, vector, string, map, unordered_map, e talvez outros

Se a saída modificando argumentos não for permitida, então:

C ++, 112 110 bytes

#include<vector>
std::vector<int>w(std::vector<int>i){for(int j=0;j<i.size();j+=2)i[j]+=(j%4)?-1:1;return i;}
HatsuPointerKun
fonte
11
Seu primeiro é E / S válida.
Pavel
11
Você pode usar j%4para salvar 2 bytes.
Nevay
11
Eu não acho que você precisa parens por aí j%4.
Zacharý
2

Dyalog APL, 13 bytes

⊢+1 0 ¯1 0⍴⍨≢

Experimente online!

Quão?

1 0 ¯1 0 - a matriz [1, 0, -1, 0]

⍴⍨≢ - remodelar para o comprimento da entrada, cíclico

⊢+ - soma vetorializada com a entrada

Uriel
fonte
2

Perl 6 , 28 bytes

{((1+0i,*×i...*)Z+$_)».re}

Experimente online!

1+0i, * × i ... *produz uma lista infinita dos números 1, i, -1, -irepetidos em um ciclo. Esses números são compactados com adição ( Z+) com a lista de entrada ( $_) e, em seguida, os componentes reais dos números complexos resultantes são extraídos ( ».re).

Sean
fonte
2

Japt , 11 10 bytes

Aproveita o empacotamento de índice do Japt.

Ë+[1TJT]gE

Teste-o


Explicação

Entrada implícita da matriz U.

Ë

Mapa sobre a matriz.

+

Para o elemento atual, adicione ...

gE

O elemento no índice atual ( E) ...

[1TJT]

Na matriz [1,0,-1,0].

Shaggy
fonte
1

Na verdade , 11 bytes

;r⌠╦½*C≈⌡M¥

Experimente online! (executa todos os casos de teste)

Explicação:

;r⌠╦½*C≈⌡M¥
;r           range(len(input))
  ⌠╦½*C≈⌡M   for each value in range:
   ˫*C      cos(pi/2*value)
       ≈     floor to integer
          ¥  pairwise addition of the input and the new list
Mego
fonte
1

Pyth, 11 bytes

.e+bss^.j)k

Try it online!

deltaepsilon3
fonte
Came up with another solution with the same byte count: .e+b@[1Z_1Z
clap
Replace ss with e for -1.
Erik the Outgolfer
Does that work? .e+be^.j)k didn't seem to work when I tried it.
deltaepsilon3
1

CJam, 15 bytes

l~_,,[1TW0]f=.+

Try it online!

Erik the Outgolfer
fonte
1

Math.JS, 34 bytes

f(k)=k.map(j(x,y,z)=x+im(i^y[1]))

Explained

f(k)=k.map(j(x,y,z)=x+im(i^y[1]))
f(k)=                               # Define a function f, which takes argument k.
     k.map(                     )   # Map k to a function
           j(x,y,z)=                # Function j. Takes arguments x, y, and z. Where x is the item, y is the index in the form [i], and z is the original list.
                      im(      )    # The imaginary component of...
                         i^y[1]     # i to the power of the index.
                    x+              # x +, which gives our wave.

Try it Online!

ATaco
fonte
1

8th, 96 63 bytes

Code

a:new swap ( swap 90 * deg>rad n:cos int + a:push ) a:each drop

This code leaves resulting array on TOS

Usage and examples

ok> [0,0,0,0,0] a:new swap ( swap 90 n:* deg>rad n:cos n:int n:+ a:push ) a:each drop .
[1,0,-1,0,1]

ok> [-4,3,0,1,7,9,8,-2,11,-88] a:new swap ( swap 90 * deg>rad n:cos int + a:push ) a:each drop .
[-3,3,-1,1,8,9,7,-2,12,-88]

Explanation

We use cos(x) in order get the right sequence [1,0,-1,0]. Each array element's index is multiplied by 90 degrees and then it is passed to cos() function to get the desired "wave factor" to be added to the corresponding item.

: f \ a -- a
  a:new    \ create output array
  swap     \ put input array on TOS
  \ array element's index is passed to cos in order to compute
  \ the "wave factor" to add to each item
  ( swap 90 n:* deg>rad n:cos n:int n:+ 
  a:push ) \ push new item into output array 
  a:each
  drop     \ get rid of input array and leave ouput array on TOS
;
Chaos Manor
fonte
1

C# (.NET Core), 50 bytes

n=>{for(int i=0;i<n.Length;i+=2)n[i]+=i%4<1?1:-1;}

Try it online!

Uses a simple lambda. Modifies the original array and returns the output through reference.

jkelm
fonte
1

05AB1E, 16 bytes

vy3L2.SR0¸«Nè+})

Try it online!


3L2.SR0¸« is the shortest thing I can think of for sin(x % 4) in 05AB1E.

Magic Octopus Urn
fonte