Делится ли число на каждую из его цифр?

47

Мой друг и я работали в лаборатории в нашем классе AP Computer Science и решили написать одну из проблем, так как у нас все еще оставалась свободная половина класса, когда мы закончили. Вот вопрос:

Учитывая число n, делится ли n на каждую из его цифр?

Например, 128 пройдет этот тест - он делится на 1,2 и 8. Любые числа с нулем автоматически дисквалифицируют число. Хотя вы можете использовать другие языки и публиковать решения с ними, если хотите, нам больше всего интересно узнать, насколько компактные люди могут создавать программы на Java, поскольку это язык, который мы используем в классе. Пока у нас обоих 51. Вот мой текущий код:

public boolean dividesSelf(int n){for(int p=n;n%10>0;)n/=p%(n%10)>0?.1:10;return n<1;}
// 51 characters

// Breakdown:
// for(int p=n;         Saves one semicolon to put declaration into for loop
// n%10>0;)             Basic check-for-zero
// n/=                  Pretty simple, discarding one number off of n at a time
// p%(n%10)>0?          If p (the given value) is not divisible by n%10 (the current digit)...
// .1:10;               Divide by .1 (multiply by 10) so it fails the check next iteration. If it is divisible, divide by 10 to truncate the last digit
// return n<1           If the number was fully divisible, every digit would be truncated, and n would be 0. Else, there would still be non-zero digits.

Требования

Подпись метода может быть любой, какой вы захотите. Просто посчитайте тело функции. Убедитесь, что метод возвращает логическое значение и передает только один числовой параметр (не строку).

Код должен быть в состоянии пройти все эти случаи (чтобы оставаться верным направлениям исходного вопроса, учитываются только логические значения true и false, если язык поддерживает логические значения. Если и только если ваш язык не имеет логических переменных, вы может представлять false с 0 и true с любым ненулевым целым числом (предпочтительно 1 или -1):

128 -> true
 12 -> true
120 -> false
122 -> true
 13 -> false
 32 -> false
 22 -> true
 42 -> false
212 -> true
213 -> false
162 -> true
204 -> false

Кроме того, мы не учитывали пробелы, поэтому не стесняйтесь делать то же самое, если только пробелы не важны для работы программы (так что переводы строки в Java не учитываются, а только один пробел между intи x=1считается). Удачи !

Мэтью Киршбаум
источник
18
Добро пожаловать в PPCG! Несколько предложений: 1. Не считая функциональных пробелов, это плохая идея. Любой ответ, написанный в Whitespace , автоматически победит. 2. Должна ли наша заявка печатать / возвращать trueи / falseили все в порядке? 3. Тэг на самом деле здесь не применяется, так как сам вызов не связан с Java. java
Деннис
Хорошо. извините за проблемы. Просто чтобы прояснить это, вы бы посчитали пространство в 'int p = n' функциональным, потому что раньше я этого не делал. Я исправлю другие проблемы, которые вы указали.
Мэтью Киршбаум
5
Все пробелы, необходимые для работы кода, являются функциональными.
FryAmTheEggman
Хорошо, спасибо за ответ!
Мэтью Киршбаум
1
@RickyDemer: так как 0 будет исключительным входом в этом случае (это единственное число с 0цифрами, кратным каждому из них), я думаю, что большинство ответов просто длиннее, неинтересным способом, чтобы включить проверку для него. Так что мне больше нравится проблема, представленная заголовком (делится на цифры, а не на кратность, исключая 0).
Йерун Мостерт

Ответы:

23

Perl 6, 13

sub golf($_) {
   $_%%.comb.all
}

Использует неявную переменную $_- $_ %% .comb.allэквивалентно $_ %% all($_.comb). %%является оператором «делится», и combбез дополнительного аргумента возвращает список символов в строке. Например, если аргумент равен 123, то функция оценивает

123 %% all(123.comb)

который

123 %% all(1, 2, 3)

автоматическое продвижение нитей делает это

all(123 %% 1, 123 %% 2, 123 %% 3)

который

all(True, False, True)

который является ложным в логическом контексте, потому что это соединение «все», и, очевидно, не все его элементы верны.

Должна быть возможность принудительно возвращать возвращаемое значение Boolи скрывать соединение от вызывающих, создавая сигнатуру функции sub golf($_ --> Bool()), но приведения в сигнатурах функций в Rakudo пока не работают. Возвращаемое значение по-прежнему правильно true или false, просто нет Trueили False.

Hobbs
источник
Если вы хотите, чтобы это вернулось, Boolпросто добавьте soв начало кода so$_%%.comb.all.
Брэд Гилберт b2gills
21

C # и System.Linq - 26/40

По правилам, не считая самого объявления метода.

bool dividesSelf(int i) { 
    return(i+"").All(d=>i%(d-48d)<1);
}

Еще раз продемонстрировав, что C # - лучший выбор, когда рассматривается Java ... Я шучу, шучу!

К сожалению, эта функция (и многие другие ответы) не даст правильных результатов для отрицательного ввода. Мы можем это исправить, но решение теряет большую часть своего очарования (и увеличивается до 46 символов в длину):

return(i+"").All(d=>d>48&&i%(d-48)==0||d==45);

Изменить : сбрил один персонаж с предложением Тима.

Изменить : с введением членов с выражением тела в C # 6, мы можем сократить это дальше, вырезав return:

bool dividesSelf(int i) =>
    (i+"").All(d=>i%(d-48d)<1);

в общей сложности 26 символов (по моему мнению, =>их не следует включать больше, чем было бы в скобках). Версия, обрабатывающая отрицательные числа, может быть сокращена аналогичным образом.

Йерун Мостерт
источник
Почему .0? Не нужно ничего, кроме целочисленного модуля.
Питер Тейлор
3
@PeterTaylor: есть, если вы хотите самую короткую программу - i % 0с iцелым числом дает DivideByZeroException.
Йерун Мостерт
2
И с двойным это дает NaN! Приятно!
Питер Тейлор
2
48dтакой же, как 48.0, но на один символ меньше (d для двойного).
Тим С.
1
@StuartLC: лямбды не являются методами; их сфера отличается, поэтому я думаю, что это слишком далеко от правил. Но начиная с C # 6 (который предшествует этому ответу), у нас есть члены с выраженным выражением, которые позволяют нам сократить определение. В отрицательном случае мы не можем использовать &именно потому, &что не закорачиваем - вы получите исключение деления на ноль %. Мы можем исправить это, сделав его двойным (с d), но затем мы снова потеряли один символ.
Йерун Мостерт
18

APL ( 13 11)

(видимо, скобки не в счет)

{0∧.=⍵|⍨⍎¨⍕⍵}

Объяснение:

  • ⍎¨⍕⍵: оценить каждый символ в строковом представлении
  • ⍵|⍨: для каждого из них найдите модуль и
  • 0∧.=: посмотреть, все ли равны 0

Testcases:

      N,[.5] {0∧.=⍵|⍨⍎¨⍕⍵} ¨ N←128 12 120 122 13 32 22 42 212 213 162 204
128 12 120 122 13 32 22 42 212 213 162 204
  1  1   0   1  0  0  1  0   1   0   1   0
Мэринус
источник
APL может сделать X%0? без метания?
Оптимизатор
@ Оптимизатор: да. 0|Xдает X.
Маринус
Сладкий. Также ваш ответ 11 байтов, а не 13
Оптимизатор
9
Только APL не даст ошибку по модулю на 0, а сбой при оценке не-bool как bool;)
FryAmTheEggman
3
На одного персонажа короче с поездом вместо dfn:(0∧.=⍎¨∘⍕|⊢)
ngn
14

Python 2: 43 символа

f=lambda n:any(n%(int(d)or.3)for d in`n`)<1

Проверяет, есть ли у числа ненулевые остатки по модулю его цифр, и выводит отрицание этого. Нулевые цифры обрабатываются странно: поскольку вычисления %0вызывают ошибку, цифры 0заменяются на .3, что, как представляется, всегда дает ненулевой результат из-за неточностей с плавающей запятой.

Тело функции составляет 32 символа.

XNOR
источник
14

Perl - 27 байт

sub dividesSelf{
    $_=pop;s/./!$&||$_%$&/ger<1
}

Не считая подписи функции, как указано в инструкции.

Пример использования:

use Data::Dump qw(dump);
for $i (128, 12, 120, 122, 13, 32, 22, 42, 212, 213, 162, 204) {
  printf "%3d -> %s\n", $i, dump(dividesSelf $i);
}

Пример вывода:

128 -> 1
 12 -> 1
120 -> ""
122 -> 1
 13 -> ""
 32 -> ""
 22 -> 1
 42 -> ""
212 -> 1
213 -> ""
162 -> 1
204 -> ""

Решение проблемы: «Только логические значения true и false учитываются. Значения Truthy / Falsey не учитываются».

use Data::Dump qw(dump);
dump(1 == 1);
dump(0 == 1);

Выходы:

1
""

«True» и «False» определяются как 1и "".

Ошибка:
как справедливо указывает Брэд Гилберт , Perl определяет true как скаляр, который одновременно является целым числом 1и строкой "1", и false как скаляр, который одновременно является целым числом 0и строкой "".

Примо
источник
Это может быть сокращен, не используя $_: pop=~s///ger<1. Я не знаю, согласятся ли с этим ОП 1и ""являются ли они действительными результатами. Если нет, то это можно исправить еще двумя байтами: просто добавьте |0.
HVd
perl -pe'$_=s/./!$&||$_%$&/ger<1|0'составляет 26 байтов, включая |0и -pфлаг. Вам не нужно использовать функцию.
hmatt1
1
На самом деле истинные и ложные ценности больше похожи dualvar(1,'1')и dualvar(0,'').
Брэд Гилберт b2gills
1
@BradGilbert Это интересно. Я достаточно хорошо знаком с perlguts, но я не знал, что true и false были в особом случае. На самом деле это «тройные скаляры», помеченные как SVIV(int), SVNV(double) и SVPV(string).
Примо
1
Фактически, когда вы впервые используете строку в качестве числа или число в качестве строки, переменная модифицируется для хранения этих дополнительных данных. Вот почему вы получаете предупреждение только при первом использовании 'abc'в качестве числа (при условии, что вы use warnings;включили.)
Брэд Гилберт b2gills
13

CJam, 11 10 байт

{
    _Ab:df%:+!
}:F;

Это определяет функцию с именем Fи отбрасывает блок из стека.

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

Контрольные примеры

$ cjam <(echo '{_Ab:df%:+!}:F;[128 12 120 122 13 32 22 42 212 213 162 204]{F}%p')
[1 1 0 1 0 0 1 0 1 0 1 0]

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

_      " Copy the integer on the stack.                                          ";
Ab     " Push the array of its digits in base 10.                                ";
:d     " Cast each digit to Double.                                              ";
f%     " Take the integer on the stack modulus each of its digits.               ";
:+     " Add the results.                                                        ";
!      " Push the logical NOT of the sum.                                        ";
Деннис
источник
Были ли у CJam функции, которые вы использовали для 10-байтового решения при написании вопроса?
lirtosiast
@ThomasKwa: Да, это так. Я протестировал код в версии 0.6.2, которая была выпущена в июле 2014 года.
Деннис
12

JavaScript ES6, 39 32 28 байт

v=>[...""+v].every(x=>v%x<1)

Спасибо core1024 за предложение заменить (""+v).split("")с [...""+v], и openorclose для предполагая использование everyфункции.

Ответ в настоящее время не содержит ни одного бита моего кода: O

Предыдущее решение

v=>[...""+v].filter(x=>v%x|!+x)==""

==""не является допустимым способом проверить, является ли массив пустым, поскольку [""]==""возвращает true, но массив гарантированно содержит непустую строку, поэтому он работает здесь.

Остальные являются стандартным сокращением типов преобразования в JavaScript.

n̴̖̋h̷͉a̷̭̿h̸̡̅ẗ̵̨d̷̰ĥ̷̳
источник
1
Вы можете сохранить некоторые символы, заменив их (""+v).split("")на [...""+v].
core1024
1
Почему бы не использовать everyметод? v=>[...""+v].every(x=>v%x<1);
openorclose
@openorclose: спасибо. Никогда не было возможности использовать его в JS, поэтому я никогда не думал о поиске такой функции.
n̴̖̋h̷͉̃a̷̭̿h̸̡̅ẗ̵̨́d̷̰̀ĥ̷̳
v=>![...""+v].some(x=>v%x)
l4m2
@ l4m2 Поскольку v%0возвращается NaNи NaN == false, таким образом , в вашем случае чисел , который содержит 0, например 10, может вернуться true.
Шиеру Асакото
9

Java 8, 46 байт (тело метода)

Использование преобразования Джероена Мостерта в двойной трюк.

public static boolean dividesSelf(int n) {
    return(""+n).chars().allMatch(x->n%(x-48d)<1);
}
CPU1
источник
8

Pyth, 12 байт

!f|!vT%vzvTz

Это фильтрует символы в строке как наличие нуля ( !vT) или не разделяет input ( %vzvT), а затем принимает логическое значение не результирующего списка.

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

isaacg
источник
Нет, я в порядке, если функция не используется. Я просто хотел указать любому, кто использовал функции, что им не нужно считать декларацию, а только код внутри.
Мэтью Киршбаум
8

Ruby, 44 байта (тело функции: 37)

Вероятно, есть потенциал, чтобы дальше играть в гольф.

f=->n{n.to_s.chars.all?{|x|x>?0&&n%x.hex<1}}

Ввод через функцию f. Пример использования:

f[128] # => true
f[12]  # => true
f[120] # => false
...
августейший
источник
1
Вы можете изменить .to_iна .hex, так как однозначные числа одинаковы в базе 16 и могут измениться ==0на <1.
гистократ
8

Python - 59 50 49 47 байт

f=lambda n:all(c>'0'and 0==n%int(c)for c in`n`)

Я уверен, что есть более быстрый способ ... о хорошо.

Редактировать - Спасибо FryAmTheEggman за советы по игре в гольф.

Редактировать 2 - FryAmTheEggman, возможно, также написал это на данный момент, ой

Редактировать 3 - Руки вверх, если вы даже не знали, что генэкспы были чем-то особенным. ...Просто я?

Kasran
источник
О, большое спасибо! Я продолжаю забывать обо всех этих вещах. (Я также не осознавал, что таким образом вы могли бы меньше чем символы).
Kasran
О, переключение логики также, кажется, немного сокращает ее f=lambda n:all([c>'0'and 0==n%int(c)for c in`n`]). И никаких проблем :)
FryAmTheEggman
О, я не понимал , что даже былall метод.
Касран
Будет 1>n%int(c)работать?
Sp3000
3
Почему список-понимание? Использование genexp: all(c>'0'and 0==n%int(c)for c in`n`)делает то же самое, с 2 символами меньше и даже с сохранением выделения списка.
Бакуриу
8

Pyth 11

!f%Q|vT.3`Q

Это объединяет ответы @ isaacg и @ xnor . Он отфильтровывает цифры из входных данных, проверяя значение input % (eval(current_digit) or .3). Затем он проверяет, является ли полученная строка пустой или нет.

Наткнулся на еще пару вариантов такой же длины:

!f%Q|T.3jQT
!f|!T%QTjQT

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

оборота FryAmTheEggman
источник
5

Bash + coreutils, 44 байта

Полное определение функции:

f()((`tr 0-9 \10<<<$1``sed "s/./||$1%&/g"<<<$1`))

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

Выход:

$ for i in 128 12 120 122 13 32 22 42 212 213 162 204; do f $i; printf "%d " $?; done
1 1 0 1 0 0 1 0 1 0 1 0 $
$
Цифровая травма
источник
Э-э, мне неясно, допустимы ли 1 и 0 или нужно ли печатать true/ false?
Цифровая травма
4

J - 14 символов

Тело функции - это часть после =:. Если мы хотим минимизировать количество символов для всей функции, это 15 символов */@(0=,.&.":|]).

f=:0*/@:=,.&.":|]

,.&.":кратчайший путь в J, чтобы развернуть как число в список его десятичных цифр: преобразовать в строку, разделить цифры и преобразовать каждую цифру обратно в число. ,.&.":|]принимает входной номер ( ]) по модулю ( |) эти цифры. 0*/@:=возвращает true, если все результаты были 0, иначе выдает false.

   f 162
1
   f every 204 212 213
0 1 0
algorithmshark
источник
3

Java - 121 102 97 79 78 байт

Я просто знаю, что это будет забито позже. Ну что ж.

boolean b(int a){int m=10,j,t=1;for(;m<a*10;m*=10){j=10*(a%m)/m;if(j==0||a%j>0)t=0;}return t>0;}

Я вернусь.

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

Хаскелл - 100 54 38

f x=all(\y->y>'0'&&x`mod`read[y]<1)$show x

Все еще учусь, критику оценили

globby
источник
У меня был комментарий здесь, но я случайно удалил его как-то ... Во всяком случае, некоторые предложения: 1) Удалите lengths, они не нужны. 2) Заменить tпо определению. 3) elem y sне нужно. 4) /='0'можно переместить в левый фильтр вместо elem y s. 5) В этом случае /='0'эквивалентно >'0', поскольку каждая буква является цифрой. 6) Положите modв кавычки, чтобы он стал инфиксом. 7) Поместите все в одну строку.
Згарб
1 и 3 были с тех пор, когда я пытался сделать это по-другому и спас код. Спасибо за советы.
сумасшедший
1
Мои предложения: вместо использования s==filter(...)sвы должны использовать all(...)s. Теперь, поскольку sв выражении появляется только один раз, вы можете заменить его на определение и удалить where. также, вместо того, чтобы ==0вы могли использовать <1.
гордый haskeller
отличное улучшение по сравнению с первой версией!
гордый haskeller
Я думаю, что вы все еще можете потерять один байт, если вы замените all(\y->...)$show xна and[...|y<-show x].
Згарб
2

CJam, 15 байтов

{_Abf{_g{%}*}:|!}

Это блок, наиболее близкий к функции в CJam. Я только считаю тело (т.е. опускаю скобки). Вы можете использовать его следующим образом:

128{_Abf{_g{%}*}:|!}~

Или, если вы хотите проверить серию входов, вы можете сделать

[128 12 120 122 13 32 22 42 212 213 162 204]{{_Abf{_g{%}*}:|!}~}%

Блок оставляет 0(ложно) или 1(верно) в стеке, чтобы указать результат. (CJam не имеет логического типа.)

Проверьте это здесь.

Объяснение:

_               "Duplicate input.";
 Ab             "Get base-10 digits.";
   f{      }    "This maps the block onto the list of digits, supplying the input each time.";
     _g         "Duplicate digit, get signum S (0 or 1).";
       { }*     "Repeat this block S times.";
        %       "Take input modulo digit.";
                "This leaves an array of zeroes for divisible digits, non-zeroes
                 for non-divisible digits, and non-zero junk for zeroes.";
            :|  "Fold OR onto this list. One could also sum the list with :+";
              ! "Logical NOT. Turns 0 into 1, and non-zero values into 0.";

Альтернатива, также 15 байтов

{:XAb{X\_X)?%},!}

объяснение

:X              "Store input in X.";
  Ab            "Get base-10 digits.";
    {       },  "Filter this list by the result of the block.";
     X\         "Push another copy of X, swap with digit.";
       _        "Duplicate digit.";
        X)      "Push X+1.";
          ?     "Select digit itself or X+1, depending on whether digit is 0 or not.";
           %    "Take modulo. X%(X+1) will always be nonzero for positive integers.";
              ! "Logical NOT. Turns an empty list into 1 and a non-empty list into 0.";
Мартин Эндер
источник
2

CJam, 15 байтов

{_Abf{_{%}1?}1b!}

{}это самая близкая вещь к функции в CJam. Я просто считаю тело функции

Используйте это так:

128{_Abf{_{%}1?}1b!}~

Получить либо 1(если число делится), либо 0(если число не делится на его цифры).

Попробуйте онлайн здесь

объяснение

_Ab                "Copy the number and split it to its digits";
   f{      }       "For each digit, run this code block on the number";
     _{%}1?        "If the digit is 0, put 1, otherwise perform number modulus digit";
            1b     "We now have an array of modulus corresponding to each digit. Sum it up";
              !    "Negate the sum. If all digits were divisible, sum of modules will be"
                   "0, thus answer should be 1 and vice versa";
оптимизатор
источник
Возможно, я что-то упустил, но после быстрого чтения CJam некоторые вещи, кажется, не имеют смысла: как Abразделить цифры? Кажется, он просто конвертирует его в базу 10. Кроме того, как% узнает, что нужно модифицировать по номеру, а не только по следующей цифре, поскольку кажется, что следующая цифра будет следующей в стеке?
Мэтью Киршбаум
Ответить на все ваши вопросы будет сложно. Это было бы очень легко узнать, помещая ed после каждого символа в коде. Попробуйте запустить128{ed_edAedbedf{ed_ed{ed%ed}1ed?ed}ed1edbed!ed}~
Оптимизатор
1
Ответы на конкретные вопросы: выполнение базы 10 дает массив из 10 преобразованных чисел, которые в данном случае являются самими цифрами. %просто возьмите последние два числа (в данном случае) и рассчитайте мод. Последние два числа здесь - это фактическое число и цифра (всегда)
Оптимизатор
Хорошо, спасибо за совет!
Мэтью Киршбаум
2

C89, 43 байта

unsigned char d(int n, int c) {
        int a=n%10;return!n||a&&!(c%a)&&d(n/10,c);
}

C89 не имеет логического типа. Надеюсь, что это работает. Также я использовал второй параметр, чтобы передать копию оригинального числа через стек, но определение может быть любым. Чтобы получить правильный результат, вам просто нужно вызвать функцию с одинаковым значением для обоих параметров ( d(128, 128)).

РЕДАКТИРОВАТЬ: применены предложенные изменения от анонимного пользователя

MarcDefiant
источник
Взгляните на codegolf.stackexchange.com/review/suggested-edits/17160 , кто-то дал вам несколько советов по игре в гольф
Джастин
Специально против правил. Один параметр.
edc65
Да, именно этот пост и послужил причиной того, что я решил создать это правило, так как было неправильно, что пользователь должен делать дублирование вместо программы.
Мэтью Киршбаум
Я думаю, мне придется добавить функцию-обертку. Добавляет ли объявление этой функции количество байтов?
MarcDefiant
2

C11 - 44 байта в теле функции

Другая версия C, не рекурсивная и без исключения с плавающей запятой.

bool digit_multiple(int i)
{
    for(int n=i;i%10&&n%(i%10)<1;i/=10);return!i;
}

Это также будет работать в C ++, Java и большинстве других C-подобных языков.

Отредактировано, чтобы включить улучшение комментария primo.

SBI
источник
1
Версия, которая компилируется в Java (1.7.0_45-b18): на int n=i;for(;i%10>0&&n%(i%10)<1;i/=10);return i<1;один байт меньше, чем код OP.
Примо
2

Юлия 32 25 23

Улучшено использование цифр

Также исправляет проблему с отрицательными числами

selfDivides(x)=sum(x%digits(x).^1.)==0

Старый метод

Все цифры делятся, если сумма всех остатков равна 0. Как и у других, есть проблема с отрицательными числами.

selfDivides(x)=sum(x.%(Float64["$x"...]-48))==0

Выход

[selfDivides(x) for x in [128,12,120,122,13,32,22,42,212,213,162,204]]
12-element Array{Any,1}:
  true
  true
 false
  true
 false
 false
  true
 false
  true
 false
  true
 false

Улучшенный метод также обрабатывает BigInt

selfDivides(BigInt(11111111111111111111111111111111111111112))
true

тем не мение

selfDivides(BigInt(11111111111111111111111111111111111111113))
false

потому что

BigInt(11111111111111111111111111111111111111113) %3
1
waTeim
источник
2

C / C ++, 58 байт (44 в теле)

Вызывает неопределенное поведение (см. Комментарии)

int d(int i){int j=i;while(i&&!(j%(i%10)))i/=10;return!i;}

trueи false являются 1 и 0, но вы можете добавить один символ подписи возвращать bool.

И для удовольствия, рекурсивная версия, которая меньше, если вы разрешаете вызовы вида r(128,128)

Изменить : теперь запрещено по правилам:

C / C ++, 53 байта (33 в теле)

int r(int i,int j){return!i||!(j%(i%10))&&r(i/10,j);}

etheranger
источник
2
# 1 умирает с исключением с плавающей запятой для чисел, содержащих 0, потому что j% (i% 10) будет недопустимым для i% 10 = 0.
ВОО
С плавающей точкой исключением? Weird. Он отлично работает на моем компиляторе, но вы правы, это неопределенное поведение. Не уверен, какова общая позиция PCG в зависимости от компилятора UB.
etheranger
Что такое UB, зависящий от компилятора? Либо это UB, либо нет (и деление на ноль, или, скорее, по модулю ноль, действительно UB). Нельзя допускать UB, потому что буквально все может случиться. Мы можем предположить, что ваша программа будет работать на машине, которая взорвется и убьет всех вокруг, когда произойдет деление на ноль. Теперь я уверен, что вы хотите, чтобы мы все жили ... В C есть концепция поведения, определяемого реализацией, но деление на ноль не подпадает под это.
Йерун Мостерт
2
@etheranger: деление на 0 называется исключением с плавающей запятой по исторической причине: stackoverflow.com/questions/16928942/…
n̴̖̋h̷͉̃a̷̭̿h̸̡̅ẗ̵̨́d̷̰̀ĥ̷̳
2
@JeroenMostert: я бы сказал, что более 90% всех ответов C на этом сайте вызывают UB. Пока это работает с некоторым компилятором на некотором компьютере, ответ считается действительным.
Денис
2

R: 72 67 65

Функция

f<-function(a)!(anyNA(a%%(d=as.double(strsplit(paste0(a),"")[[1]])))|sum(a%%d))

Спасибо @AlexA и @plannapus за экономию

Тестовый забег

i=c(128,12,120,122,13,32,22,42,212,213,162,204)
for(a in i){print(f(a))}
[1] TRUE
[1] TRUE
[1] FALSE
[1] TRUE
[1] FALSE
[1] FALSE
[1] TRUE
[1] FALSE
[1] TRUE
[1] FALSE
[1] TRUE
[1] FALSE
MickyT
источник
В настоящее время я считаю в вашем теле функции 70 байт, а не 72. Но вы можете уменьшить его до 67 используя d=as.double(strsplit(toString(a),"")[[1]]);!(anyNA(a%%d)|sum(a%%d)). :)
Алекс А.
@AlexA. Благодарю. Одна из моих первых попыток с R. Обязательно вернусь :)
MickyT
@MickyT paste(a)вместо toString(a)дает тот же результат.
plannapus
@plannapus Спасибо, аккуратный трюк. Необходимо помнить, что
MickyT
1

GNU Awk: 53 символа

Подсчитанная часть:

for(;++i<=split($1,a,//);)r=r||!a[i]||v%a[i];return!r

Вся функция:

function self_divisible(v, i, r)
{
    for (; ++i <= split($1, a, //); )
        r = r || ! a[i] || v % a[i]

    return ! r
}

Поскольку Awk не имеет логических значений, возвращает 1 для true и 0 для false.

manatwork
источник
1

JavaScript (ES6) 30

Функция с одним числовым параметром. Используя% и вычитание, нет необходимости в специальном случае '0', потому что 0% 0 - это NaN в JavaScript.

Редактировать Сохранено 1 символ thx DocMax

F=n=>[for(d of t=n+'')t-=n%d]&&t==n 

Просто ради забавы, нарушая правило о неисчислении подписи функции, 4

Check=(n,t=n+'',q=[for(d of t)n-=t%d])=>t==n

Тест в консоли FireFox / FireBug

console.log([128, 12, 120, 122, 13, 32, 22, 42, 212, 213, 162, 204]
.map(x=>+x + ' -> ' + F(x)).join('\n'))

Выход

128 -> true
12 -> true
120 -> false
122 -> true
13 -> false
32 -> false
22 -> true
42 -> false
212 -> true
213 -> false
162 -> true
204 -> false
edc65
источник
Я собираюсь сказать нет вводу строки.
Мэтью Киршбаум
1
Консоль Firefox довольна заменой of(t=n+'')только of t=n+''на сохранение 1.
DocMax
1

PHP: 85 байт (64 байта на теле)

Чтобы эта функция работала, просто передайте строку или число.

0 правильно вернет false.

Код:

function f($n,$i=0){for($n.='';$n[$i]&&$t=!($n%$n[$i++]););return$t&&$i==strlen($n);}

Пожалуйста, НЕ УСТАНАВЛИВАЙТЕ 2-Й ПАРАМЕТР!

Javascript: 76 байт (61 байт на теле)

Это переписать предыдущую функцию.

Не сильно изменилось между обеими версиями.

Вот код:

function f(n){for(i=0,n+='';n[i]/1&&(t=!(n%n[i++])););return t&&i==n.length}

Полиглот: Javascript + PHP 187 217 байты (76 84 байта без шаблона):

Почему я это сделал?

Из-за причины и, возможно, потому что я могу!

Просто игнорируйте ошибку на PHP: все равно работает!
Больше не нужно, это было исправлено путем удаления 3 байтов.

Вот этот шедевр:

if('\0'=="\0"){function strlen($s){return $s['length'];}}
function toString(){return'';}
function f($n){for($i=0,$n=$n.toString();$n[$i]/1&&($t=!($n%$n[$i++])););return $t&&$i==strlen($n);}

Вы можете запустить этот код как на своей консоли, так и на интерпретаторе PHP!


Старая версия:

if('\0'=="\0"){function strlen($s){return $s['length'];}}
function s($s){return('\0'=="\0")?$s+'':str_replace('','',$s);}
function f($n,$i){for($i=0,$n=s($n);$n[$i]/1&&($t=!($n%$n[$i++])););return $t&&$i==strlen($n);}
Исмаэль Мигель
источник
msgstr "и передает только один числовой параметр". Без этого вы можете вычислить ($ x) и передать весь код в $ x
abc667
@ abc667 Извините, но я не понимаю.
Исмаэль Мигель
1

Октава, 33 (39, включая настройку функций)

Использование числового преобразования в матрицу:

f=@(a)sum(mod(a./(num2str(a)-48),1))==0

Разделите число поэлементно на матрицу X, где X получается путем преобразования числа в строку и вычитания 48, чтобы снова перейти от значений ASCII к числам. Возьмите по модулю 1, чтобы получить десятичную часть каждого деления, подтвердите, что все они равны нулю (если они равны NaN из-за / 0, сумма будет равна NaN и, следовательно, не равна нулю).

Пример ввода с использованием www.octave-online.net:

f=@(a)sum(mod(a./(num2str(a)-48),1))==0
for j=[128,12,120,122,13,32,22,42,212,213,162,204]
f(j)
end

Выход:

ans =  1
ans =  1
ans = 0
ans =  1
ans = 0
ans = 0
ans =  1
ans = 0
ans =  1
ans = 0
ans =  1
ans = 0
Йорген
источник
Как мы можем проверить это?
Исмаэль Мигель
octave-online.net - введите определение кода сверху и затем (например) f (128). Добавлю вывод
Йорген
Я нашел компилятор и попробовал его, прежде чем спрашивать. Но, похоже, работает нормально (кроме f(123)делимого на 1, 2 и 3). Но это работает для предоставленных тестовых случаев.
Исмаэль Мигель
1

MATLAB - 39 символов

function [b] = dividesSelf(i)
b=all(~mod(i,sscanf(num2str(i),'%1d')))
end
Bastian35022
источник
1

BASH - 117 знаков

f(){ [[ $1 =~ 0 ]]&& return 0 || r=;n=$1;for((i=0;i<${#n};i++));do r=$(($r+${n}%${n:$i:1}));done;return $(($r==0));}

тесты

for N in 128 12 120 122 13 32 22 42 212 213 162 204; do
  f $N
  echo "${N} ->  $?"
done

128 ->  1
12 ->  1
120 ->  0
122 ->  1
13 ->  0
32 ->  0
22 ->  1
42 ->  0
212 ->  1
213 ->  0
162 ->  1
204 ->  0
Брайан
источник
1

PHP - 74 71 64 символов

Golfed:

function t($n){while($n>1){if(!($b=$n%10)||($n%$b)){return 0;}$n/=10;}return 1;}

Меньше гольфа

function t($n){
    while($n>1){
        if( !($b=$n%10) || ($n%$b) )
            { return 0; }
        $n/=10;
    }
    return 1;
}

Результаты теста:

(Код)

$ans = array(128,12,120,122,13,32,22,42,212,213,162,204);
foreach($ans as $a)
{ echo "$a -> ".(t($a)?"True":"False").PHP_EOL; }

(Выход)

128 -> True
12 -> True
120 -> False
122 -> True
13 -> False
32 -> True
22 -> True
42 -> True
212 -> True
213 -> True
162 -> False
204 -> False
JPMC
источник