Знайдіть суму перших n придурливих чисел


19

Термінологія

Збільшується число - це те, коли кожна цифра більша або дорівнює всім цифрам зліва від неї (наприклад, 12239)

Зменшується число - це число, коли кожна цифра менше або дорівнює всім цифрам зліва від неї (наприклад, 95531)

Прискіпливе число - це будь-яке число, яке не збільшується і не зменшується. Оскільки для цього потрібні щонайменше 3 цифри, перше підйомні число - 101

Завдання

Давши ціле число n, що більше або дорівнює 1, знайдіть суму перших n випромінюваних чисел

Правила

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

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

1 > 101
10 > 1065
44701 > 1096472981

3
Я не впевнений, що я розумію ваші обмеження. Чи можу я sortцифри і перевірити, чи вони такі, як і вихідні? Для цього використовується вбудований ( sort), але це не є вбудованим, щоб перевірити, чи збільшується він. Ознайомтеся з невідповідними вимогами програми та зробіть X без Y на нашому мета-публікації "Що слід уникати".
AdmBorkBork

5
Привіт, Ласкаво просимо до PPCG! Хоча це хороший перший пост (+1), у мене є деякі невеликі пропозиції: Немає вбудовані команди , що перевірка , якщо число збільшується , може використовуватися , Ні вбудовані команди , які перевіряють , чи є рядок лексичний збільшенням може бути використана (забороняючи вбудовані команди) - річ, якої слід уникати під час написання викликів ; у нас є пісочниця для запропонованих викликів , де ви можете поділитися своєю ідеєю після публікації, щоб отримати зворотній зв'язок та вказівки :)
Містер Xcoder

Я оновив обмеження, щоб краще відповідати категорії "Винятки" за посиланням, яке ви опублікували
авер

4
Я досі не бачу сенсу мати таке обмеження в першу чергу. Звичайно, від вас залежить, зберігати чи ні, але забороняти вбудовані програми - це як правило погана практика. Якщо ви відчуваєте, що виклик тривізований вбудованими модулями, слід зазначити, що просто обмеження їх не робить рішення завдання більш цікавим, а скоріше додає котельну плиту. Чи можете ви розглянути можливість зняття цього обмеження? (до речі, це все ще підпадає під Do X без Y ) Інакше ідея мені дуже подобається, і я не хотів би, щоб трохи суб'єктивне обмеження відволікало від фактичного завдання.
Містер Xcoder

10
Однак я усунув обмеження, оскільки зрозуміло, що громаді це приємніше таким чином, і я буду довіряти вказівкам та передовим практикам, які забезпечують виклики найкращої якості
авер

Відповіді:


8

Желе , 10 8 байт

ṢeṚƬ¬µ#S

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

Як це працює

ṢeṚƬ¬µ#S  Main link. No arguments.

      #   Read an integer n from STDIN and call the chain to the left with argument
          k = 0, 1, 2, ... until n of them return a truthy result.
          Yield the array of successful values of k.
     µ    Monadic chain. Argument: k (integer)
Ṣ           Sort, after promoting k to its digit array.
  ṚƬ        Reverse 'til the results are no longer unique and yield unique results.
            Calling Ṛ on k promotes it to its digit array. If k = 14235, the 
            result is [14235, [5,3,2,4,1], [1,4,2,3,5]].
 e          Check if the result to the left appears in the result to the right.
    ¬       Negate the resulting Boolean.
       S  Take the sum.

4
+1 ṚƬнадзвичайно акуратно ...
Містер Xcoder

6

Pyth , 10 байт

s.f!SI#_B`

Спробуйте тут!

Як це працює?

sf! SI # _B` - Повна програма. Бере ціле число Q від STDIN і виводить на STDOUT.
 .f - Знайдіть перші Q додатні цілі числа, які задовольняють певній умові.
   ! SI # _B - Умова. Повертає істину лише для підстрибуючих чисел.
       _B` - Відкиньте число до рядка та роздвойте (з’єднайте) його зворотним боком.
      # - Фільтруйте - зберігайте ці ...
     Я - інваріант під ...
    S - сортування.
           - Для уточнення, я (інваріант) - оператор Pyth, який приймає два входи, a 
             функція та значення та перевіряє, чи функція (значення) == значення, значить
             це технічно не є вбудованим.
   ! - Логічного немає. Порожній список відображається на істинне, інші значення - на хибне.
s - сума.

4

K (нг / к) , 37 байт

{+/x{{(a~&\a)|a~|\a:10\x}(1+)/x+1}\0}

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

{ } - це функція з аргументом x

x{ }\0застосовується {}на 0 xчас, зберігаючи проміжні результати

(1+) є функцією правонаступника

{ }(1+)/x+1застосовує функцію наступника , починаючи з x+1доти , {}поки повертає істину

10\x - десяткові цифри x

a: призначити a

|\ - це максимальне сканування (часткові максимуми) a

&\ аналогічно - це міні-сканування

a~|\aце aвідповідає його максимальному скануванню?

| або

a~&\a його міні-сканування?

+/ сума


4

JavaScript (ES6), 77 байт

f=(i,n=0,k,p)=>i&&([...n+''].map(x=>k|=x<p|2*(p<(p=x)))|k>2&&i--&&n)+f(i,n+1)

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

Прокоментував

f = (                     // f = recursive function taking:
  i,                      //   i = number of bouncy numbers to find
  n = 0,                  //   n = current value
  k,                      //   k = bitmask to flag increasing/decreasing sequences
  p                       //   p = previous value while iterating over the digits
) =>                      //
  i && (                  // if there's still at least one number to find:
    [...n + '']           //   turn n into a string and split it
    .map(x =>             //   for each digit x in n:
      k |=                //     update k:
        x < p |           //       set bit #0 if x is less than the previous digit
        2 * (p < (p = x)) //       set bit #1 if x is greater than the previous digit
                          //       and update p
    )                     //   end of map()
    | k > 2               //   if both bits are set (n is bouncy):
    && i--                //     decrement i
    && n                  //     and add n to the total
  ) + f(i, n + 1)         //   add the result of a recursive call with n + 1

3

Python 2, 110 92 89 байт

n=input()
x=s=0
while n:b={-1,1}<=set(map(cmp,`x`[:-1],`x`[1:]));s+=x*b;n-=b;x+=1
print s

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

Ця функція визначає, наскільки прискіпливе число:

lambda x:{-1,1}<=set(map(cmp,`x`[:-1],`x`[1:]))

Ви можете безпосередньо порівнювати персонажів. Насправді ваше розуміння може стати вашим set(map(cmp,`x`[:-1],`x`[1:])).
Якоб

@Jakob Дякую Я завжди забуваю, що ти можеш використовувати mapсаме так.
mbomb007

1
x=s=0\nwhile n:b={-1,1}<=set(map(cmp,`x`[:-1],`x`[1:]));s+=x*b;n-=b;x+=1економить 3 байти
пан Xcoder


3

Сітківка , 93 байти

K`:
"$+"{/(?=.*;(_*);\1_)(?=.*_(_*);\2;)/^{`:(#*).*
:$1#;$.($1#
)`\d
*_;
)`:(#+).*
$1:$1
\G#

Спробуйте в Інтернеті! Пояснення:

K`:

Ініціалізація s=i=0. ( sце число #s до :, iчисло #s після.)

"$+"{
...
)`

Повторіть nрази.

/(?=.*;(_*);\1_)(?=.*_(_*);\2;)/^{`
...
)`

Повторюйте, поки iне підстрибуйте.

:(#*).*
:$1#;$.($1#

Збільшити iі зробити копію в десятковій частині.

\d
*_;

Перетворіть цифри копії в одинакові. У тесті на придатність використовується одинарна копія, тому вона працює лише один раз, iі принаймні один раз вона була підвищена.

:(#+).*
$1:$1

Додати iдо sі видалити копію одинарних цифр, так що для наступного проходу внутрішнього циклу тест bounciness зазнає невдачі , і iвід нього бере зріст принаймні один раз.

\G#

Перетворити sв десятковий.

Версія 121 байт обчислюється у десятковій частині, тому може працювати для більших значень n:

K`0:0
"$+"{/(?=.*;(_*);\1_)(?=.*_(_*);\2;)/^{`:(\d+).*
:$.($1*__);$.($1*__)
)+`;\d
;$&*_;
)`\d+:(\d+).*
$.(*_$1*):$1
:.*

Спробуйте в Інтернеті! Пояснення:

K`0:0

Ініціалізація s=i=0.

"$+"{
...
)`

Повторіть nрази.

/(?=.*;(_*);\1_)(?=.*_(_*);\2;)/^{`
...
)

Повторюйте, поки iне підстрибуйте.

:(\d+).*
:$.($1*__);$.($1*__)

Збільшити iта зробити копію.

+`;\d
;$&*_;

Перетворіть цифри копії в одинакові. У тесті на придатність використовується одинарна копія, тому вона працює лише один раз, iі принаймні один раз вона була підвищена.

\d+:(\d+).*
$.(*_$1*):$1

Додати iдо sі видалити копію одинарних цифр, так що для наступного проходу внутрішнього циклу тест bounciness зазнає невдачі , і iвід нього бере зріст принаймні один раз.

:.*

Видалити i.



3

Java 8, 114 112 байт

n->{int i=0,s=0;for(;n>0;++i)s+=(""+i).matches("0*1*2*3*4*5*6*7*8*9*|9*8*7*6*5*4*3*2*1*0*")?0:--n*0+i;return s;}

Використовує регулярний вираз, щоб перевірити, збільшується чи зменшується число. Спробуйте його онлайн тут .

Безголівки:

n -> { // lambda
    int i = 0, // the next number to check for bounciness
        s = 0; // the sum of all bouncy numbers so far
    for(; n > 0; ++i) // iterate until we have summed n bouncy numbers, check a new number each iteration
        s += ("" + i) // convert to a String
             .matches("0*1*2*3*4*5*6*7*8*9*" // if it's not an increasing  number ...
             + "|9*8*7*6*5*4*3*2*1*0*") ? 0 // ... and it's not a decreasing number ...
             : --n*0 // ... we have found another bouncy number ...
               + i; // ... add it to the total.
    return s; // return the sum
}

2

Python 2, 250 байт

n=input()
a=0
b=0
s=0
while a<n:
    v=str(b)
    h=len(v)
    g=[int(v[f])-int(v[0]) for f in range(1,h) if v[f]>=v[f-1]]
    d=[int(v[f])-int(v[0]) for f in range(1,h) if v[f]<=v[f-1]]
    if len(g)!=h-1 and len(d)!=h-1:
       a+=1
       s+=b
    b+=1
print s

Ласкаво просимо! Ви можете переглянути цю сторінку для Поради щодо гольфу в Python
mbomb007

1
Я пропоную використати ;якомога більше висловлювань на одному рядку, видаливши пробіл та визначивши функцію для двох довгих рядків, які дуже схожі, тому ви можете повторно використовувати частину коду. Також можна робити a=b=s=0і len(g)!=h-1!=len(d).
mbomb007

Дякую за поради. Я мушу зараз їхати. але я над цим працюю пізніше.
Hashbrowns


0

Червоний , 108 байт

func[n][i: s: 0 until[t: form i
if(t > u: sort copy t)and(t < reverse u)[s: s + i n: n - 1]i: i + 1
0 = n]s]

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

Більш зрозумілі:

f: func [ n ] [
    i: s: 0
    until [
       t: form i  
       if ( t > u: sort copy t ) and ( t < reverse u ) [
            s: s + i
            n: n - 1
       ]
       i: i + 1
       0 = n
    ]
    s
]

Хороша можливість використання form- form iце на 5 байт коротшеto-string i


0

MATL , 31 30 байт

l9`tVdZS&*0<a?wQtG>?.]y]QT]xvs

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

l       % Initialize counter to 1
9       % First value to check for being bouncy
`       % Start do-while loop
  tV    % duplicate the current number and convert to string
        % (let's use the iteration where the number is 1411)
        % stack: [1, 1411, '1411']
  d     % compute differences between the characters
        %  For non-bouncy numbers, this would be either all 
        %  positive values and 0, or all negative values and 0
        % stack: [1, 1411, [3 -3 0]]
  ZS    % keep only the signs of those differences
        % stack: [1, 1411, [1 -1 0]]
  &*    % multiply that array by its own transpose, so that
        %  each value is multiplied by every other value
        %  for non-bouncy numbers, all products will be >=0
        %  since it would have had only all -1s or all 1 (other than 0s)
        % stack: [1, 1411, [1 -1  0
                            -1  1 0
                            0  0  0]]
  0<a?  % if any value in the matrix is less than 0
    wQ    % switch the counter to top, increment it
          % stack: [1411, 2]
    tG>?  % duplicate it, check if it's gotten greater than the input limit
      .]    % if so, break out of loop
  y]    % else, duplicate bouncy number from inside stack,
        %  keeping a copy to be used for summing later
        % stack: [1411, 2, 1411]
  QT    % increment number, push True to continue loop
]     % loop end marker
x     % if we've broken out of loop, remove counter from stack
v     % concatenate the bouncy numbers we've collected in stack and 
s     % sum them

0

R , 96 байт

function(n){while(n){y=diff(T%/%10^(0:log10(T))%%10);g=any(y<0)&any(y>0);F=F+T*g;n=n-g;T=T+1};F}

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

Пояснення:

while(n){                         # while n > 0

        T%/%10^(0:log10(T))%%10   # split T into digits(T==TRUE==1 at the 1st loop)
y=diff(                         ) # compute the diff of digits i.e. digits[i] - digits[i+1]

g=any(y<0)&any(y>0)               # if at least one of the diff is < 0 and 
                                  # at least one is > 0 then T is "bouncy"

F=F+T*g                           # if bouncy increment F (F==FALSE==0 at the 1st loop)

n=n-g                             # decrement n by 1 if bouncy

T=T+1}                            # increment T by 1 and loop

F}                                # return F

0

Ruby (123 байти)

o=0
x=["0"]
while n>0 do x=(x.join.to_i+1).to_s.split('')
(x.sort!=x&&x.sort!=x.reverse ? (n-=1;o+=x.join.to_i):'')
end
p o

На мене це виглядає досить некрасиво. У цьому блоці визначена прискіпливістьx.sort!=x&&x.sort!=x.reverse



0

C (gcc), 104 байти

f(b,o,u,n,c,y){for(o=u=0;b;u+=y?0:o+0*--b,++o)for(n=o,y=3;n/10;)c=n%10,n/=10,y&=(c-=n%10)<0?:c?2:y;b=u;}

Спробуйте його онлайн тут .

Безголівки:

f(n, // function: return type and type of arguments defaults to int;
     // abusing extra arguments to declare variables
  i,   // number currently being checked for bounciness
  s,   // sum of the bouncy numbers
  j,   // copy of i to be used for checking bounciness in a loop
  p,   // used for investigating the last digit of j
  b) { // whether i is not bouncy; uses the two least significant bits to indicate increasing/decreasing
    for(i = s = 0; // check numbers from zero up; initial sum is zero
        n; // continue until we have n bouncy numbers
        s += b ? 0 // not bouncy, no change to the sum
        : i + 0* --n, // bouncy, add it to the sum and one less bouncy number to go
        ++i) // either way, move to the next number
        for(j = i, b = 3; j/10; ) // make a copy of the current number, and truncate it from the right until there is just one digit left
        // bounciness starts as 0b11, meaning both increasing and decreasing; a value of 0 means bouncy
            p = j % 10, // get the last digit
            j /= 10, // truncate one digit from the right
            b &= // adjust bounciness:
                 (p -= j % 10) // compare current digit to the next
                 < 0 ? // not an increasing number, clear second to least significant bit
                 : p ? 2 // not a decreasing number, clear least significant bit
                 : b; // keep it the same
    n = s; // gcc shortcut for return s
}

Запропонуйте u+=!y?--b,o:0,++oзамість u+=y?0:o+0*--b,++o, ;y&=(c-=n%10)<0?:c?2:y)c=n%10,n/=10;а не;)c=n%10,n/=10,y&=(c-=n%10)<0?:c?2:y;
roofcat
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.