Нумерологический калькулятор

19

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

  • Ввод может быть любым удобным способом (стандартный ввод, аргументы, отдельный файл).
  • Ввода может содержать любые печатаемые ASCII символов, но только буквенно - цифровой ( A-Z, a-zи 0-9) должны быть рассмотрены.
  • Выходные данные должны быть цифрой между 1и 9или звездочкой, *если ни одна буква и ни одна цифра не были найдены ... (или даже 0если входные данные содержат любое число 0 и ничего кроме этого, это не имеет значения).
  • Значения букв отображаются следующим образом:

    1  2  3  4  5  6  7  8  9
    a  b  c  d  e  f  g  h  i
    j  k  l  m  n  o  p  q  r
    s  t  u  v  w  x  y  z
    
  • Нумерологии цифра вычисляется путем добавления каждого значения строки, а затем повторить , пока не будет только одна цифра. Образец для 13579, Hello, world!, 00 0 00, !@#$%^&*();, и 3.141592:

    13579 => 1 + 3 + 5 + 7 + 9 = 25 => 2 + 5 = 7
    Hello, world! => 8 + 5 + 3 + 3 + 6 + 5 + 6 + 9 + 3 + 4 = 52 => 5 + 2 = 7
    00 0 00 => 0 + 0 + 0 + 0 + 0 = 0
    !@#$%^&*(); => *
      => *
    3.141592 => 3 + 1 + 4 + 1 + 5 + 9 + 2 = 25 => 2 + 5 = 7
    3.1415926535897932384 => 
     3 + 1 + 4 + 1 + 5 + 9 + 2 + 6 + 5 + 3 + 5 + 8 + 9 + 7 + 9 + 3 + 2 + 3 + 8 + 4
     = 97 => 9 + 7 = 16 => 1 + 6 = 7
    

    (Это здорово, большинство примеров дают 7! Но это всего лишь пример;)

    Еще несколько тестов:

    Bob  => 2 + 6 + 2 = 10 => 1 + 0 = 1
    Charlie => 3 + 8 + 1 + 9 + 3 + 9 + 5 = 38 => 3 + 8 = 11 => 1 + 1 = 2
    Anna => 1 + 5 + 5 + 1 = 12 => 1 + 2 = 3
    Fana => 6 + 1 + 5 + 1 = 13 => 1 + 3 = 4
    Gregory => 7 + 9 + 5 + 7 + 6 + 9 + 7 = 50 => 5 + 0 = 5
    Denis => 4 + 5 + 5 + 9 + 1 = 24 => 2 + 4 = 6
    Erik => 5 + 9 + 9 + 2 = 25 => 2 + 5 = 7
    Helen => 8 + 5 + 3 + 5 + 5 = 26 => 2 + 6 = 8
    Izis => 9 + 8 + 9 + 1 = 27 => 2 + 7 = 9
    

Это , поэтому выигрывает самый короткий ответ в байтах.

Самый короткий по языку

Ф. Хаури
источник
0 исчезнет при добавлении к чему-либо еще
Ф. Хаури
0 не актуально! Соответствующий вывод между 1и 9!
Ф. Хаури
Не могли бы вы добавить несколько примеров без буквенно-цифровых символов?
ETHproductions
1
Я имею в виду что-то вроде !@#$%^&*(); то, что должно вернуться *.
ETHproductions
Было очень трудно удержаться от упоминания этого в некоторых дискуссиях по техническому анализу в области денег ... ;-p
keshlam

Ответы:

6

Matlab, 121 байт

s=[input('','s'),'*'];while nnz(s)>1;s=num2str(sum(mod([s(48<s&s<58)-4,s(96<s&s<123)+2,s(64<s&s<91)-2],9)+1));end;disp(s)

Matlab просто не для строк = (

flawr
источник
3
Эй, по крайней мере, это не так многословно, как Mma: P
LegionMammal978
+1, я не мог сделать лучше. Кстати, нужно ли ставить dispзаявление?
Brainkz
@brainkz Возможно, но я обычно склоняюсь к тому, чтобы быть на безопасной стороне
flawr
3

Mathematica, 174 168 163 байта

Catch[#-9Floor[Max[#-1,1]/9]&@If[(a=Tr[Characters@#/.{(b=a_String)?DigitQ:>FromDigits@a,b?LetterQ:>LetterNumber@a~Mod~9,b->0}])<1&&#~StringFreeQ~"0",Throw@"*",a]]&

Делает первый шаг, затем вычисляет цифровой корень.

LegionMammal978
источник
3

Рубин, 97 74 персонажа

n=->s{(t=eval s.tr('a-z9A-Z','1-9'*6).scan(/\d/)*?+)&&t>9?n[t.to_s]:t||?*}

Образец прогона:

2.1.5 :001 > n=->s{(t=eval s.tr('a-z9A-Z','1-9'*6).scan(/\d/)*?+)&&t>9?n[t.to_s]:t||?*}
 => #<Proc:0x00000001b4b3f0@(irb):4 (lambda)> 

2.1.5 :002 > puts ['13579', 'Hello, world!', '00 0 00', '!@#$%^&*();', ' ', '3.141592', '3.1415926535897932384', 'Bob', 'Charlie', 'Anna', 'Izis'].map{|s|'%s : %s'%[n[s],s]}
7 : 13579
7 : Hello, world!
0 : 00 0 00
* : !@#$%^&*();
* :  
7 : 3.141592
7 : 3.1415926535897932384
1 : Bob
2 : Charlie
3 : Anna
9 : Izis
manatwork
источник
3

Perl, 91 89 76 74 байта

73 + 1 для -pвыключателя

s/[a-z]/(ord($&)%32-1)%9+1/eig;$t="*",s/\d/$t+=$&/eg,$_=$t until/^[*\d]$/

тесты

for test in '13579' 'Hello, world!' '00 0 00' '!@#$%^&*();' ' ' \
    '3.141592' '3.1415926535897932384' \
    Bob Charlie Anna Fana Gregory Denis Erik Helen Izis ;do  
    perl -pe '
      s/[a-z]/(ord($&)%32-1)%9+1/eig;$t="*",s/\d/$t+=$&/eg,$_=$t until/^[*\d]$/
      ' < <(echo -n "$test")
    echo "  $test"
done
7  13579
7  Hello, world!
0  00 0 00
*  !@#$%^&*();
*   
7  3.141592
7  3.1415926535897932384
1  Bob
2  Charlie
3  Anna
4  Fana
5  Gregory
6  Denis
7  Erik
8  Helen
9  Izis

Спасибо @manatwork за помощь в экономии 2 14 16 15 17 символов!

... Я думал о: N % 32 + Yможет заменить( N & 31 ) + Y !

Ф. Хаури
источник
1
Если вы захватите всю совпавшую подстроку, $1равно $&. Так что лучше удали захват и поменяй имя переменной.
manatwork
1
whileБлок может быть $t="*";s/\d/$t+=$&/eg;$_=$t.
Манатворк
1
Извините, но на самом деле это на 1 символ длиннее, так как вы должны включить -pпереключатель в счетчик.
manatwork
1
Снова извините (на этот раз больше, извините), но он не работает при вводе одного неслова, например, «!» Приводит к тому же самому! ». ( Кажется, работает, потому что здесь-строка ввод всегда передается с новой строки, поэтому в ваших тестовых случаях никогда не проходит ввод одного символа. Используйте, echo -n "$test" | perl -pe '…'чтобы увидеть, что я имею в виду.) Вы можете решить это, изменивwhile условие на !/^[\d*]$/. В то время, чтобы уменьшить потери балла, вы можете заменить весь whileс этим: $t="*",s/\d/$t+=$&/eg,$_=$t until/^[\d*]$/.
manatwork
1
Есть еще одно улучшение, которое вы можете сделать. Как &имеет более низкий приоритет , чем +, в то время как %имеет более высокое, изменение &31%32и вы можете удалить скобки вокруг этого подвыражения.
manatwork
3

ES6, 98 байт

s=>(m=s.match(/[1-9a-z]/gi))?(t=8,m.map(c=>t+=c>'@'?c.charCodeAt()&31:+c),t%9+1):/0/.test(s)?0:'*'

Ungolfed:

function(s) {
    var m = s.match(/[1-9a-z]/gi);
    if (m) {
        var t = 0;
        for (var i = 0; i < m.length; i++) {
            if (m[i] > '@')
                t += m[i].charCodeAt(0) & 31;
            else
                t += parseInt(m[i]);
        }
        return t % 9 || 9;
    }
    return /0/.test(s) ? 0 : "*";
}

94-байтовая версия, которая работает только на коротких строках:

s=>(m=s.match(/[1-9a-z]/gi))?m.map(c=>c>'@'?c.charCodeAt()&31:c).join``%9||9:/0/.test(s)?0:'*'

Использование match, mapи joinоказалось короче, чем использование в replaceдва раза:

s=>(m=s.replace(/[^1-9a-z]/gi,''))?m.replace(/[a-z]/gi,c=>c.charCodeAt()&31)%9||9:/0/.test(s)?0:'*'

Проверьте это здесь: https://jsbin.com/zizosayisi/edit?js,console

Нил
источник
Это не работает для ввода00 0 00
rink.attendant.6
@ rink.attendant.6 Ах, РЕПЛ я использовал только отпечатки , ["0"]как 0и я не мог сказать разницу.
Нил
Гораздо лучше, чем версия @ rink.attendant.6! Умные гольф-хаки используются. Не могли бы вы объяснить > '@', & 31а || 9?
Павло
1
@Pavlo > '@'просто различает буквы и цифры. & 31полезный способ игнорировать разницу между кодами символов в верхнем и нижнем регистре, поскольку он также удобно напрямую отображает коды символов в значения 1..26. || 9используется потому, что % 9возвращает 0 для кратных 9, но многократное добавление цифр возвращает 9 для ненулевых кратных 9. В коде для гольфа, который я использую (t + 8) % 9 + 1вместо этого, происходит то же самое.
Нил
Язык обычно называют «JavaScript (ES6)»
edc65
2

Гема, 161 персонажа

*=@n{*}
n:\A=@set{t;};<L1>=@set{t;@add{$t;@add{@mod{@sub{@mod{@char-int{$0};32};1};9};1}}};<D1>=@set{t;@add{$t;$0}};?=;\Z=@cmps{$t;;;\*;@cmpn{$t;9;$t;$t;@n{$t}}}

(Написано только для того, чтобы проверить, работают ли рекурсивные доменные вызовы.)

Образец прогона:

bash-4.3$ for input in '13579' 'Hello, world!' '00 0 00' '!@#$%^&*();' ' ' '3.141592' '3.1415926535897932384'; do
>     echo -n "'$input' : "
>     gema '*=@n{*};n:\A=@set{t;};<L1>=@set{t;@add{$t;@add{@mod{@sub{@mod{@char-int{$0};32};1};9};1}}};<D1>=@set{t;@add{$t;$0}};?=;\Z=@cmps{$t;;;\*;@cmpn{$t;9;$t;$t;@n{$t}}}' <<< "$input"
>     echo
> done
'13579' : 7
'Hello, world!' : 7
'00 0 00' : 0
'!@#$%^&*();' : *
' ' : *
'3.141592' : 7
'3.1415926535897932384' : 7
manatwork
источник
1

JavaScript (ES6), 162 159 157 байт

f=_=>{for(_=_.replace(/\W+/g,''),g=$=>''+[...$.toUpperCase()].reduce((p,a)=>isNaN(a)?p+(a.charCodeAt()-64)%9:+a+p,0);1<(l=_.length);_=g(_));return!l?'*':g(_)}

Все еще пытаюсь найти способ использовать неявный возврат во внешней функции.

Ungolfed + Unminified

f = str => {
  str = str.replace(/\W+/g, '');
  recursiveFunc = $ => String([...$.toUpperCase()].reduce(
    (prev, val) => isNaN(val) ? prev + (val.charCodeAt() - 64) % 9 : Number(val) + prev,
    0
  ));
  while (1 < (len = str.length)) {
    str = recursiveFunc(str);
  }
  return len === 0 ? '*' : recursiveFunc(str)
}
  1. Удаляет все не алфавитно-цифровые символы
  2. Вызывает функцию рекурсивно, чтобы привести символы к их соответствующим значениям, пока строка длиннее 1 символа.
    1. Преобразует строку в верхний регистр, чтобы легко работать с кодами ASCII
    2. Конвертировать в массив с помощью оператора распространения и запустить аккумулятор
    3. Используйте глобальную isNaNфункцию (которая вводит свои данные), чтобы проверить, не является ли это число
      • Если это не так, преобразуйте в код ASCII и мод 9, чтобы получить соответствующее значение
      • Если это число, бросьте его
  3. Если длина равна 0, алфавитно-цифровые символы отсутствуют (возвращают звездочку), в противном случае возвращаются выходные данные рекурсивной функции.
rink.attendant.6
источник
Это правило разрыва 00 0 00, вывод *вместо 0!
Ф. Хаури
@ F.Hauri Исправлено.
rink.attendant.6
Вы можете избежать явного возврата, используя оператор запятой: _=>{expr;return expr}=>_=>(expr,expr)
Pavlo
@Pavlo Как мне применить это в моем случае?
rink.attendant.6
У меня не работает, выдает SyntaxError: jsbin.com/havotusoqa/1/edit?js,console
Павло
1

Haskell, 126 байт

l x=[n|(c,n)<-zip(['0'..'9']++['a'..'z']++'\0':['A'..'Z'])$0:cycle[1..9],c==x]
g[]="*"
g[x]=show x
g x=f$show$sum x
f=g.(l=<<)

Использование: f "Hello, world!"-> "7".

lтаблица поиска символов для списка целых чисел (одноэлементный список, если символ найден, пустой список в противном случае). fпросматривает каждый символ его аргумента и выравнивает возвращенный список списков в простой список целых чисел и вызывает, gчтобы проверить условие конца (пустой список (-> *) или одно целое число) или вызвать fсумму списка для другого раунда.

Ними
источник
1

MATL , 64 байта

jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']

При этом используется текущая версия (4.0.0) языка.

У меня такое ощущение, что это можно сделать короче ...

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

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> 13579
7

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> Hello, world!
7

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> 00 0 00
0

>> matl 
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> !@#$%^&*();
*

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> 3.141592
7

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> 3.1415926535897932384
7

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> Bob
1

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> Charlie
2

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> Anna
3

>> matl
 > jk42ht`YUt'[a-z]'XXY}3+w'[1-9]'XXY}6+h,9X\st9>]w2Y24Y2h!=~?x'*']
 > 
> Izis
9
Луис Мендо
источник
К сожалению, я не смог проверить ваш образец. Пожалуйста, опубликуйте целые тестовые случаи, включая недавно добавленные тесты (извините).
Ф. Хаури
@ F.Hauri Да, тебе нужен Матлаб, чтобы проверить их. К сожалению об этом, онлайн-компилятора пока нет. Я добавил тестовые случаи
Луис Мендо
1

Серьезно, 50 байтов

,$ù;ú1╤▀+;#pX╗@-@-;Y`'*.`╬X1WX`╜í;s9*@%u`MΣ$;lDWX

Шестнадцатеричный дамп:

2c24973ba331d1df2b3b237058bb402d402d3b5960272a2e7f
60ce5831575860bda13b73392a402575604de4243b6c445758

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

Разъяснение:

,$ù                                               Read input, make it a string, lowercase
    ú                                             Push lowercase alphabet
     1╤▀+                                         Prepend base 10 digits.
         ;#pX╗                                    Remove "0" from a copy and stash in reg0
   ;          @-                                  Remove alphanumerics from input copy
                @-                                Remove nonalphanumerics from input
                  ;Y                              Push 1 if string is empty, else 0
                    `'*.`╬                        If top is truthy, output * and halt
                          X                       Discard boolean
                           1                      Push 1 to enter loop
                            WX                 WX Loop while top of stack is truthy
                              `         `M        Map this function over the string
                               ╜                  Push alphanumeric string from reg0
                                í                 Push index of this char in it
                                 ;s9*             Push 9 if found, else -9
                                     @%u          Take index mod previous: this yields the
                                                  correct conversion from the numerology
                                          Σ       Sum the resulting digits.
                                           $      Convert the sum to a string.
                                            ;lD   push 1 less than its length
quintopia
источник
@ F.Hauri Нажмите «Попробуйте онлайн». введите ваши тесты в поле ввода.
Quintopia
1

Pyth, 39 байт

IK@J+jkUTGrz0WtK=K`smh%xtJd9-K\0)K).?\*

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

Я просто делаю это, потому что не могу спать. Может быть, я объясню это завтра ...

quintopia
источник
1

Чистый bash, 199 194 байта

eval a+={a..z};r="$1";while [ "${r:1}" ];do o=;for ((i=0;i<${#r};i++));do
l=${r:i:1};case $l in [a-zA-Z])d=${a%${l,}*};((o+=$((${#d}%9+1))));;[0-9])
((o+=l));;esac;done;r="$o";done;echo "${o:-*}"

(разрыв второй строки предназначен только для обхода полосы прокрутки)

Тестовое правило:

numerology() {
    eval a+={a..z};
    r="$1";
    while [ "${r:1}" ]; do
        o=;
        for ((i=0; i<${#r}; i++))
        do
            l=${r:i:1};
            case $l in 
                [a-zA-Z])
                    d=${a%${l,}*};
                    ((o+=$((${#d}%9+1))))
                ;;
                [0-9])
                    ((o+=l))
                ;;
            esac;
        done;
        r="$o";
    done;
    echo "${o:-*}"
}

for test in '13579' 'Hello, world!' '00 0 00' '!@#$%^&*();' ' ' \
            '3.141592' '3.1415926535897932384'\
            Bob Charlie Anna Fana Gregory Denis Erik Helen Izis ;do
    echo "$(numerology "$test")" $test
done
7 13579
7 Hello, world!
0 00 0 00
* !@#$%^&*();
*
7 3.141592
7 3.1415926535897932384
1 Bob
2 Charlie
3 Anna
4 Fana
5 Gregory
6 Denis
7 Erik
8 Helen
9 Izis
Ф. Хаури
источник
1

JavaScript (ES6), 78 83

Рекурсивное решение. Будучи хвостовой рекурсией, переменные t и r не должны быть локальными.

f=x=>(t=r=0,x.replace(/\w/g,d=>t+=1+~-parseInt(d,r=36)%9),t>9?f(''+t):r?t:'*')

Разъяснения

f=x=>(
  t = 0, // set initial value of counter to 0 
  r = 0, // flag to verify that we found at last one alphanumeric chars
  x.replace(/\w/g, d => ( // execute the following for each alphanumeric character
    // t += 1 + ~-parseInt(d,r=36) % 9 explained below
    r = 36, // set flag, could be any nonzero value
    d = parseInt(d,36), // convert to numeric. a..z -> 10..25, case insensitive.
    d = 1 + (d-1) % 9, // this is the arithmetic conversion required (
                       // works also with 0 because the strange implementation of % in javascript for negative numbers
    t = t + d // add to global counter
  ) ), 
  t > 9 // if t > 9 then we found some alphanumeric char, but we must repeat the loop on t
    ? f(''+t) // recursive call, t is numeric and must become a string
    : r // check flag r 
      ? t // if alphanumeric found, return t 
      : '*' // else return '*'
)

Тестовый фрагмент

f=x=>(t=r=0,x.replace(/\w/g,d=>t+=1+~-parseInt(d,r=36)%9),t>9?f(''+t):r?t:'*')

console.log=x=>O.textContent+=x+'\n';

;[['13579',7],['Hello, world!',7],['00 0 00',0],['!@#$%^&*();','*'],
['3.141592',7],['3.1415926535897932384',7],
['Bob', 1],['Charlie', 2],['Anna', 3],['Fana', 4],['Gregory', 5],
['Denis', 6],['Erik', 7],['Helen', 8],['Izis', 9]]
.forEach(t=>{
  i=t[0]+''
  k=t[1]
  r=f(i)
  console.log(i+' -> ' + r + (k==r? ' OK':' Fail - expected '+k))
  })
<pre id=O></pre>

edc65
источник
0

Python, 154 байта

def A(S):
 D=lambda x:int(x)if x.isdigit()else (ord(x.lower())-6)%9
 while len(S)>1:S=str(sum([D(c)for c in S if c.isalnum()]))
 print S if int(S)else"*"
Киран Хант
источник
Это провальный тест с 00 0 00!
Ф. Хаури
0

Mathematica, 133 байта

f[s_]:= ToCharacterCode@ToUpperCase@s-64/.{a_/;17>-a>6:>a+16,b_/;b<1||b>26:>""}//If[Or@@NumberQ/@#,Tr@#/.""->0//.a_:>Tr@IntegerDigits@a,"*"]&

Немного отличается от LegionMammal978 выше. Моя функция превращает все в символьный код, затем отфильтровывает не алфавитно-цифровые элементы (заменяя их пустыми строками). Если алфавитно-цифровых символов нет, то возвращается *, в противном случае он принимает цифровой корень. Это могло бы быть значительно (~ 15B) короче, если бы мне не приходилось иметь дело со случаем со всеми нулями. Такова жизнь.

Mathematica magic, для непосвященных: foo//.a_:>Tr@IntegerDigits@a это повторяющаяся замена: она заменяет любые числа aв foo суммой их цифр, затем она заменяет снова, пока не достигнет фиксированной точки, то есть aперестает меняться при замене.

тесты:

f /@ {"13579", "Hello,world!", "00 0 00", "!@#$%^&*()", "3.141592","3.1415926535897932384"}
     => {7, 7, 0, "*", 7, 7}
f /@ {"Bob", "Charlie", "Anna", "Fana", "Gregory", "Denis", "Erik",  "Helen", "Izis"}
     => {1, 2, 3, 4, 5, 6, 7, 8, 9}
hYPotenuser
источник
К сожалению, я не смог проверить ваш образец. Пожалуйста, опубликуйте тестовые случаи
Ф. Хаури
По пути. Собирая тесты, я также нашел несколько опечаток, которые нужно исправить. Спасибо;)
hYPotenuser