Сортуйте спочатку непарні числа


20

Перестановіть заданий список таким чином, щоб усі непарні числа з’явились перед усіма непарними числами. Крім цієї вимоги, вихідний список може бути в будь-якому порядку.

Вхід буде містити лише цілі числа, але вони можуть бути негативними і можуть бути дублікати, і вони можуть відображатися в будь-якому порядку.

Найкоротше рішення виграє.

Тестові справи

[1,2][1,2]

[2,1][1,2]

[1,0,0][1,0,0]

[0,0,-1][-1,0,0]

[3,4,3][3,3,4]

[-4,3,3][3,3,-4]

[2,2,2,3][3,2,2,2]

[3,2,2,2,1,2][1,3,2,2,2,2]або[3,1,2,2,2,2]

[-2,-2,-2,-1,-2,-3][-1,-3,-2,-2,-2,-2,]або[-3,-1,-2,-2,-2,-2,]

[][]


Тай. Гарне питання. Відповідь: непарні номери можуть бути в будь-якому порядку. :)
display_name

11
Незважаючи на те, що виклик досить простий, додавання деяких тестових випадків було б непогано. Наприклад, на перший погляд я подумав, що блок непарних і парних чисел також потребує сортування.
Лайконі

1
@AsoneTuhid Так :), цифри можуть повторюватися.
display_name

11
@Willmore Ніколи не знаєш з кодом гольф, важливі правила. Будь ласка, використовуйте пісочницю наступного разу, щоб уточнити своє запитання перед тим, як опублікувати його.
Асона Тухід

12
Будь ласка, відредагуйте своє запитання, щоб включити уточнення, які ви дали у коментарях.
Лайконі

Відповіді:





9

R , 30 24 байти

(x=scan())[order(!x%%2)]

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

-6 байт завдяки JayCe


Приємно! ви можете зробити 26 байт
JayCe


@JayCe Facepalm гольфом о 3 годині ночі з новим дитиною не є оптимальним. Спасибі!
Джузеппе

5
Вітаю! Майбутній гольфіст Scratch?
JayCe

1
Мої привітання також Джузеппе, як сказав нам наш доктор, молодець, у вас дорогий :)
MickyT

9

C ++, 79 76 64 байт

[](auto a,auto b){while(a<--b)*a%2?*++a:(*a^=*b,*b^=*a,*a^=*b);}

Ця функція приймає пару ітераторів (які повинні бути ітераторами випадкового доступу) і неухильно переміщує їх назустріч один одному. Коли aвказує на непарне число, воно просувається. Інакше aвказує на парне число; bзменшується та iter_swapредагується за допомогою a. (Ми використовуємо XOR своп, що дозволяє економити нам необхідність включити <algorithm>- або<utility> дляstd::swap ).

Коли виникають зайві заміни, коли b вказує на парне число, але ми займаємось гольфом, не знижуючи ефективність!

Демо

auto f=[](auto a,auto b){while(a<--b)*a%2?*++a:(*a^=*b,*b^=*a,*a^=*b);};

#include <array>
#include <iostream>
int main()
{
    auto a = std::array{ 3,2,2,5,2,1,2 };

    f(a.begin(),a.end());

    for (auto i: a)
        std::cout << i << " ";
    std::cout << std::endl;
}

Неконкурентна відповідь

Природний метод C ++ є std::partition, але він виходить у 83 байти:

#include<algorithm>
[](auto a,auto b){std::partition(a,b,[](auto x){return x&1;});}

Я вважаю, що це 80 байт, оскільки вам потрібен новий рядок після #includeдирективи. Моя математика гасить хоч ^^. Ви можете замінити !=з -, економлячи 1 байт. Мені подобається ваш підхід, це розумно!
OOBalance

1
інакше ітератори могли проходити один одного, не стаючи рівними. Якщо ви використовуєте RandomAccessIterator , ви можете використовувати, while(a<b)якщо це зручніше, ніж a!=bвикористовувати версію @ OOBalance a-b.
Пітер Кордес

Ви можете скоротити 83-Byte відповіді трохи, замінивши algorithmз regex: codegolf.stackexchange.com/a/150895
OOBalance


7

Perl 6 , 12 байт

*.sort(*%%2)

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

Деякий код, який би сортував вхід за парністю, спочатку з непарними числами. Ви можете видалити a, %щоб отримати спочатку парні числа. Зауважте, що "Що б там не було" - це назва цього роду анонімної функції.


1
Вибачте! Я випадково відредагував вашу відповідь замість моєї!
Час Браун




5

Haskell , 23 22 байти

f odd<>f even
f=filter

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

g x = filter odd x ++ filter even x

-1 байт завдяки Лінні


Інші підходи:

(<>).($odd)<*>($even)$filter
f x=[i|m<-[0,1],i<-x,odd$m+i]
f x=[i|m<-[1,0],i<-x,mod i 2==m]
f x=id=<<filter<$>[odd,even]<*>[x]

Але це не потрібно import Data.Semigroup?
AlexJ136

1
@ AlexJ136 Станом на GHC 8.4.1, (<>)є частиною Прелюдії . Оскільки TIO все ще працює зі старшою версією, імпорт потрібен там. Але ти маєш рацію, я мав би це прямо згадати.
Лайконі

1
k odd<>k even;k=filterзберігає байт.
Лінн


5

JavaScript (Node.js) , 29 байт

a=>a.sort((a,b)=>(b&1)-(a&1))

Спробуйте в Інтернеті! Збережіть 4 байти, підтримуючи лише додатні значення b%2-a%2. Якщо ви пишете це як:

function(a){return a.sort((a,b)=>(b&1)-(a&1))}

тоді він буде працювати над усіма видами старих реалізацій JavaScript, які не сортувались стабільно.


1
Не a=>a.sort((a,b)=>b&1-a&1)працює?
Алексіс Факс

1
@AlexisFacques Ні, це розбирає як b&(1-a)&1.
Ніл

1
a=>a.sort(a=>++a&1)коротше :)
Макс

@Max Це може працювати на наведених тестових випадках, але я не здивуюсь, якщо хтось знайде приклад, коли це не працює.
Ніл

1
@Max Ви можете також подати це як власну відповідь.
Ніл

5

T-SQL, 26 байт

SELECT*FROM t ORDER BY~i&1

Використовує побітовий AND оператор "&" для порівняння останньої цифри з 1.

EDIT: Побіт НЕ потім коротший, ніж додавання 1. EDIT2: упорядкувати, щоб дозволити видалення пробілу.


1
Приємно! Бийте мене на 5! Збережіть ще один байт, ORDER BY~i&1
поміняючи


4

JavaScript, 22 20 байт

a=>a.sort(a=>!(a%2))

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


Я думаю, ви можете скинути дужки навколо своєї третьої a.
Джонатан Фрех

Не працює, якщо 0він включений до масиву.
Кудлатий

Це неправильно. js компаратор не працює таким чином. developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/…
Qwertiy

2
According to the ECMA specification, "If comparefn is not undefined and is not a consistent comparison function for the elements of this array (see below), the behaviour of sort is implementation-defined." This compare function is not consistent. So this is not a JavaScript answer, but it might be an answer for some particular JavaScript implementation, and you'd have to name which implementation.
user5090812

1
I think this fails for [1,2,3,4,5,6,6,-1,-2,-3,-4]. JavaScript array.sort is weird.
Chas Brown

4

PHP, 55 bytes

~14 months later and I'm a little bit better at golfing now:

for(;''<$n=$argv[++$i];$s=$n%2?"$n $s":"$s $n");echo$s;

Try it online!


PHP (>=5.4), 84 82 bytes

(-2 bytes, thanks to Ismael Miguel)

<?array_shift($a=&$argv);usort($a,function($i){return$i%2==0;});echo join(' ',$a);

To run it:

php -n <filename> <number_1> <number_2> ... <number_n>

Example:

php -n sort_odds_first.php 2 0 1 -1 3 8 29 -666

Or Try it online!


1
Instead of $a=array_slice($argv,1);, use array_shift($a=&$argv);, which saves 1 byte. Also, remove the space before $a in join(' ', $a), saving another byte. Also, PHP 5.3 gives different results. You should specify for which version of PHP this solution is for.
Ismael Miguel

1
@IsmaelMiguel: Thanks for the array_shift idea and pointing out the space mistake. I'm not sure how did I miss the space :D I have added the PHP >= 5.4 in title as well.
Night2

It is a common mistake. I actually was surprised by the array_shift when I tried it and worked.
Ismael Miguel


3

Husk, 4 bytes

↔Ö%2

Try it online!

Explanation

 Ö     sort input according to the result of the following function
  %2   modulo 2
↔      reverse result to get odd numbers to the front


3

C#, 23 bytes

i=>i.OrderBy(u=>u%2==0)

Quite straigt forward really: This basically converts the numbers to booleans, whilst true means that the number is even and false that it's odd. Because true is higher than false the even numbers appear first.

The formatted version looks like that:

i => i.OrderBy (u => u % 2 == 0)

And you can test it like that:

Console.WriteLine (string.Join (",", new Func <IEnumerable <int>, IEnumerable <int>> (
                                    i => i.OrderBy (u => u % 2 == 0)
                                ).Invoke (new [] {3, 2, 2, 2, 1, 2, 5, 5})));

Which results in the following:

3,1,5,5,2,2,2,2


3

JavaScript, 23 bytes

6 bytes shorter than @Neil's answer using the same language :D

a=>a.sort(n=>-(n&1)||1)

Explanation:

The function passed to sort only cares about the first parameter. If it is odd it returns -1 (the result of -(n&1)). Otherwise (when -(n&1) yields 0) it returns 1.

Try it online!


2
Welcome to PPCG!
Jonathan Frech


3

JavaScript (Chrome v67) - 24 19 23 bytes

a=>a.sort(a=>!(a&1)-.5)

The use of &1 rather than Math.abs()%2 was stolen from @Neil. Thanks!

Thanks to @Shaggy for showing my hacky 19 byte solution wasn't valid. If anyone wants it:

Depends on how the browser handles a hacky return value of 0. Chrome v67, after 100000 iterations of random arrays never sorted it wrong. I'm very sure it works -- and it relies on the specific sort algorithm Chrome uses too, I believe. (It might work in other browsers, that's not the point)

a=>a.sort(a=>++a&1)


Welcome to PPCG :) This fails for input [-5,-4,-3,-2,-1,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19] in my Chrome 67 console, outputting [7,-5,-3,17,-1,15,1,13,3,11,5,9,2,19,14,-4,6,18,-2,16,0,10,8,12,4].
Shaggy

@Shaggy -- oops! you're absolutely right!
Max


3

PowerShell, 22 19 bytes

$args|sort{!($_%2)}

Try it online!

Takes input via splatting, e.g., $a=(3,4,3); .\sort-odd-numbers-first.ps1 @a, which on TIO manifests as separate arguments for each entry.

Like some other answers here, Sort-Object can compare based on an expression. Here the expression is !($_%2), i.e., odds get sorted to $false and evens get sorted to $true. Thanks to how Boolean values are compared, falsey values are sorted first. This moves the odds to the beginning of the output, and the evens to the end. Sort-Object is stable, so the ordering of the individual items in their respective categories doesn't change (as in the TIO example).

-3 bytes thanks to mazzy.


It can to use a splatting. For example $a=(3,4,3); .\sort-odd-numbers-first.ps1 @a. So $args|sort{!($_%2)} is enough. Isn't it?
mazzy

why "cheating"? it's native powershell feature. One more question: can we to use splatting inside codeGolf solution? for example, a solution contains several functions. if we can then why external call should not? if we cannot then why this feature banned? and what features are banned too?
mazzy

1
@mazzy Thanks for pointing that out. I've updated my submission.
AdmBorkBork

3

Ruby, 23 bytes

->a{a.sort_by{|i|~i%2}}

Try it online!

Explanation:

sort_by sorts every number as if its value were the result of the block (~i%2)

~x is equivalent to -x-1 and takes precedence over %2

Odd numbers will evaluate to 0 and even numbers will evaluate to 1 so odd numbers will be sorted first.

Barely related: this works on ruby from homebrew 2.5.1p57 (because it's based on a small bug) but only for non-negative integers, 20 bytes

->a{a.sort{|i|i%-2}}

Explanation:

This uses sort which expects a block that takes 2 values and returns -1, 0 or 1 depending on whether the first one is bigger, they're equal or the second one is bigger.

The block given here ignores the second value and returns -1 if the first number is odd or 0 if it's even.

It's not guaranteed to work but it does in some (I think buggy) implementations.


We define languages by their implementation here so your 20 byte solution is valid.
Shaggy

@Shaggy Never mind, I had messed up my testing yesterday.
Asone Tuhid

3

6502 machine code routine, 47 bytes

A0 00 84 FE B1 FB 4A 90 07 C8 C4 FD F0 20 D0 F4 2A 85 02 84 FE A4 FD 88 C4 FE
F0 12 B1 FB 4A 90 F6 2A AA A5 02 91 FB A4 FE 8A 91 FB 90 D6 60

Expects a pointer to an array of numbers in $fb/$fc and the length of this array in $fd. Manipulates the array in place to have all odd numbers in front. This is position independent code, so no load address is needed.

As the 6502 is an 8bit chip (so the instructions only deal with 8bit values, optionally signed), the valid number range is [-128 .. 127] and the maximum array size is 256.

Commented disassembly

; function to "partially sort" array, so all odd numbers come before all
; even numbers.
;
; input:
;   $fb/$fc: address of array to sort
;   $fd:     length of array to sort, 0 means 256 (maximum size)
;
; clobbers:
;   $fd/$fe: position from back/front of array
;   $2:      temporary for exchanging two values
;   A, X, Y

 .oddfirst:
A0 00       LDY #$00            ; initialize index from front
84 FE       STY $FE             ; to 0

 .search_front:
B1 FB       LDA ($FB),Y         ; load number from front
4A          LSR A               ; check for even/odd by shifting
90 07       BCC .search_back    ; if odd -> to searching from back
C8          INY                 ; next position from front
C4 FD       CPY $FD             ; same as position searching from back?
F0 20       BEQ .done           ; then we're finished
D0 F4       BNE .search_front   ; else check next from front
 .search_back:
2A          ROL A               ; shift carry back in
85 02       STA $02             ; and save number to temp
84 FE       STY $FE             ; save index from front
A4 FD       LDY $FD             ; load index from back
 .sb_loop:
88          DEY                 ; previous position from back
C4 FE       CPY $FE             ; same as position searching from front?
F0 12       BEQ .done           ; then we're finished
B1 FB       LDA ($FB),Y         ; load number from back
4A          LSR A               ; check for even/odd by shifting
90 F6       BCC .sb_loop        ; if odd -> check previous position
2A          ROL A               ; shift carry back in
AA          TAX                 ; remember in X
A5 02       LDA $02             ; load temporary from front
91 FB       STA ($FB),Y         ; store at current position
A4 FE       LDY $FE             ; load index from front
8A          TXA                 ; load remembered number
91 FB       STA ($FB),Y         ; store at current position
90 D6       BCC .search_front   ; and back to searching from front
 .done:
60          RTS

Example C64 assembler program using the routine:

Online demo

screenshot

Code in ca65 syntax:

.import oddfirst ; link with routine above

.segment "BHDR" ; BASIC header
                .word   $0801           ; load address
                .word   $080b           ; pointer next BASIC line
                .word   2018            ; line number
                .byte   $9e             ; BASIC token "SYS"
                .byte   "2061",$0,$0,$0 ; 2061 ($080d) and terminating 0 bytes

.bss
linebuf:        .res    5               ; maximum length of a valid signed
                                        ; 8-bit number input
convbuf:        .res    3               ; 3 BCD digits for signed 8-bit
                                        ; number conversion
numbers:        .res    $100            ; maximum array size that can be
                                        ; directly handled with indexing
                                        ; instructions

.data
prompt:         .byte   "> ", $0
message:        .byte   $d, $d, "Enter one number per line.", $d
                .byte   "just press enter (empty line) when done.", $0
errmsg:         .byte   "Error parsing number, try again.", $d, $0

.code
                lda     #$17            ; set upper/lower mode
                sta     $d018

                lda     #0
                sta     $2a             ; index for number array
                sta     $52             ; flag that at least one number was
                                        ; entered

                lda     #<message       ; display message
                ldy     #>message
                jsr     $ab1e

inputloop:
                lda     #<prompt        ; display prompt
                ldy     #>prompt
                jsr     $ab1e

                lda     #<linebuf       ; read string into buffer
                ldy     #>linebuf
                ldx     #5
                jsr     readline

                lda     linebuf         ; empty line?
                beq     process         ; -> start processing

                lda     #<linebuf       ; convert input to int8
                ldy     #>linebuf
                jsr     toint8
                bcc     numok           ; successful -> store number
                lda     #<errmsg        ; else show error message and repeat
                ldy     #>errmsg
                jsr     $ab1e
                bcs     inputloop

numok:          ldx     #$ff            ; set flag that we have a number
                stx     $52
                ldx     $2a
                sta     numbers,x
                inc     $2a             ; next index
                bne     inputloop       ; if array not full, next input

process:        lda     $52             ; check we have some numbers
                beq     exit            ; otherwise exit program

                lda     #<numbers       ; address of array to $fb/fc
                sta     $fb
                lda     #>numbers
                sta     $fc
                lda     $2a             ; length of array to $fd
                sta     $fd
                jsr     oddfirst        ; call "sorting" function

                lda     #$0             ; index variable for output loop
                sta     $52
outloop:        ldy     $52             ; load current index
                lda     numbers,y       ; load current number
                jsr     printnum        ; -> output
                inc     $52             ; next index
                lda     $52             ; compare with ...
                cmp     $2a             ; ... array size
                bne     outloop         ; not reached yet -> repeat

exit:           rts                     ; done, exit program

; read a line of input from keyboard, terminate it with 0
; expects pointer to input buffer in A/Y, buffer length in X
.proc readline
                dex
                stx     $fb
                sta     $fc
                sty     $fd
                ldy     #$0
                sty     $cc             ; enable cursor blinking
                sty     $fe             ; temporary for loop variable
getkey:         jsr     $f142           ; get character from keyboard
                beq     getkey
                sta     $2              ; save to temporary
                and     #$7f
                cmp     #$20            ; check for control character
                bcs     checkout        ; no -> check buffer size
                cmp     #$d             ; was it enter/return?
                beq     prepout         ; -> normal flow
                cmp     #$14            ; was it backspace/delete?
                bne     getkey          ; if not, get next char
                lda     $fe             ; check current index
                beq     getkey          ; zero -> backspace not possible
                bne     prepout         ; skip checking buffer size for bs
checkout:       lda     $fe             ; buffer index
                cmp     $fb             ; check against buffer size
                beq     getkey          ; if it would overflow, loop again
prepout:        sei                     ; no interrupts
                ldy     $d3             ; get current screen column
                lda     ($d1),y         ; and clear 
                and     #$7f            ;   cursor in
                sta     ($d1),y         ;   current row
output:         lda     $2              ; load character
                jsr     $e716           ;   and output
                ldx     $cf             ; check cursor phase
                beq     store           ; invisible -> to store
                ldy     $d3             ; get current screen column
                lda     ($d1),y         ; and show
                ora     #$80            ;   cursor in
                sta     ($d1),y         ;   current row
                lda     $2              ; load character
store:          cli                     ; enable interrupts
                cmp     #$14            ; was it backspace/delete?
                beq     backspace       ; to backspace handling code
                cmp     #$d             ; was it enter/return?
                beq     done            ; then we're done.
                ldy     $fe             ; load buffer index
                sta     ($fc),y         ; store character in buffer
                iny                     ; advance buffer index
                sty     $fe
                bne     getkey          ; not zero -> ok
done:           lda     #$0             ; terminate string in buffer with zero
                ldy     $fe             ; get buffer index
                sta     ($fc),y         ; store terminator in buffer
                sei                     ; no interrupts
                ldy     $d3             ; get current screen column
                lda     ($d1),y         ; and clear 
                and     #$7f            ;   cursor in
                sta     ($d1),y         ;   current row
                inc     $cc             ; disable cursor blinking
                cli                     ; enable interrupts
                rts                     ; return
backspace:      dec     $fe             ; decrement buffer index
                bcs     getkey          ; and get next key
.endproc

; print an int8 number to the screen
; input:
;   A - the number to print
; clobbers:
;   X, Y
.proc printnum
                bpl     doprint         ; positive? -> direct number output
                eor     #$ff            ; else invert, 
                sta     $2              ; ...
                inc     $2              ; add one,
                lda     #'-'            ; output a minus sign
                jsr     $e716
                lda     $2
doprint:        tax                     ; number to X reg
                lda     #$0             ; set A to 0
                jsr     $bdcd           ; routine for uint16 in X/A output
                lda     #' '            
                jmp     $e716           ; and print a space
.endproc

; parse / convert int8 number using a BCD representation and double-dabble,
; handle negative numbers.
.proc toint8
                sta     $fb
                sty     $fc
                ldy     #$0
                sty     $fd
                sty     $fe
                sty     convbuf
                sty     convbuf+1
                sty     convbuf+2
scanloop:       lda     ($fb),y
                beq     copy
                iny
                cmp     #$20
                beq     scanloop
                cmp     #$2d
                beq     minus
                cmp     #$30
                bcc     error
                cmp     #$3a
                bcs     error
                inc     $fd
                bcc     scanloop
minus:          lda     $fd
                bne     error
                lda     $fe
                bne     error
                inc     $fe
                bne     scanloop
error:          sec
                rts
copy:           dey
                bmi     error
                ldx     #$2
copyloop:       lda     ($fb),y
                cmp     #$30
                bcc     copynext
                cmp     #$3a
                bcs     copynext
                sec
                sbc     #$30
                sta     convbuf,x
                dex
copynext:       dey
                bpl     copyloop
                lda     #$0
                sta     $fb
                ldx     #$8
loop:           lsr     convbuf
                lda     convbuf+1
                bcc     skipbit1
                ora     #$10
skipbit1:       lsr     a
                sta     convbuf+1
                lda     convbuf+2
                bcc     skipbit2
                ora     #$10
skipbit2:       lsr     a
                sta     convbuf+2
                ror     $fb
                dex
                beq     done
                lda     convbuf
                cmp     #$8
                bmi     nosub1
                sbc     #$3
                sta     convbuf
nosub1:         lda     convbuf+1
                cmp     #$8
                bmi     nosub2
                sbc     #$3
                sta     convbuf+1
nosub2:         lda     convbuf+2
                cmp     #$8
                bmi     loop
                sbc     #$3
                sta     convbuf+2
                bcs     loop
done:           lda     $fe
                beq     positive
                lda     #$ff
                eor     $fb
                sta     $fb
                inc     $fb
positive:       lda     $fb
                clc
                rts
.endproc


2

Clojure - 35 bytes

(defn o[c](sort(fn[p _](odd? p))c))

Ungolfed:

(defn oddsort [col]
  (sort (fn [p _] (odd? p)) col))

There is lots of room for improvement, for example you can submit an anonymous function which has a shorter creation syntax via #(...). Also you could give sort-by a try, although the submission already exists.
NikoNyrh

@NikoNyrh: tried a #() anonymous function but got an arity error as two parameters were passed but only on expected/used, and bringing %2 into it added more characters. Would be interested to see how this could be done.
Bob Jarvis - Reinstate Monica
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.