Мінімально сортуйте список у матриці


18

Враховуючи несортований список унікальних суворопозитивних цілих чисел, мінімально сортуйте їх у 2D матриці. Список входу гарантовано композітнога довжини, що означає , що вихід матриця не обов'язково квадрата, але має розміру n x mз n,m > 1.

"Мінімально сортувати" тут означає наступне:

  • Сортувати список у порядку зростання.
  • Максимально ущільнюйте вихідну матрицю - мінімізуйте суму розмірів матриці (наприклад, для 20вхідних елементів як вхідних даних потрібна матриця a 5x4або 4x5вихід, а не a 2x10).
  • Ущільнюйте відсортовані числа якомога далі в лівій верхній частині матриці, починаючи з першого елемента в відсортованому списку.
  • Це можна вважати сортуванням списку, а потім нарізанням його по антидіагоналях матриці, починаючи з верхнього лівого.

Приклади:

Для вихідного 1..20виходу використовується матриця 5x4 або 4x5 наступним чином:

 1  2  4  7 11
 3  5  8 12 15
 6  9 13 16 18
10 14 17 19 20

 1  2  4  7
 3  5  8 11
 6  9 12 15
10 13 16 18
14 17 19 20

Для вихідного [3, 5, 12, 9, 6, 11]виходу є 2x3 або 3x2 наступним чином

3  5  9
6 11 12

 3  5
 6  9
11 12

Для введення [14, 20, 200, 33, 12, 1, 7, 99, 58], вихід 3x3 наступним чином

 1   7  14
12  20  58
33  99 200

Для введення 1..10вихід повинен бути 2x5 або 5x2 наступним чином

1 2 4 6  8
3 5 7 9 10

1  2
3  4
5  6
7  8
9 10

Для вихідного [5, 9, 33, 65, 12, 7, 80, 42, 48, 30, 11, 57, 69, 92, 91]виходу є 5x3 або 3x5 наступним чином

 5  7 11 33 57
 9 12 42 65 80
30 48 69 91 92

 5  7 11
 9 12 33
30 42 57
48 65 80
69 91 92

Правила

  • Можна вважати, що вхід відповідає вхідному цілому типу вашої мови.
  • Введення та вихід можуть бути надані будь-яким зручним методом .
  • Прийнятна або повна програма, або функція. Якщо функція, ви можете повернути вихід, а не надрукувати його.
  • Стандартні лазівки заборонені.
  • Це тому застосовуються всі звичайні правила гольфу, і найкоротший код (у байтах) виграє.

1
О, ух, слова, якого я не бачив з моменту лінійної алгебри; легко не помітити. Мої вибачення.
Чарівний восьминога Урна

@LuisMendo Додано 15тестовий випадок елемента.
AdmBorkBork

Відповіді:


10

Желе , 24 22 20 байт

pS€ỤỤs
LÆDżṚ$SÞḢç/ịṢ

Спробуйте в Інтернеті!

Збережено 2 байти завдяки @ Jonathan Allan .

Пояснення

pS€ỤỤs  Helper link. Input: integer a (LHS), integer b (RHS)
p       Cartesian product between [1, 2, ..., a] and [1, 2, ..., b]
 S€     Sum each pair
   Ụ    Grade up
    Ụ   Grade up again (Obtains the rank)
     s  Split into slices of length b

LÆDżṚ$SÞḢç/ịṢ  Main link. Input: list A
L              Length
 ÆD            Divisors
     $         Monadic pair
    Ṛ            Reverse
   ż             Interleave
                 Now contains all pairs [a, b] where a*b = len(A)
      SÞ       Sort by sum
        Ḣ      Head (Select the pair with smallest sum)
         ç/    Call helper link
            Ṣ  Sort A
           ị   Index into sorted(A)

L%J¬TżṚ$-> LÆDżṚ$повинні врятувати двох, я думаю
Джонатан Аллан

Першим посиланням може стати pSÞỤs.
Денніс

4

Пітон 2 , 160 158 153 151 байт

-2 байта завдяки Еріку Перевершив
-2 байти завдяки містеру Xcoder

s=sorted(input())
l=len(s)
x=int(l**.5)
while l%x:x+=1
n=1
o=eval(`l/x*[[]]`)
while s:
 for i in range(l/x)[max(0,n-x):n]:o[i]+=s.pop(0),
 n+=1
print o

Спробуйте в Інтернеті! або Спробуйте всі тестові приклади


Вірю, ви можете використовувати max(0,n-x)для -2 байт.
Містер Xcoder

4

R 110 95 байт

function(x){n=sum(x|1)
X=matrix(x,max(which(!n%%1:n^.5)))
X[order(col(X)+row(X))]=sort(x)
t(X)}

Спробуйте в Інтернеті!

Як це працює

f <- function(x) {
  n <- sum(x|1)                           # length
  p <- max(which(!n%%1:n^.5))             # height of matrix
  X <- matrix(x, p)                       # initialize matrix
  X[order(col(X) + row(X))] <- sort(x)    # filling the matrix using position distance to the top left corner
  t(X)                                    # probably required by OP
}

Наступні хитрощі Джузеппе врятував колосальних 15 (!) Байтів

  • заміни length(x)на sum(x|1)(-1 байт)
  • floor()не потрібно, так як :все-таки круги вниз (-7)
  • ^.5коротше sqrt()(-3)
  • використання col(X) + row(X)замість outer(приємно!)
  • не міг позбутися t(X)хоч - невтішно;)

Оригінальне рішення

function(x){
n=length(x)
p=max(which(!n%%1:floor(sqrt(n))))
X=outer(1:p,1:(n/p),`+`)
X[order(X)]=sort(x)
t(X)}

Здавалося б, більш фантазійним outerє його заміна row(X)+col(X), але для цього потрібно спочатку ініціалізувати вихідну матрицю X.

Спробуйте в Інтернеті!


2
Дуже хороша! Ви можете опуститися до 95 байт
Джузеппе,

1
Можливо, я можу використовувати щось від мого рішення до пов'язаного завдання, щоб допомогти і тут.
Джузеппе

Це дійсно тісно пов'язане. Дуже приємний підхід!
Майкл М

3

JavaScript (ES6), 172 байти

l=>(n=l.sort((a,b)=>b-a).length,w=l.findIndex((_,i)=>!(i*i<n|n%i)),a=l=>[...Array(l)],r=a(n/w).map(_=>a(w)),a(w+n/w).map((_,x)=>r.map((s,y)=>x-y in s&&(s[x-y]=l.pop()))),r)

Пояснення

l=>(                                // Take a list l as input
 l.sort((a,b)=>b-a),                // Sort it
 n=l.length,                        // Get the length n
 w=l.findIndex((_,i)=>!(i*i<n|n%i)),// Find the first integer w where w >= √n and n % w = 0
 a=l=>[...Array(l)],                // Helper function a
 r=a(n/w).map(_=>a(w)),             // Create the grid r of size w, n/w
 a(w+n/w).map((_,x)=>               // For every x from 0 to w + n/w:
  r.map((s,y)=>                     //  For every row s in r:
   x-y in s&&(                      //   If the index x-y is in s:
    s[x-y]=l.pop()))),              //    Set s[x-y] to the next element of l
 r)                                 // Return r

Випробування


3

Perl 5 , 132 байти

sub d{$,=0|sqrt(@_=sort{$a-$b}@_);--$,while@_%$,;map{$r++,$c--for@_/$,..$c;$a[$r++][$c--]=$_;$c=++$i,$r=0if$r<0||$c<0||$r>=$,}@_;@a}

Спробуйте в Інтернеті!

Підпрограма повертає двовимірний масив. Посилання TIO включає код нижнього колонтитулу для відображення результату тесту.


3

Октава , 151 байт

function f(v)n=floor(sqrt(l=nnz(v)));while i=mod(l,n);++n;end;A=nan(m=l/n,n);for k=[1:m 2*m:m:l];do A(k)=sort(v)(++i);until~mod(k+=m-1,m)|k>l;end;A'end

Використання трьох різних типів контурних конструкцій.

Спробуйте в Інтернеті!

Розгорнуто:

function f(v)
    n = floor(sqrt(l=nnz(v)));

    while i = mod(l,n);
        ++n;
    end;

    A = nan(m=l/n, n);

    for k = [1:m 2*m:m:l];
        do
            A(k) = sort(v)(++i);
        until ~mod(k+=m-1, m) | k>l;
    end;

    A'
end

Гарна відповідь! Чому 'в nnz(v') необхідності?
Луїс Мендо

1
@LuisMendo Дякую! Виявляється значення 'не потрібно, якщо я обертаю вираз діапазону, наприклад 1:20, навколо дужок ( [1:20]) на сайті виклику (щоб зробити його фактичним вектором). Мабуть, в Октаві оператор двокрапки не створює вектор , а константа діапазону, яка займає набагато менше місця в пам'яті. Чомусь nnz()це не працює з таким типом, але транспонування константи діапазону дає вектор, тому воно працює з апострофом. Виклик функції за допомогою фактичного вектора знімає потребу в '.
Steadybox

1
Дякую за пояснення. Я не знав, що діапазон експресії мав таку особливу обробку в Октаві. У всякому разі, той факт, що він не створює вектор для ефективності пам'яті, повинен бути прозорим для програміста. Тобто, те, що nnz(1:20)не працює, ймовірно, є помилкою ( max(1:20)і sum(1:20)т. Д. Дійсні).
Луїс Мендо

1
Ми повинні повідомити про це . Це може вплинути на інші функції, ніж nnz. Ви хочете зробити це самостійно, чи я?
Луїс Мендо

1
Повідомляється . Це також вплинуло на MATL; тепер вирішено . Дякуємо, що помітили це!
Луїс Мендо

0

Лушпиння , 15 байт

ḟȯΛ≤Σ∂MCP¹→←½ḊL

Це працює з грубою силою, тому довші тестові справи можуть затримати час. Спробуйте в Інтернеті!

Пояснення

ḟȯΛ≤Σ∂MCP¹→←½ḊL  Implicit input, a list of integers x.
              L  Length of x (call it n).
             Ḋ   List of divisors.
            ½    Split at the middle.
          →←     Take last element of first part.
                 This is a divisor d that minimizes d + n/d.
        P¹       List of permutations of x.
      MC         Cut each into slices of length d.
ḟ                Find the first of these matrices that satisfies this:
     ∂            Take anti-diagonals,
    Σ             flatten them,
 ȯΛ≤              check that the result is sorted (each adjacent pair is non-decreasing).


0

JavaScript (ES6), 233 байти

f=s=>{l=s.length;i=Math.sqrt(l)|0;for(;l%++i;);p=(x)=>(x/i|0+x%i)*l+x%i;m=[...Array(l).keys()].sort((x,y)=>p(x)-p(y));return s.sort((a,b)=>a-b).map((x,i)=>m.indexOf(i)).reduce((a,b,d,g)=>!(d%i)?a.concat([g.slice(d,d+i)]):a,[])}

Пояснення

f=s=>{                         // Take array `s` of numbers as input
  l=s.length                   // short-hand for length
  i=Math.sqrt(l)|0             // = Math.floor(Math.sqrt(l))
  for(;l%++i;);                // i = width           
  j=l/i                        // j = height

  p=(x)=>(x/i|0+x%i)*l+x%i     // helper to calculate (sort-of) ~manhattan
                                 // distance (horizontal distance weighted
                                 // slightly stronger), from top-left corner
                                 // to the number x, if numbers 0,...,l are
                                 // arranged left-to-right, top-to-bottom
                                 // in an l=i*j grid

  m=[...Array(l).keys()]         // range array
  .sort((x,y)=>p(x)-p(y)),       // manhatten-sorted, sort-of...

  return s.sort((a,b)=>a-b)      // sort input array by numbers,
    .map((x,i,w)=>w[m.indexOf(i)])    // then apply inverse permutation of the
                                 // range-grid manhatten-sort mapping.
    .reduce(                     // slice result into rows
      (a,b,d,g)=>!(d%i)?a.concat([g.slice(d,d+i)]):a
      ,[]
     )
}

0

Java 10, 199 188 186 байт

a->{int j=a.length,m=0,n,i=0,k=0;for(n=m+=Math.sqrt(j);m*n<j;n=j/++m);var R=new int[m][n];for(java.util.Arrays.sort(a);i<m+n;i++)for(j=0;j<=i;j++)if(i-j<n&j<m)R[j][i-j]=a[k++];return R;}

Спробуйте в Інтернеті.

Виходячи з моєї відповіді тут .

Пояснення:

a->{                        // Method with int-array parameter and int-matrix return-type
  int j=a.length,           //  Length of the input-array
      m=0,n,                //  Amount of rows and columns
      i=0,k=0;              //  Index integers
   for(n=m+=Math.sqrt(j);   //  Set both `m` and `n` to floor(√ `j`)
       m*n<j;               //  Loop as long as `m` multiplied by `n` is not `j`
       n=j/++m);            //   Increase `m` by 1 first with `++m`
                            //   and then set `n` to `j` integer-divided by this new `m`
   var R=new int[m][n];     //  Result-matrix of size `m` by `n`
   for(java.util.Arrays.sort(a);
                            //  Sort the input-array
       i<m+n;)              //  Loop as long as `i` is smaller than `m+n`
     for(j=0;j<=i;j++)      //   Inner loop `j` in range [0,`i`]
       if(i-j<n&j<m)        //    If `i-j` is smaller than `n`, and `j` smaller than `m`
                            //    (So basically check if they are still within bounds)
         R[j][i-j]=a[k++];  //     Add the number of the input array at index `k`,
                            //     to the matrix in the current cell at `[j,i-j]`
  return R;}                //  Return the result-matrix
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.