Weakened Binary Walls

21

1

Inspired by Create a binary wall

Given a list of positive integers, we can write them out all above each other like so, for [2, 6, 9, 4] as an example:

0010
0110
1001
0100

We can imagine this as a wall:

..#.
.##.
#..#
.#..

However, this is a very weak wall, and it has collapsed! Each 1 (#) falls down until it hits the "ground" or another 1 (#). The 0s (.s) are present in spots left by moved 1s.

This becomes the following:

....
....
.##.
####

Which translates back to:

0000
0000
0110
1111

Which, as a list of numbers, is [0, 0, 6, 15].

Another test case

[10, 17, 19, 23]

This becomes:

01010
10001
10011
10111

which becomes:

00000
10011
10011
11111

translating back to:

[0, 19, 19, 31]

Challenge

Given a list of positive integers, apply this transformation to the list. Input/Output as lists of positive integers in any reasonable format. Standard loopholes apply.

This is a , so the shortest answer in bytes wins!

HyperNeutrino

Posted 2017-07-29T19:35:52.737

Reputation: 26 575

Sandbox Post – HyperNeutrino – 2017-07-29T19:36:10.333

1More testcases? You know, non-square testcases would be good. – Leaky Nun – 2017-07-29T19:44:19.023

@LeakyNun Sure. I'll do that. – HyperNeutrino – 2017-07-29T19:44:37.533

That's just a sorting problem for bit arrays. – Marcus Müller – 2017-07-30T19:50:08.397

@MarcusMüller You're right - I realized that after the MATL answer :P – HyperNeutrino – 2017-07-30T20:14:27.617

@TheLethalCoder :) It was a nice challenge :) – HyperNeutrino – 2017-07-31T16:27:59.450

Answers

29

MATL, 4 bytes

BSXB

Try it at MATL Online

Explanation

    % Implicitly grab input as an array 
    %   STACK: [10, 17, 19, 23]
B   % Convert each element to binary where each decimal number results in a row
    %   STACK: [0 1 0 1 0;
    %           1 0 0 0 1;
    %           1 0 0 1 1;
    %           1 0 1 1 1]
S   % Sort each column, placing all of the 1's at the bottom of each column
    %   STACK: [0 0 0 0 0;
    %           1 0 0 1 1;
    %           1 0 0 1 1;
    %           1 1 1 1 1] 
XB  % Convert each row from its binary representation to its decimal number
    %   STACK: [0, 19, 19, 31]
    % Implicitly display the result

Suever

Posted 2017-07-29T19:35:52.737

Reputation: 10 257

o_O How does this work :o – HyperNeutrino – 2017-07-29T19:48:21.790

1Did MATL just out-golf Jelly by 4 bytes? o_O – totallyhuman – 2017-07-29T19:49:16.780

5 bytes now :-p – Leaky Nun – 2017-07-29T19:52:01.937

I never thought there'd be a built-in to move the ones to the bottom xD +1 – HyperNeutrino – 2017-07-29T19:52:03.673

@HyperNeutrino I basically also used sort on my answer – Leaky Nun – 2017-07-29T19:52:27.040

1@totallyhuman well, wait till Dennis comes – JungHwan Min – 2017-07-29T23:58:52.610

@LeakyNun The thing is, well, you had to use zip. – Erik the Outgolfer – 2017-07-30T07:56:52.017

@JungHwanMin I have tried this in Jelly too, I don't think it's really possible under 9 bytes... – Erik the Outgolfer – 2017-07-30T07:58:57.433

5

Python, 68 bytes

f=lambda a:a and[x|y&a[0]for x,y in zip([0]+f(a[1:]),f(a[1:])+[-1])]

Try it online!

Anders Kaseorg

Posted 2017-07-29T19:35:52.737

Reputation: 29 242

5

JavaScript (ES6), 50 bytes

f=a=>a.map(_=>a.map((e,i)=>a[a[i]|=a[--i],i]&=e))&&a

Explanation: Suppose two rows of the wall were like this:

0011
0101

The result needs to be this:

0001
0111

In other words, the first row becomes the AND of the two rows and the second row becomes the OR of the two rows. This just needs to be repeated enough times for all the bits to fall to the bottom.

Neil

Posted 2017-07-29T19:35:52.737

Reputation: 95 035

4

Jelly, 9 bytes

BUz0Ṣ€ZUḄ

Try it online!

Leaky Nun

Posted 2017-07-29T19:35:52.737

Reputation: 45 011

2

Japt, 16 bytes

m¤z3 ®¬n qÃz mn2

Try it online! using the -Q flag to format the array result.

Explanation

m¤z3 ®¬n qÃz mn2    Implicit: U = input array.
                        [10, 17, 19, 23]
m¤z3                Map U to binary strings and rotate the array left 90°
                         1010       0111
                        10001   ->  1011
                        10011       0001
                        10111       1000
                                     111
®¬n qà              Sort each binary string, putting 0s and spaces at the start
                        0111
                        0111
                        0001
                        0001
                         111
z mn2               Rotate right 90° and convert each back to a number
                         0000       0
                        10011   ->  19
                        10011       19
                        11111       31
                    Implicit output of resulting array

Justin Mariner

Posted 2017-07-29T19:35:52.737

Reputation: 4 746

I think you can save a byte with mì2 z3 mn z mì2 – ETHproductions – 2017-07-30T01:10:23.853

@ETHproductions It seems rotating the 2D array, instead of rotating the array of strings, pads each inner array with null instead of spaces. So that doesn't seem to work. And null is sorted to the right of the 1s, unlike spaces, which are sorted to the left. – Justin Mariner – 2017-07-30T01:15:51.533

2

Octave, 29 25 bytes

4 bytes saved thanks to @Stewie

@(x)bi2de(sort(de2bi(x)))

Suever

Posted 2017-07-29T19:35:52.737

Reputation: 10 257

de2bi/bi2de saves 4 bytes in octave. Works on octave-online.net. – Stewie Griffin – 2017-07-30T15:27:09.313

@StewieGriffin Thanks! – Suever – 2017-07-30T18:02:18.697

2

Mathematica, 64 bytes

#~FromDigits~2&/@(Sort/@(PadLeft[#~IntegerDigits~2&/@#]))&

 is \[Transpose]

This converts the input (a list of numbers) to a list of lists of digits, pads it to be a square matrix, transposes it, sorts the rows so the 1's "fall" to the bottom, transposes back, then converts back into numbers.

DanTheMan

Posted 2017-07-29T19:35:52.737

Reputation: 3 140

2

Python 3.5, 60 bytes

def f(a,*t):
 if t:b,*r=f(*t);t=f(a|b,*r);a&=b
 return(a,*t)

Try it online!

Takes input like f(2, 6, 9, 4). Assumes input is non-empty. Uses a lot of tuple unpacking.

xnor

Posted 2017-07-29T19:35:52.737

Reputation: 115 687

1

J, 13 bytes

/:~"1&.|:&.#:

Try it online!

Explanation

/:~"1&.|:&.#:  Input: array M
           #:  Convert each in M to binary with left-padding
       |:&     Transpose
/:~"1&         Sort each row
     &.|:      Inverse of transpose (which is just transpose)
         &.#:  Inverse of converting to binary

miles

Posted 2017-07-29T19:35:52.737

Reputation: 15 654

There's that binary left-padding again, +1. And also, can you explain why you would need to use the inverse of transpose, since it is just transpose? – Zacharý – 2017-07-30T15:32:16.047

@Zacharý The inverses are used to undo the operations used before sorting each row. It's true that the inverse of transpose is just transpose, but another way to see this is as <convert from binary> <transpose> <sort each row> <transpose> <convert to binary> M, where the first two functions are just the inverses of the last two. – miles – 2017-08-01T02:58:21.830

1

05AB1E, 9 bytes

bí0ζR€{øC

Try it online!

Kinda different algorithm from Magic's.

Erik the Outgolfer

Posted 2017-07-29T19:35:52.737

Reputation: 38 134

ζ, damnit. Deleted mine, take my +1. – Magic Octopus Urn – 2017-07-31T13:28:04.473

@MagicOctopusUrn Why did you delete yours? No need to. – Erik the Outgolfer – 2017-07-31T13:29:18.213

It's not really much different (in terms of algorithm) and this was 25% better. – Magic Octopus Urn – 2017-07-31T13:31:29.220

1

Dyalog APL (23 characters)

{2⊥¨↓⍉↑{⍵[⍋⍵]}¨↓2⊥⍣¯1⊢⍵}
  1. Convert the input arguments into a binary matrix
  2. Split the matrix into columns
  3. Sort the columns into ascending order
  4. Convert the sorted rows back into decimal

Example

  {2⊥¨↓⍉↑{⍵[⍋⍵]}¨↓2⊥⍣¯1⊢⍵}10 17 19 23
      0 19 19 31

Thanks to Zacharý for correcting me on this one.

James Heslip

Posted 2017-07-29T19:35:52.737

Reputation: 139

You can replace with (⊥⍣¯1)⍵ with ⊥⍣¯1⊢⍵. Also, I don't think you need the axis specification on split (↓[1]=>). – Zacharý – 2017-07-30T14:33:27.410

Oh, and you're supposed to convert it back to a list! – Zacharý – 2017-07-30T14:46:41.770

This is invalid. – Zacharý – 2017-07-30T17:28:45.453

Thank you, Zacharý, I was working on this late last night and I think I misread the problem. I've modified my solution now. – James Heslip – 2017-07-30T18:59:59.303

1Well, good job! (⊥⍣¯1 really needs to be a builtin). And thank you for actually getting my username right. – Zacharý – 2017-07-30T19:32:13.503

1

Dyalog APL, 24 21 19 bytes

2⊥↑{⍵[⍋⍵]}¨↓2⊥⍣¯1⊢⎕

Try it online! (modified so TryAPL accepts it as valid)

How?

  • evaluated input (arrays are space separated)
  • 2⊥⍣¯1⊢ converts each each of the arguments to binary (transposed of what is in the question)
  • turns a 2D array into a vector of vectors
  • {⍵[⍋⍵]}¨ sorts each of the elements of the vector
  • turns the vector of vectors into a 2D array again
  • 2⊥ convert from binary (since it sort of transposes it, we arrive at the correct result)

Zacharý

Posted 2017-07-29T19:35:52.737

Reputation: 5 710

0

JavaScript, 127 125 bytes

a=>a[m='map'](_=>b[m]((n,i)=>n&&(b[i]--,d|=1<<i),d=0)&&d,b=[...Array(32)][m]((_,c)=>a[m](e=>d+=!!(2**c&e),d=0)&&d)).reverse()

Try it online

-2 bytes thanks to Cows quack

user72349

Posted 2017-07-29T19:35:52.737

Reputation:

(1<<c)&e can become 2**c&e – user41805 – 2017-07-29T20:30:01.910

0

Python 2, 142 bytes

... and still golfing... hopefully –– Any help appreciated!

def c(l):b=[bin(n)[2:]for n in l];print[int(n,2)for n in map(''.join,zip(*map(sorted,zip(*['0'*(len(max(b,key=len))-len(x))+x for x in b]))))]

A big chunk of this is for padding the numbers with zeroes.

More readable:

def collapse(nums):
    bins = [bin(n)[2:] for n in nums]
    bins = [('0'*(len(max(bins, key = len)) - len(x))) + x for x in bins]
    print [int(n, 2) for n in map(''.join, zip(*map(sorted, zip(*bins))))]

This creates an array of the binary string representations, pads it, rotates it 90º clockwise, sorts each row, rotates it back 90º, and then creates integers out of each row.

Daniel

Posted 2017-07-29T19:35:52.737

Reputation: 6 425

142 bytes, you have some redundant parenthesis. – Mr. Xcoder – 2017-07-30T05:39:49.380

@Mr.Xcoder , oh yes that was silly – Daniel – 2017-07-30T16:24:45.747