Обчисліть найдовший ряд 1 у цілому двійковому значенні


32

Мета

Враховуючи невід'ємне ціле число, створіть функцію, яка повертає вихідну позицію числа найбільших послідовних 1-х у бінарному значенні цього цілого числа.

Коли вам дано вхід 0, поверніться 0.

Якщо число має кілька смуг однакової довжини, ви повинні повернути положення останньої смуги.

Вхідні дані

Ціле число, що перевищує або дорівнює 0.

Вихід

Ціле число, обчислене, як пояснено нижче.

Правила

  • Це код-гольф, тому виграє найкоротший код у байтах на кожній мові.
  • Стандартні лазівки заборонені.

Приклади та тестові випадки

Приклад 1

  • Ваша функція передається цілим числом 142
  • 142 дорівнює 10001110 у двійкових
  • Найдовша смуга - "111" (смуга з трьох)
  • Рядок починається з положення 2 ^ 1
  • Ваша функція повертає 1 як результат

Приклад 2

  • Ваша функція передається цілим числом 48
  • 48 дорівнює 110000 у двійкових
  • Найдовша смуга - "11" (смуга з двох)
  • Полоска починається з положення 2 ^ 4
  • Ваша функція повертає 4 як результат

Приклад 3

  • Ваша функція передається цілим числом 750
  • 750 дорівнює 1011101110 у двійковій формі
  • Найдовша смуга - "111" (смуга з трьох)
  • Оскільки є дві смуги однакової довжини, ми повертаємо більш пізню смугу.
  • Пізніша смуга починається з положення 2 ^ 5
  • Ваша функція повертає 5 як результат

1
Вам потрібен критерій виграшу, наприклад код-гольф
Окс

@Okx Про це згадувалося в самому тілі, тому я додав тег.
повністюлюдський

Переконайтесь, що люди перевіряють 0. Це важливий тестовий випадок.
mbomb007

2
Замість "останньої смуги" або "останньої смуги", я б запропонував "смугу з найбільшою цінністю місця".
aschepler

@Okx Чому необхідний критерій виграшу? Чому це не може бути просто головоломка?
corsiKa

Відповіді:


21

Желе , 10 8 байт

Ba\ÐƤṀċ¬

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

Як це працює

Ba\ÐƤṀċ¬  Main link. Argument: n


B         Binary; convert n to base 2.

   ÐƤ     Apply the link to the left to all postfixes of the binary array.
 a\         Cumulatively reduce by logical AND.

          For example, the array

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

          becomes the following array of arrays.

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

     Ṁ    Take the lexicographical maximum, i.e., the array with the most 1's at
          the beginning, breaking ties by length.

       ¬  Yield the logical NOT of n, i.e., 0 if n > 0 and 1 if n = 0.

      ċ   Count the occurrences of the result to the right in the one to the left.

Вітаємо!
Дефакто

24

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

Збережено 4 байти завдяки @ThePirateBay

f=x=>(k=x&x/2)?f(k):Math.log2(x)|0

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

Як?

Загальний випадок x> 0

Ми рекурсивно І вхід x з x / 2, який поступово зменшує шаблони послідовних наборів бітів, поки не залишиться лише крайній правий біт послідовності. Ми зберігаємо копію останнього ненульового значення і визначаємо положення його найбільш значущого біта шляхом округлення підлоги логарифмом бази-2.

Нижче наведено кроки, які ми проходимо для x = 750 ( 1011101110 у двійковій).

    1011101110 --.
,----------------'
'-> 1011101110
AND 0101110111
    ----------
=   0001100110 --.
,----------------'
'-> 0001100110
AND 0000110011
    ----------
=   0000100010 --.  --> output = position of MSB = 5  (0000100010)
,----------------'                                         ^
'-> 0000100010
AND 0000010001
    ----------
=   0000000000

Випадок краю x = 0

Особливий випадок x = 0негайно призводить до Math.log2(0) | 0. Логарифм 0обчислювачів до -Infinityта двійкового бітового АБО примушує примусити до 32-бітного цілого числа. Відповідно до специфікації абстрактної операції ToInt32 , це дає очікувані 0:

Якщо число є NaN , + 0 , -0 , + ∞ або -∞ , повернути +0


Це помилки для вводу 0, який є частиною діапазону введення.
Джастін Марінер

Виправлено @JustinMariner.
Арнольд

А як щодо Math.log2(k)|0цього 31-Math.clz32(k)? Або я щось пропускаю?

@ThePirateBay Math.log2(k)|0насправді як коротший, так і простіший у спеціальному випадку x=0. Спасибі. :)
Арнольд

1
Дуже приємний алгоритм та гарне пояснення. Я реалізував його в 14 байтах машинного коду x86 .
Пітер Кордес

12

машинний код x86, 14 байт

Використання @ алгоритму Arnauld в частині x &= x>>1і беруть найвищу бітову позицію , встановлену в попередньому кроці xстає 0.

Телефонує з C / C ++ з підписом unsigned longest_set(unsigned edi);у x86-64 System V ABI. Ті самі байти машинного коду будуть декодувати однаково в 32-бітному режимі, але стандартні 32-бітні конвенції виклику не ставлять перший аргумент edi. (Зміна вхідного регістра на Windows ecxабо edxдля нього _fastcall/ _vectorcallабо gcc -mregparmможна зробити, не порушуючи нічого.)

   address   machine-code
             bytes
                         global longest_set
 1                       longest_set:
 2 00000000 31C0             xor  eax, eax    ; 0 for input = 0
 3                       
 4                       .loop:               ; do{
 5 00000002 0FBDC7           bsr  eax, edi    ;  eax = position of highest set bit
 6                           ;; bsr leaves output unmodified when input = 0 (and sets ZF)
 7                           ;; AMD documents this; Intel manuals say unspecified but Intel CPUs implement it
 8 00000005 89F9             mov  ecx, edi
 9 00000007 D1E9             shr  ecx, 1
10 00000009 21CF             and  edi, ecx
11 0000000B 75F5             jnz .loop        ; } while (x &= x>>1);
12                       
13 0000000D C3               ret

BSRІнструкція x86 (Bit Scan Reverse) ідеально підходить для цього, даючи нам бітовий індекс безпосередньо, а не рахуючи провідні нулі. ( bsrне створює безпосередньо 0 для введення = 0, як 32-lzcnt(x)хотілося б, але нам потрібні bsr = 31-lzcnt для ненульових входів, тому lzcntнавіть не збережеш інструкції, не кажучи вже про кількість підрахунків байтів. Нульовий еакс перед циклом працює через bsr' s напівофіційна поведінка залишати пункт призначення незмінним, коли вхід дорівнює нулю.)

Це було б ще коротше, якби ми могли повернути позицію MSB в найдовшій перспективі. У цьому випадку lea ecx, [rdi+rdi](3 байти) буде скопійовано + вліво- зміщення замість mov+ shr(4 байти).

Перегляньте це посилання TIO для того, хто викликає поштовхexit(longest_set(argc-1));

Тестування з петлею оболонки:

l=(); for ((i=0;i<1025;i++));do 
    ./longest-set-bitrun "${l[@]}";   # number of args = $i
    printf "$i %#x: $?\n" $i; 
    l+=($i); 
done | m

0 0: 0
1 0x1: 0
2 0x2: 1
3 0x3: 0
4 0x4: 2
5 0x5: 2
6 0x6: 1
7 0x7: 0
8 0x8: 3
9 0x9: 3
10 0xa: 3
11 0xb: 0
12 0xc: 2
13 0xd: 2
14 0xe: 1
15 0xf: 0
16 0x10: 4
17 0x11: 4
18 0x12: 4
19 0x13: 0
20 0x14: 4
21 0x15: 4

...

747 0x2eb: 5
748 0x2ec: 5
749 0x2ed: 5
750 0x2ee: 5
751 0x2ef: 0
752 0x2f0: 4
753 0x2f1: 4
754 0x2f2: 4

1
Приємно! Примітка для тих, хто (як я) більше знайомий з іншими діалектними мовленнями: мнемонічна x86 BSR означає "Зворотне сканування бітів", а не "Відгалуження до SubRoutine".
Арнольд

@Arnauld: хороший момент, відредагований для розшифровки мнемоніки, а також посилання на ручний запис із посиланням на Insn.
Пітер Кордес

5

Желе , 19 17 11 байт

HÐĿ&\ḟ0ṪBL’

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

-6 (!) Байтів завдяки пильним спостереженням @ Денніса

Як це працює

HÐĿ&\ḟ0ṪBL’
HÐĿ         - halve repeatedly until reaching 0 due to rounding
   &\       - cumulative AND
     ḟ0Ṫ    - final non-zero, or 0 if all elements are 0
        BL  - length of binary representation (log base 2). 0->[0]->1
          ’ - decrement

Помилки для 0, яка знаходиться в діапазоні введення.
Містер Xcoder

@ Mr.Xcoder Виправлено
fireflame241

Ви можете зберегти байт для виправлення за допомогоюȯ-µ...
Джонатан Аллан

@JonathanAllan Я не розумію, як допоможе АБО. У вас є код?
fireflame241

1
Оскільки Bзавжди повертає масив, який починається з 1 , BUṪMṪ’×Ṡможе стати ṪBL’. Крім того, вам не потрібно і ḟ0зберігає один байт ¹Ðf.
Денніс

5

Python 2 , 45 байт

f=lambda x:f(x&x/2)if x&x/2else len(bin(x))-3

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

Збережено багато байт завдяки Деннісу! (Головами len(bin(...))-3замість math.frexp)

Дякуємо @xnor за те, що знайшов помилку, яку, на щастя, було легко виправити!


This seems to give wrong answers such as for x=3, I think because the and/or short-circuits wrong when the function returns 0.
xnor

@xnor Thanks for noticing that! It's fixed.
Mr. Xcoder

4

Perl 6, 45 35 bytes

This highly improved version is courtesy of @nwellnhof.

{.msb+1-(.base(2)~~m:g/1+/).max.to}

Try it online!


You can simply match with :g to get all matches. With the smart-match operator, I could golf it down to 40 bytes: {sort(.base(2).flip~~m:g/1+/).tail.from}
nwellnhof


@nwellnhof, thank you very much. I somehow missed the :g and didn't figure out how I can use adverb with the smartmatch operator. Also the .msb method is quite helpful here and I didn't know about it before.
Ramillies

3

Python 2, 89 78 bytes

m=t=i=j=0
for c in bin(input()):
 t=-~t*(c>'0');i+=1
 if t>m:j=i;m=t
print i-j

Try it online!

EDIT: Saved 11 bytes thanks to Mr. Xcoder.




@Mr.Xcoder I also thought of that, though the question specifically asks to write a function.
Jonathan Frech

@JonathanFrech I don't think the OP really meant function. Probably just a generic term.
Mr. Xcoder

@Mr.Xcoder Please explain this. Thanks,
lifebalance

3

05AB1E, 14 12 11 bytes

bDγàŠrkrJgα

Try it online! or run test cases.

Explanation

bDγàŠrkrJgα  Implicit input (ex: 750)
bD           Convert input to binary string and duplicate
                 '1011101110', '1011101110'
  γ          Split into groups of 1s or 0s
                 '1011101110', ['1', '0', '111', '0', '111', '0']
   à         Pull out max, parsing each as an integer
                 '1011101110', ['1', '0', '0', '111', '0'], '111'
    Šrk      Rearrange stack and get first index of max run of ones
                 ['1', '0', '0', '111', '0'], 2
       rJg   Reverse stack, join the array to a string, and get its length
                 2, 7
          α  Get absolute difference
                 5

3

J, 18 17 bytes

(#-0{#\\:#.~\)@#:

Try it online!

Explanation

(#-0{#\\:#.~\)@#:  Input: integer n
               #:  Binary
     #\            Length of each prefix
       \:          Sorted down using
         #.~\      Mixed base conversion on each prefix
   0{              Get the value at index 0
  -                Subtract from
 #                 Length

Very slick. Not sure I fully understand what's going on with the mixed base conversion though. Why does it count the number of consecutive ones at the end of the string? Also, if the bases being specified on the x of the dyad are all 0 and 1, what does that mean? A base 2 number has digits 0 and 1, so a base 1 number has digits... 0? then what does a 1 mean in that context? And is a base 0 number just always 0?
Jonah

@Jonah It's more due to how mixed base conversion is implemented. x #. y first computes w =: */\. }. x , 1 then returns +/ w * y
miles

so would you consider this a golf hack rather than a legitimate use of #., since you are relying on internal implementation details rather than the public api?
Jonah

3

C# (.NET Core), 64 60 bytes

T=a=>(a&a/2)>0?T(a&a/2):Math.Log(a,2)<0?0:(int)Math.Log(a,2)

Try it online!

A C# version of @Arnauld's answer

Acknowledgements

4 bytes saved thanks to Kevin Cruijssen.

C# (.NET Core), 131 123+18=141 bytes

a=>{string s=Convert.ToString(a,2),t=s.Split('0').OrderBy(x=>x.Length).Last();return a<1?0:s.Length-s.IndexOf(t)-t.Length;}

Try it online!

+18 bytes for using System.Linq;

Acknowledgements

8 bytes saved thanks to Grzegorz Puławski.

Degolfed

a=>{
    string s=Convert.ToString(a,2),      // Convert to binary
    t=s.Split('0')                       // get largest group of 1's
       .OrderBy(x=>x.Length)
       .Last();
    return 
        a<1?0:                          // handle 0 case
        s.Length-s.IndexOf(t)-t.Length; // get position in reversed string
}

C# (.NET Core), 164 161 bytes

a=>{var s=Convert.ToString(a,2);int c=0,l=0,p=0,k=0,j=s.Length-1,i=j;for(;i>=0;i--){if(s[i]>'0'){if(i==j||s[i+1]<'1'){p=i;c=0;}if(++c>=l){l=c;k=p;}}}return j-k;}

Try it online!

A non-Linq solution, which I'm sure could be improved, though nothing is immediately apparent.

Degolfed

a=>{
    var s=Convert.ToString(a,2); // Convert to binary
    int c=0,l=0,p=0,k=0,j=s.Length-1,i=j;
    for(;i>=0;i--)               // Loop from end of string
    {
        if(s[i]>'0')             // if '1'
        {
            if(i==j||s[i+1]<'1') // if first digit or previous digit was '0'
            {
                p=i;             // set the position of the current group
                c=0;             // reset the count
            }
            if(++c>=l)           // if count is equal or greater than current largest count
            {
                l=c;             // update largest count
                k=p;             // store position for this group
            }
        }
    }
    return j-k;                  // as the string is reversed, return string length minus position of largest group
}

1
You can save 8 bytes by ditching r in the first answer: tio.run/##dY9RS8MwFIWfza/…
Grzegorz Puławski

You can save two bytes in your port of Arnauld's like this: T=a=>{int x=(int)Math.Log(a,2);return(a&=a/2)>0?T(a):x<0?0:x;}
Kevin Cruijssen

1
Or even two more bytes saved (60 bytes): T=a=>(a&a/2)>0?T(a&a/2):Math.Log(a,2)<0?0:(int)Math.Log(a,2)
Kevin Cruijssen

2

Husk, 12 bytes

→S§-€¤≠Lo▲gḋ

Try it online!

Explanation

                 Implicit input, e.g                           750
           ḋ     Convert to binary                             [1,0,1,1,1,0,1,1,1,0]
          g      Group equal elements                          [[1],[0],[1,1,1],[0],[1,1,1],[0]]
        o▲       Maximum                                       [1,1,1]
    €            Index of that substring in the binary number  3
     ¤≠L         Absolute difference of lengths                abs (3 - 10) = 7
 S§-             Subtract the two                              7 - 3 = 4
→                Increment                                     5

2

Jelly,  14 13 12  11 bytes

Bµṣ0Ṫ$ƤMḢạL

A monadic link taking and returning non-negative integers.

Try it online! or see the test-suite.

How?

Bµṣ0Ṫ$ƤMḢạL - Main link: number, n                   e.g. 750
B           - convert to a binary list                    [1,0,1,1,1,0,1,1,1,0]
 µ          - monadic chain separation, call that b
      Ƥ     - map over prefixes:  (i.e. for [1], [1,0], [1,0,1], [1,0,1,1],...)
     $      -   last two links as a monad:                e.g.: [1,0,1,1,1,0,1,1]
   0        -     literal zero                                   0
  ṣ         -     split (prefix) at occurrences of (0)           [[1],[1,1,1],[1,1]]
    Ṫ       -     tail                                                        [1,1]
       M    - maximal indices                             [5,9]
        Ḣ   - head                                        5
          L - length (of b)                               10
         ạ  - absolute difference                         5

I had a similar solution but used ŒrṪP on the prefixes instead.
miles

2

Jelly, 19 bytes

BŒgḄṀB
BwÇɓÇL+ɓBL_‘

Try it online!

Thanks to Jonathan Allan for saving  4  6 bytes!

I've worked too much to just abandon this, although it is kind of long. I really wanted to add a solution that literally searches for the longest substring of 1s in the binary representation...

Explanation

BŒgḄṀB  - Monadic helper link. Will be used with Ç in the next link.

B       - Binary representation.
 Œg     - Group runs of consecutive equal elements.
   Ḅ    - Convert from binary to integer.
    Ṁ   - Maximum value.
     B  - Convert from integer to binary.


BwÇɓÇL+ɓBL_‘  - Main link.

B             - Binary representation (of the input).
  Ç           - Last link as a monad. Takes the input integer.
 w            - First sublist index.
   ɓ          - Start a separate dyadic chain. Reverses the arguments.
    Ç         - Last link as a monad.
     L        - Length.
      +       - Addition
       ɓ      - Start a separate dyadic chain. Reverses the arguments.
        B     - Binary.
         L    - Length.
          _‘  - Subtract and increment the result (because Jelly uses 1-indexing).
              - Implicitly output.

Your helper link could be BŒgḄṀB instead
Jonathan Allan

@JonathanAllan Oh wow thanks!
Mr. Xcoder

Your main link could be BwÇɓÇL+ɓBL_‘
Jonathan Allan

@JonathanAllan Wow, thanks again!
Mr. Xcoder

2

Kotlin, 77 bytes

{val b=it.toString(2)
b.reversed().lastIndexOf(b.split(Regex("0+")).max()!!)}

Beautified

{
    val b = it.toString(2)
    // Find the left position of the first instance of
    b.reversed().lastIndexOf(
            // The largest group of 1s
            b.split(Regex("0+")).max()!!)
}

Test

var s:(Int)->Int =
{val b=it.toString(2)
b.reversed().lastIndexOf(b.split(Regex("0+")).max()!!)}
fun main(args: Array<String>) {
    r(0, 0)
    r(142, 1)
    r(48, 4)
    r(750, 5)
}

fun r(i: Int, i1: Int) {
    var v = s(i)
    println("$i -> $v [$i1] ${i1 == v}")
}

I did not test but think this works in scala too.
V. Courtois

2

Haskell, 101 98 96 75 bytes

snd.maximum.(`zip`[0..]).c
c 0=[0]
c n|r<-c$div n 2=sum[r!!0+1|mod n 2>0]:r

Try it online! Usage: snd.maximum.(`zip`[0..]).c $ 142 yields 1.

Explanation:

  • c converts the input into binary while at the same time counting the length of streaks of one, collecting the results in a list. r<-c$div n 2 recursively computes the rest r of this list, while sum[r!!0+1|mod n 2>0]:r adds the current length of the streak to r. The list comprehension checks if mod n 2>0, that is whether the current binary digit is a one, and if so, takes the length of the previous streak (the first element of r) and adds one. Otherwise the list comprehension is empty, and sum[] yields 0. For the example input, c 142 yields the list [0,3,2,1,0,0,0,1,0].

  • (`zip`[0..]) adds the position to each element of the previous list as the second component of a tuple. For the example this gives [(0,0),(3,1),(2,2),(1,3),(0,4),(0,5),(0,6),(1,7),(0,8)].

  • maximum finds the lexicographically maximal tuple in this list, that is the streak lengths are considered first as they are the first component, and in the event of a tie the second component, namely the larger index, decides. This yields (3,1) in the example, and snd returns the second component of the tuple.




2

MS SQL Server, 437 426 407 398 bytes

SQL Fiddle

I'm sure that I could remove line breaks, etc, but this is as compact as I was willing to make it:

create function j(@ int)
returns int
as BEGIN
declare @a varchar(max)='',@b int,@c int=0,@d int=0,@e int=0,@f int=0,@g int=0,@h int=0
while @>0 BEGIN SELECT @a=cast(@%2 as char(1))+@a,@=@/2
END
SET @b=len(@a)
while @<@b
BEGIN
select @c=@d,@d=cast(substring(@a,@b-@,1)as int)
IF @d=1
BEGIN IF @c=0
SELECT @e=@,@g=1
else SET @g+=1 END
IF @g>=@h BEGIN select @h=@g,@f=@e END
SET @+=1
END
return @f
END

Here's a more readable version:

create function BinaryString(@id int)
returns int
as BEGIN
  declare @bin varchar(max)
  declare @binLen int
  declare @pVal int = 0
  declare @Val int = 0
  declare @stC int = 0 --start of current string of 1s
  declare @stB int = 0 --start of biggest string of 1s
  declare @lenC int = 0 --length of current string of 1s
  declare @lenB int = 0 --length of biggest string of 1s

  set @bin = ''

    while @id>0
      BEGIN
        SET @bin = cast(@id%2 as varchar(1)) + @bin
        SET @id = @id/2
      END

    SET @binLen = len(@bin)

    while @id<@binLen
      BEGIN
        set @pVal = @Val
        set @Val = cast(substring(@bin,@binLen-@id,1) as int)
        IF @Val = 1 and @pVal = 0
          BEGIN 
            SET @stC = @id
            SET @lenC = 1
          END
        IF @Val = 1 and @pVal = 1
          BEGIN 
            SET @lenC = @lenC + 1
          END
        IF @lenC >= @lenB
          BEGIN
            set @lenB = @lenC
            set @StB = @StC
          END

        SET @id = @id + 1 
      END

  return @StB
END

The real trick is that as far as I could find, there is no native SQL functionality to convert a number from decimal to binary. As a result, I had to code the conversion to binary manually, then I could compare that as a string, one character at a time until I found the right number.

I'm sure there's a better way to do this, but I didn't see a(n) SQL answer, so I figured I'd throw it out there.


If you can golf it more, please do so. But otherwise, this is great! Welcome to PPCG!
NoOneIsHere

@NoOneIsHere thanks! I realized I can shorten my function name too ;)
phroureo

2

APL (Dyalog Unicode), 22 chars = 53 bytes

Requires ⎕IO←0 which is default on many systems.

⊃⌽⍸(∨⌿↑⊆⍨b)⍷⌽b2⊥⍣¯1⊢⎕

Try it online!

 prompt for input

 yield that (separates ¯1 from )

2⊥⍣¯1 convert to base-2, using as many positions as needed

b← store as b (for binary)

 reverse

() mark the starting positions of the following in that:

⊆⍨b self-partition b (i.e. the 1-streaks of b)

 mix (make list of lists into matrix, padding with zeros)

∨⌿ vertical OR reduction (yields longest streak)

ɩndices of starting positions

 reverse

 pick the first one (yields zero if none available)


you're missing a ∇ in the footer on tio
ngn

@ngn Right you are.
Adám

1

MATL, 15 bytes

`tt2/kZ&t]xBnq&

Try it online!

Uses the half and AND idea. The k is necessary only to make it terminate for 1 - for some reason, 1 AND 0.5 returns 1, causing an infinite loop.

(alternate solution: BtnwY'tYswb*&X>)- by converting to binary and run-length encoding)


1

Google Sheets, 94 bytes

=Len(Dec2Bin(A1))-Find(MAX(Split(Dec2Bin(A1),0)),Dec2Bin(A1))-Len(MAX(Split(Dec2Bin(A1),0)))+1

No, it's not very pretty. It'd be real nice to be able to store Dec2Bin(A1) as a variable for reference.

Key point: Like Excel, the Dec2Bin function has a max input value of 511. Anything larger than that returns an error, as seen below.

Results


1

R, 117 Bytes

z=rev(Reduce(function(x,y)ifelse(y==1,x+y,y),strtoi(intToBits(scan())),,,T));ifelse(!sum(z),0,33-which.max(z)-max(z))

104 bytes! Instead of rev, use Reduce(...,T,T) to accumulate from the right (switching x,y in the function definition). Then use 1+max(z)-which.max(z) since the result is somewhat different. Use "if" instead of "ifelse" since we don't need the vectorization; aaand if you use any(z) instead of !sum(z) you drop a byte.
Giuseppe

I think we should be able to get this to less than 100 bytes.
Giuseppe

@giuseppe I feel somewhat a cheater for being here before you! Will do times tnx a bunch!
Zahiro Mor

But a nice approach no ?
Zahiro Mor

1
Oh, no worries, I saw this earlier but didn't feel like answering it because R is so bad at bit operations... Yeah, good work, you got my +1
Giuseppe

1

Excel VBA, 54 44 Bytes

-10 Bytes thanks to @EngineerToast

Anonymous VBE immediate window function that takes input from range [A1] and outputs to the VBE immediate window

?Instr(1,StrReverse([Dec2Bin(A1)]),1)+[A1>0]

1
Besides the oversight having C1 still in there instead of A1, I think you can output the Instr results directly with a little twisting for the zero input correction: ?Instr(1,StrReverse([Dec2Bin(A1)]),1)+([A1]>0) (46 bytes). True = -1 because... VBA.
Engineer Toast

@EngineerToast - Sweet! I should have seen that; I was able to drop it down 2 bytes by bringing the >0 into the [A1] notation
Taylor Scott



0

R, 66 Bytes

function(i){a=rle(intToBits(i));max(0,a[[1]][which(a[[2]]==1)])}

Explanation:

function(i){
  a = rle(                  # Run-length encode
    intToBits(i)            # The bits in i
  );                        # And assign it to a.
  max(0,                    # Return the maximum of zero and
      a[[1]][               # The lengths of a,
        which(a[[2]]==1)    # But only those where the repeated bit is 1
        ])
}

1
This returns the length of the longest streak, not its position. Check against the test cases and the specs at the top.
user2390246

I will change my downvote to an upvote once this answer is corrected. Also, note that you can use a$l instead of a[[1]] and a$v instead of a[[2]] to save some bytes :), as well as >0 instead of ==1.
Giuseppe


0

Javascript, 54 chars

f=i=>i.toString(2).split(0).sort().reverse()[0].length
  • i.toString(2) gets the binary string for the integer.
  • The .split(0) gets each sequential ones part in an array element.
  • .sort().reverse() gets us the highest value as first.
  • The [0].length gives us the length of that first value.

the starting position of number of largest consecutive 1's
L3viathan

0

Perl 5, 45 + 1 (-p)

(sprintf"%b",$_)=~/(1+)(?!.*1\1)/;$_=length$'

If you write this out on the command line of most shells, you may have to type this as:

perl -pE'(sprintf"%b",$_)=~/(1+)(?!.*1\1)/;$_=length$'"'"

The dance of the quotes at the end is just to get perl see a ', which would otherwise be consumed by the shell.


0

Retina, 52 43 bytes

Convert to binary, then replace with the length of what follows the largest string of ones.

.*
$*
+`(1+)\1
$+0
01
1

$'¶
O`
A-3`
^1+

.

Try it online - all test cases

Saved 9 bytes thanks to Martin.


You can use $+ for ${1}. But you can save even more by replacing the last stage with a bunch of stages like this: tio.run/##K0otycxL/K/…
Martin Ender

@MartinEnder Ok. The ${1} was copied from your tutorial on Github.
mbomb007
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.