Является ли число двоичным-тяжелым?

58

Целое число является двоичным тяжелым, если его двоичное представление содержит больше 1s, чем 0s, игнорируя при этом ведущие нули. Например, 1 двоично-тяжелый, поскольку его двоичное представление просто 1, однако 4 не двоично-тяжелый, как его двоичное представление 100. В случае связи (например, 2 с двоичным представлением 10) число не считается двоичным.

Учитывая положительное целое число в качестве входных данных, выведите истинное значение, если оно двоично-тяжелое, и ложное значение, если это не так.

Testcases

Формат: input -> binary -> output

1          ->                                1 -> True
2          ->                               10 -> False
4          ->                              100 -> False
5          ->                              101 -> True
60         ->                           111100 -> True
316        ->                        100111100 -> True
632        ->                       1001111000 -> False
2147483647 ->  1111111111111111111111111111111 -> True
2147483648 -> 10000000000000000000000000000000 -> False

счет

Это поэтому побеждает меньше байтов на каждом языке

Skidsdev
источник
Что если мой язык не может обработать последний контрольный пример, потому что он находится за пределами того, что считается положительным целым числом?
musicman523
1
@ musicman523 afaik Стандартные правила ввода / вывода гласят, что вы должны принимать только числа, представляемые в формате чисел вашего языка. Обратите внимание, что «игра» с использованием чего-то вроде boolfuck считается стандартной
лазейкой
Имеет ли значение истинность / ложь значение или нам нужны два разных значения?
Эрик Outgolfer
@EriktheOutgolfer любое значение
Skidsdev
6
Ака A072600 , если это кому-нибудь поможет.
dcsohl

Ответы:

28

Машинный код x86, 15 14 байт

F3 0F B8 C1 0F BD D1 03 C0 42 2B D0 D6 C3

Это функция, использующая соглашение о вызовах __fastcall от Microsoft (первый и единственный параметр в ecx, возвращаемое значение в eax, вызываемый объект может закрывать edx), хотя ее можно легко изменить для других соглашений о вызовах, которые передают аргументы в регистрах.

Возвращает 255 как правдивую и 0 как ложную.

Он использует недокументированный (но широко поддерживаемый) код операции salc.

Разборка ниже:

;F3 0F B8 C1 
  popcnt eax, ecx ; Sets eax to number of bits set in ecx

;0F BD D1
  bsr edx, ecx    ; Sets edx to the index of the leading 1 bit of ecx

;03 C0
  add eax, eax

;42
  inc edx

;2B D0
  sub edx, eax

  ; At this point, 
  ;   edx = (index of highest bit set) + 1 - 2*(number of bits set)
  ; This is negative if and only if ecx was binary-heavy.

;D6
  salc           ; undocumented opcode. Sets al to 255 if carry flag 
                 ; is set, and to 0 otherwise. 

;C3
  ret

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

Спасибо Питеру Кордесу за предложение заменить lzcntна bsr.

Ж. Лю
источник
Приятно. Я дошел до очевидного popcntперед прокруткой вниз, чтобы посмотреть на ответы, но не думал о том, lzcntчтобы иметь дело только с существенными цифрами, как того требует вопрос.
Питер Кордес
Есть ли способ получить чистую экономию от использования bsrвместо lzcnt(ака rep bsr)? Вы должны использовать subвместо, leaтак как это дает вам 32-lzcnt. (Или оставляет dst неизмененным для src = 0 на всем существующем оборудовании Intel и AMD. AMD даже документирует это поведение, но Intel говорит, что не определено ... Во всяком случае, OP сказал положительно , что исключает 0.)
Питер Кордес
1
Я определенно думал в том же духе, что и @Peter, так как задача явно ограничивает входные значения целыми положительными числами. На самом деле, у меня было решение, составленное с использованием popcntи bsr, но это было 17 байт. Я думал, что это было довольно хорошо по сравнению с первым ответом на вопрос, который я увидел , но этот хитрый leaтрюк превосходит его. Я тоже посмотрел на сравнение bsfи popcnt. Но я не вижу, как это решение можно превзойти, даже принимая во внимание 1 байт, который вы могли бы сохранить, отбросив repпрефикс.
Коди Грей,
1
salcне эквивалентно , чтобы setc al: последний устанавливает alв 1 , если CF установлен, чтобы не 255.
Руслан
1
Фактический эквивалент salcравен sbb al, al, но вы получаете экономию в 1 байт для его кодирования. Кстати, это будет документировано AMD, и это широко поддерживается Intel, с мнемоническим даже наступающим от карты от Intel P6 опкода. Так что это на самом деле довольно безопасно для использования. Кроме того, хорошее улучшение здесь, чтобы думать, чтобы использовать эту инструкцию! Это в основном то, что сделал мой первоначальный черновик, за исключением того, что (1) я использовал код x86-64, поэтому incдля кодирования было вдвое больше времени, и (2) я не думал salc, поэтому выполнял ту же работу в более длинный путь Жаль, что я могу проголосовать только один раз.
Коди Грей
17

Желе , 5 байт

Bo-SR

Выдает непустой вывод (истинный) или пустой вывод (ложный).

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

Как это устроено

Bo-SR  Main link. Argument: n

B      Binary; convert n to base 2.
 o-    Compute the logical OR with -1, mapping 1 -> 1 and 0 -> -1.
   S   Take the sum s. We need to check if the sum is strictly positive.
    R  Range; yield [1, ..., s], which is non-empty iff s > 0.
Деннис
источник
Приятно. Я имел Bo-S, но я не мог найти 1-байтовый атом, который преобразовал бы положительное / не положительное в правдивое / ложное ...
ETHproductions
Логично или с -1, верно?
Линн
@ Линн Да, действительно. Благодарю.
Деннис
1
4 байта
caird coinheringaahing
@cairdcoinheringaahing Спасибо, но тогда Æṃне существовало.
Деннис
14

Python 2 , 35 байт

lambda n:max('10',key=bin(n).count)

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

Старый ответ, 38 байт

Выводы 0как ложные и / -2или -1правдивые

lambda n:~cmp(*map(bin(n).count,'10'))

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

прут
источник
2
Приводит ли 0 к возврату binпричины этого решения проблемы?
Тень
3
@shadow Нет проблем, из-за способа maxработы. В случае привязки max вернет первое значение в итерируемом, которое имеет максимальное значение. Этот код использует этот факт, чтобы убедиться, что 1 возвращается в случае связи, что фактически означает, что существует больше единиц, чем нулей, так как дополнительный ноль был добавлен bin. Это будет на самом деле неправильно, если написано таким образом, если бы не дополнительный ноль.
FryAmTheEggman
@FryAmTheEggman это также верно для старого ответа, где cmpвозвращается, 0когда они оба равны
Род
11

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

@(n)mode(de2bi(n))

TIO не работает, так как инструментарий связи не включен. Это можно проверить на Octave-Online .

Как это работает:

de2biпреобразует десятичное число в двоичный числовой вектор, а не в строку, как это dec2binделает.

modeвозвращает наиболее частую цифру в векторе. По умолчанию это самый низкий уровень в случае ничьей.

@(n)                % Anonymous function that takes a decimal number as input 'n'
    mode(        )  % Computes the most frequent digit in the vector inside the parentheses
         de2bi(n)   % Converts the number 'n' to a binary vector
Стьюи Гриффин
источник
Является ли набор инструментов для коммуникации стандартной частью Octave, или это больше похоже на библиотеку на других языках?
dcsohl
Это пакет, который поставляется с установкой. Вы должны специально загружать его в некоторых установках, и он автоматически загружается как стандарт в других. Это часть стандарта на Octave-Online.net, поэтому я использую это в качестве ссылки. (Код должен работать хотя бы в одном интерпретаторе, существовавшем до вызова).
Стьюи Гриффин
9

JavaScript (ES6), 36 34 байта

f=(n,x=0)=>n?f(n>>>1,x+n%2-.5):x>0
ETHproductions
источник
f=(n,x=0)=>n?f(n>>>1,x+=n%2-.5):x>0для 35 байтов.
овс
Используйте n>>1вместо того, n>>>1чтобы сохранить байт, так как ввод никогда не бывает отрицательным.
kamoroso94
@ kamoroso94 Спасибо, но тогда это не получится 2147483648.
ETHproductions
@ETHproductions Черт, и n/2|0не лучше: /
kamoroso94
8

Mathematica, 22 байта

Сохранено один байт благодаря @MartinEnder и @JungHwanMin .

#>#2&@@#~DigitCount~2&
alephalpha
источник
2
Я думаю, что инфиксная запись имеет более высокий приоритет, чем @@.
Мартин Эндер
1
-1 байт (как заметил @MartinEnder):#>#2&@@#~DigitCount~2&
JungHwan Мин
7

Брахилог , 6 байт

ḃọtᵐ>₁

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

объяснение

Example input: 13

ḃ        Base (default: binary): [1,1,0,1]
 ọ       Occurences:             [[1,3],[0,1]]
  tᵐ     Map Tail:               [3,1]
    >₁   Strictly decreasing list

Поскольку никогда не объединять свой вывод со списком цифр с ведущими нулями, мы знаем, что вхождения 1всегда будут первыми, а вхождения 0всегда будут вторыми после .

Fatalize
источник
6

C (gcc) , 51 48 41 40 байт

i;f(n){for(i=0;n;n/=2)i+=n%2*2-1;n=i>0;}

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

cleblanc
источник
На основании разъяснений ОП, вы можете удалитьunsigned
musicman523
Поскольку nnn положительно, вы можете изменить n>>=1на n/=2. Я также думаю, что вы можете использовать ~nвместо n^-1, что также должно позволить вам перейти &&на&
musicman523
Странные вещи случаются, когда я редактирую комментарии - «nnn» означает n, и не берите в голову изменение &&на &, я не думаю, что это будет работать. Но, *похоже, это работает
musicman523
@ musicman523 &&Только для обработки беззнакового случая, но так как мне нужно обрабатывать только положительные целые числа, я могу удалить все это вместе. Хороший pouint о /=том, что короче, >>=хотя, спасибо!
cleblanc
Вы можете сохранить один байт, изменив n&1?++i:--1на i+=n%2*2-1. Вы также можете избавиться от >0этого, заявив, что вы получите ноль для тяжелого и ненулевой для не тяжелого
musicman523
6

R , 54 53 51 байт

-1 байт благодаря Max Lawnboy

n=scan();d=floor(log2(n))+1;sum(n%/%2^(0:d)%%2)*2>d

читает со стандартного ввода; возвращает TRUEдвоичные тяжелые числа. dколичество двоичных цифр; sum(n%/%2^(0:d)%%2вычисляет сумму цифр (т. е. количество единиц).

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

Giuseppe
источник
Ваш ответ увидели только после публикации моего сообщения ... В любом случае, вы можете использовать log2(n)вместо того, log(n,2)чтобы сохранить 1 байт
Максим Михайлов
@MaxLawnboy ах, конечно. Спасибо!
Джузеппе
Golfed от еще 12 байт: codegolf.stackexchange.com/a/132396/59530
JAD
6

машинный код x86_64, 23 22 21 байт

31 c0 89 fa 83 e2 01 8d 44 50 ff d1 ef 75 f3 f7 d8 c1 e8 1f c3

Разобрал:

  # zero out eax
  xor  %eax, %eax
Loop:
  # copy input to edx
  mov  %edi, %edx
  # extract LSB(edx)
  and  $0x1, %edx
  # increment(1)/decrement(0) eax depending on that bit
  lea -1(%rax,%rdx,2), %eax
  # input >>= 1
  shr  %edi
  # if input != 0: repeat from Loop
  jnz  Loop

  # now `eax < 0` iff the input was not binary heavy,
  neg %eax
  # now `eax < 0` iff the input was binary heavy (which means the MSB is `1`)
  # set return value to MSB(eax)
  shr  $31, %eax
  ret

Спасибо @Ruslan, @PeterCordes за -1байт!

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

ბიმო
источник
Есть ли какая-то конкретная причина, почему вы используете 8d 1fвместо 89 fb?
Руслан
2
На самом деле вопрос в том, есть ли какая-то особая причина, по которой вы используете этот отвратительный синтаксис AT & T?!? Кроме того, разборка и ваша разборка согласны, что у вас есть add eax, 2+ dec eax, но ваши комментарии предполагают, что вы хотите увеличить ebx, а не eax.
Коди Грей,
1
Вы можете заменить jnz Next/ add/ dec(7 байтов) на lea -1(%rax, %rbx, 2), %eax(4 байта), чтобы сделать eax += 2*ebx - 1(как в другом ответе машинного кода x86 ). Затем за пределами цикла neg %eax(2 байта), прежде чем сдвинуть бит знака вниз. Чистая экономия 1 байта. Или test %eax,%eax/ setge %alтакже будет работать, если ваше возвращаемое значение равно boolили int8_t.
Питер Кордес
1
@PeterCordes Я думаю, я знаю, что произошло, но я не уверен: я мог бы не пытаться, lea -1(%rax,rbx,2)а только lea -1(%eax,%eax,2)и тратить байты таким образом ... В любом случае, вы оба были правы, я могу сохранить такой байт, как этот. Большое спасибо (взамен я изменю leaэто, movпока я в этом)!
მოიმო
1
@ moonheart08: Я не знал об этом тогда, но кто-то опубликовал ответ, который сэкономил 7 байтов.
მოიმო
5

Perl 6 ,  32  30 байт

{[>] .base(2).comb.Bag{qw<1 0>}}

Проверь это

{[>] .polymod(2 xx*).Bag{1,0}}

Проверь это

Expanded:

{      # bare block lambda with implicit parameter 「$_」

  [>]  # reduce the following with &infix:« > »

    .polymod(2 xx *) # turn into base 2 (reversed) (implicit method call on 「$_」)
    .Bag\            # put into a weighted Set
    { 1, 0 }         # key into that with 1 and 0
                     # (returns 2 element list that [>] will reduce)
}
Брэд Гилберт b2gills
источник
5

Мудрый , 40 39 байт

::^?[:::^~-&[-~!-~-~?]!~-?|>]|:[>-?>?]|

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

объяснение

::^?                                      Put a zero on the bottom
    [                                     While
     :::^~-&                              Get the last bit
            [-~!-~-~?]!~-?|               Increment counter if 0 decrement if 1
                           >              Remove the last bit
                            ]|            End while
                              :[>-?>?]|   Get the sign
Мастер пшеницы
источник
5

Хаскелл, 41 34

g 0=0
g n=g(div n 2)+(-1)^n
(<0).g

Если nэто нечетно, возьмите, -1если это четное, возьмите 1. Добавить рекурсивный вызов с помощью n/2и прекратить, если n = 0. Если результат меньше, чем 0число является двоичным тяжелым.

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

Редактировать: @ Ørjan Йохансен нашел несколько ярлыков и сохранил 7 байтов. Спасибо!

Ними
источник
mod n 2может быть просто n, и это меньше байта без аккумулятора. Попробуйте онлайн!
Эрджан Йохансен
5

Сетчатка , 37 34 байта

.+
$*
+`(1+)\1
$1@
@1
1
+`.\b.

1+

Попробуйте онлайн! Ссылка включает в себя меньшие тестовые случаи (более крупные, вероятно, не хватило бы памяти). Редактировать: 3 байта сохранены благодаря @MartinEnder. Объяснение: Первый этап преобразуется из десятичного в унарный, а следующие два этапа преобразуются из унарного в двоичный (это почти прямо из унарной арифметической страницы в вики Retina, за исключением того, что я использую @вместо 0). Третий этап ищет пары разнородных символов, которые могут быть как @1или 1@, так и удаляет их, пока не останется ни одного. Последний этап затем проверяет оставшиеся 1 с.

Нил
источник
${1}может быть $+. Или вы можете использовать !вместо, 0а затем сократить 01|10до .\b..
Мартин Эндер
@MartinEnder Да, делает $+ли правильно, когда шаблон содержит |? Интересно, мог бы я использовать это раньше ...
Нил
2
нет, $+это супер глупо и просто использует группу с наибольшим номером, независимо от того, использовалась она или нет. Это полезно только для игры в гольф, когда у вас более девяти групп или в ситуации, подобной той, что здесь, и я не знаю, почему я когда-либо использовал это в регулярных выражениях.
Мартин Эндер
5

R , 43 байта

max(which(B<-intToBits(scan())>0))/2<sum(B)

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

             intToBits(scan())              # converts to bits
          B<-                 >0            # make logical and assign to B
max(which(                      ))/2        # get the length of the trimmed binary and halve
                                    <sum(B) # test against the sum of bits
MickyT
источник
+1 аккуратное решение! Я даже не знал оintToBits
Джузеппе
Golfed от другого 4 байта: codegolf.stackexchange.com/a/132396/59530
JAD
5

Котлин , 50 байтов

{i:Int->i.toString(2).run{count{it>'0'}>length/2}}

Лямбда неявного типа (Int) -> Boolean. Версия 1.1 и выше только за счет использования Int.toString(radix: Int).

К сожалению, среда выполнения Kotlin в TIO выглядит как 1.0.x, так что вместо ссылки TIO здесь есть грустная собака:

Ф. Джордж
источник
4

Pyth, 9 7 байт

ehc2S.B

Попробуй это здесь.

-2 благодаря FryAmTheEggman .

Эрик Outgolfer
источник
Еще один 9-байтовый подход:>ysJjQ2lJ
KarlKastor
1
7 байт, но я чувствую, что все еще должно быть что-то более короткое ...
FryAmTheEggman
@FryAmTheEggman Хм ... это может работать только как полная программа. (Я знал, что есть способ использовать .B!)
Эрик Outgolfer
4

R, 39 37 байт

sum(intToBits(x<-scan())>0)>2+log2(x)

Это комбинация методов, используемых @MickyT и @Giuseppe, сохраняя еще несколько байтов.

sum(intToBits(x) > 0)подсчитывает количество 1бит и 2+log2(x)/2составляет половину от общего количества бит при округлении в меньшую сторону. Нам не нужно округлять из-за поведения, когда два значения равны.

JAD
источник
4

C # (.NET Core) , 62 , 49 байтов

Без LINQ.

РЕДАКТИРОВАТЬ: дана с -13 байтовым гольфом, изменяющим время на рекурсивный для и возвращающим бул вместо целого числа.

x=>{int j=0;for(;x>0;x/=2)j+=x%2*2-1;return j>0;}

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

Destroigo
источник
4

Регулярное выражение (ECMAScript), 85 73 71 байт

^((?=(x*?)\2(\2{4})+$|(x*?)(\4\4xx)*$)(\2\4|(x*)\5\7\7(?=\4\7$\2)\B))*$

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

объяснение по Deadcode

Более ранняя 73-байтовая версия описана ниже.

^((?=(x*?)\2(\2{4})+$)\2|(?=(x*?)(\4\4xx)*$)(\4|\5(x*)\7\7(?=\4\7$)\B))+$

Из-за ограничений регулярного выражения ECMAScript эффективная тактика часто заключается в преобразовании числа на один шаг за раз, сохраняя при этом обязательное свойство неизменным на каждом этапе. Например, чтобы проверить идеальный квадрат или степень 2, уменьшите число в размерах, сохраняя при этом квадрат или степень 2 (соответственно) на каждом шаге.

Вот что делает это решение на каждом этапе:

111100101ones>zeroes1

ones>zeroesones1>zeroes1

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

И конечно, хотя эти шаги описаны выше в терминах типографских манипуляций над двоичным представлением числа, они фактически реализованы как унарная арифметика.

# For these comments, N = the number to the right of the "cursor", a.k.a. "tail",
# and "rightmost" refers to the big-endian binary representation of N.
^
(                          # if N is even and not a power of 2:
    (?=(x*?)\2(\2{4})+$)   # \2 = smallest divisor of N/2 such that the quotient is
                           # odd and greater than 1; as such, it is guaranteed to be
                           # the largest power of 2 that divides N/2, iff N is not
                           # itself a power of 2 (using "+" instead of "*" is what
                           # prevents a match if N is a power of 2).
    \2                     # N = N - \2. This changes the rightmost "10" to a "01".
|                          # else (N is odd or a power of 2)
    (?=(x*?)(\4\4xx)*$)    # \4+1 = smallest divisor of N+1 such that the quotient is
                           # odd; as such, \4+1 is guaranteed to be the largest power
                           # of 2 that divides N+1. So, iff N is even, \4 will be 0.
                           # Another way of saying this: \4 = the string of
                           # contiguous 1 bits from the rightmost part of N.
                           # \5 = (\4+1) * 2 iff N+1 is not a power of 2, else
                           # \5 = unset (NPCG) (iff N+1 is a power of 2), but since
                           #   N==\4 iff this is the case, the loop will exit
                           #   immediately anyway, so an unset \5 will never be used.
    (
        \4                 # N = N - \4. If N==\4 before this, it was all 1 bits and
                           # therefore heavy, so the loop will exit and match. This
                           # would work as "\4$", and leaving out the "$" is a golf
                           # optimization. It still works without the "$" because if
                           # N is no longer heavy after having \4 subtracted from it,
                           # this will eventually result in a non-match which will
                           # then backtrack to a point where N was still heavy, at
                           # which point the following alternative will be tried.
    |
        # N = (N + \4 - 2) / 4. This removes the rightmost "01". As such, it removes
        # an equal number of 0 bits and 1 bits (one of each) and the heaviness of N
        # is invariant before and after. This fails to match if N is a power of 2,
        # and in fact causes the loop to reach a dead end in that case.
        \5                 # N = N - (\4+1)*2
        (x*)\7\7(?=\4\7$)  # N = (N - \4) / 4 + \4
        \B                 # Assert N > 0 (this would be the same as asserting N > 2
                           # before the above N = (N + \4 - 2) / 4 operation).
    )
)+
$       # This can only be a match if the loop was exited due to N==\4.
Grimmy
источник
2
Хотя это вдохновлено ответом Deadcode , алгоритм достаточно отличается, так что я чувствовал, что заслуживает отдельного ответа, а не комментария.
Grimmy
2
Это феноменально, и именно это я и хотел увидеть (кто-то выдувает мое регулярное выражение из воды с помощью гораздо более лаконичного алгоритма). Но ваши комментарии на самом деле не объясняют это вообще, и прокомментированная 73-байтовая версия регулярного выражения даже не работает (обратные ссылки \5на одно отключены). Я изучил это, объяснил и прокомментировал это в своем ответе (потому что StackExchange не разрешает многострочные ответы).
Deadcode
4

Регулярное выражение (ECMAScript), 183 байта

Это была еще одна интересная проблема, которую нужно решить с помощью регулярного выражения ECMA. «Очевидный» способ справиться с этим - подсчитать количество 1битов и сравнить их с общим количеством битов. Но вы не можете напрямую сосчитать вещи в регулярном выражении ECMAScript - отсутствие постоянных обратных ссылок означает, что в цикле можно изменить только одно число, и на каждом шаге его можно только уменьшить.

Этот унарный алгоритм работает следующим образом:

  1. Возьмите квадратный корень из наибольшей степени 2, который вписывается в N, и обратите внимание на то, был ли квадратный корень идеальным или его нужно было округлить в меньшую сторону. Это будет использовано позже.
  2. В цикле переместите каждый старший значащий 1бит в наименее значимое положение, где есть 0бит. Каждый из этих шагов является вычитанием. В конце цикла оставшееся число (как оно будет представлено в двоичном виде) представляет собой строку 1s без 0s. Эти операции фактически выполняются в унарном порядке; это только концептуально, что они делаются в двоичном формате.
  3. Сравните эту "двоичную строку 1s" с квадратным корнем, полученным ранее. Если квадратный корень нужно было округлить в меньшую сторону, используйте двойную версию. Это гарантирует, что «двоичная строка 1s» должна иметь более половины числа двоичных цифр, чем N, чтобы иметь место окончательное совпадение.

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

Без дальнейших церемоний, регулярное выражение:

^(?=.*?(?!(x(xx)+)\1*$)(x)*?(x(x*))(?=(\4*)\5+$)\4*$\6)(?=(((?=(x(x+)(?=\10$))*(x*))(?!.*$\11)(?=(x*)(?=(x\12)*$)(?=\11+$)\11\12+$)(?=.*?(?!(x(xx)+)\14*$)\13(x*))\16)*))\7\4(.*$\3|\4)

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

# For the purposes of these comments, the input number = N.
^
# Take the floor square root of N
(?=
    .*?
    (?!(x(xx)+)\1*$)    # tail = the largest power of 2 less than tail
    (x)*?               # \3 = nonzero if we will need to round this square root
                        #      up to the next power of two
    (x(x*))             # \4 = potential square root; \5 = \4 - 1
    (?=
        (\4*)\5+$       # Iff \4*\4 == our number, then the first match here must result in \6==0
    )
    \4*$\6              # Test for divisibility by \4 and for \6==0 simultaneously
)
# Move all binary bits to be as least-significant as possible, e.g. 11001001 -> 1111
(?=
    (                                 # \7 = tool for making tail = the result of this move
        (
            (?=
                (x(x+)(?=\10$))*(x*)  # \11 = {divisor for getting the least-significant 0 bit}-1
            )
            (?!.*$\11)                # Exit the loop when \11==0
            (?=
                (x*)                  # \12 = floor((tail+1) / (\11+1)) - 1
                (?=(x\12)*$)          # \13 = \12+1
                (?=\11+$)
                \11\12+$
            )
            (?=
                .*?
                (?!(x(xx)+)\14*$)     # tail = the largest power of 2 less than tail
                \13                   # tail -= \13
                (x*)                  # \16 = tool to move the most-significant 1 bit to the
                                      # least-significant 0 bit available spot for it
            )
            \16
        )*
    )
)
\7                  # tail = the result of the move
\4                  # Assert that \4 is less than or equal to the result of the move
(
    .*$\3
|
    \4              # Double the value of \4 to compare against if \3 is non-empty,
                    # i.e. if we had an even number of total digits.
)
Deadcode
источник
1
-98 байт
Grimmy
3

Желе , 6 байт

Bµċ0<S

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

Эрик Outgolfer
источник
Bo-Sможет использоваться для вычисления двоичного «веса» входных данных, к сожалению, самый короткий способ использования, который, по-видимому, Bo-S>0...
ETHproductions
@ETHproductions Да, пока нет "положительного" атома.
Эрик Outgolfer
Ну видимо работает: P
ETHproductions
3

J , 12 байт

(+/>-:@#)@#:

J выполняет глаголы справа налево, поэтому давайте начнем с конца и продолжим путь к началу.

объяснение

         #:       NB. Convert input to list of bits
       -:@#       NB. Half (-:) the (@) length (#)
          >       NB. Greater than 
         +/       NB. Sum (really plus (+) reduce (/)
Дэн Брон
источник
1
(#<2*+/)@#:должен сохранить 1, если я что-то упустил.
FrownyFrog
2

Python 2 , 44 байта

f=lambda n,c=0:f(n/2,c+n%2*2-1)if n else c>0

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

Старый ответ, 47 байт

c,n=0,input()
while n:c+=n%2*2-1;n/=2
print c>0

Это просто порт ответа C cleblanc . Это длиннее, чем другие ответы Python, но я решил, что это стоит опубликовать, поскольку это совершенно другой метод поиска ответа.

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

musicman523
источник
2

C #, 82 байта

n=>{var s=System.Convert.ToString(n,2);return s.Replace("0","").Length>s.Length/2}
TheLethalCoder
источник
Вы можете обрезать еще немного, рассматривая строку как IEnumerable <char>. n=>{var s=Convert.ToString(n,2);return s.Count(c=>c=='1')>s.Length/2;}
GalacticCowboy
@GalacticCowboy Это добавляет 11 байтов, потому что вы должны полностью квалифицировать Convertи включить using System.Linq;(написано короче как namespace System.Linq{}). Хорошая идея просто не бреет достаточно, чтобы оправдать экономию в этом случае.
TheLethalCoder