Генерация чисел Фридмана

9

Фридман число это число , которое может быть выражено путем применения основных математических операций (^, /, *, +, -) для всех его цифр. Операции не должны применяться к каждой отдельной цифре, но все цифры должны быть включены. То есть 121 = 11 ^ 2 -> все цифры задействованы, но 1 и 1 были объединены, чтобы получить 11.

Использование скобок разрешено, но тривиальное решение x= (x)не является допустимым. Также не действует x= +x.

Примеры

  • 25 = 5 ^ 2
  • 121 = 11 ^ 2
  • 343 = (3 + 4) ^ 3
  • 2048 = (8 ^ 4) / 2 + 0

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

Вход -

n m    | n, m integers, n>=0, m>n

Вывод -

count    | number of Friedman numbers in the given range
fn1 exp1 | Friedman number, expression
fn2 exp2
fn3 exp3
.
.
.

Самый короткий код, опубликованный в воскресенье 29 июля 00:00 по Гринвичу, станет победителем.

elssar
источник
2
Можете ли вы добавить пример числа Фридмана и объяснить, как /работает? Например, что это 1/3?
JPvdMerwe
Число выражается путем применения операций ко всем его цифрам. то есть 25 = 5 ^ 2, 126 = 6 * 21, 343 = (3 + 4) ^ 3 и т. д.
Эльссар
Разрешаете ли вы одинарный минус? например -5?
JPvdMerwe
@JPvdMerwe проверим спецификацию ввода, вам не нужно этого делать, но если вы хотите, вырубитесь. Хотя унарный плюс не допускается. т.е. +5 не является правильным решением
elssar
1
Вы не ответили на вопрос JPvdMerwe о разделении. Это должно быть точно? Могут ли промежуточные результаты быть нецелыми?
Питер Тейлор

Ответы:

3

Рубин, 456 438 408 390 370 349 344 334 [исправлено]

g={}
f=->a,b{a.permutation(b).to_a.uniq.flatten.each_slice b}
F,T=$*
([F.to_i,10].max..T.to_i).map{|c|f[a="#{c}".split(''),v=a.size].map{|m|f[[?+,?-,?*,?/,'','**'],v-1].map{|w|(d=(s=m.zip(w)*'').size)==v&&next
0.upto(d){|y|y.upto(d+1){|u|begin(r=eval t="#{s}".insert(y,?().insert(u,?)))==c&&g[r]=t
rescue Exception
end}}}}}
p g.size,g

Вывод:

% ruby ./friedman-numbers.rb 1 300
9
{25=>"(5)**2", 121=>"(11)**2", 125=>"5**(2+1)", 126=>"(6)*21", 127=>"(2)**7-1", 128=>"2**(8-1)", 153=>"(3)*51", 216=>"6**(1+2)", 289=>"(9+8)**2"}

Также это работает относительно быстро для больших чисел:

% time ruby friedman-numbers.rb 3863 3864   
1
{3864=>"(6**4-8)*3"}
ruby friedman-numbers.rb 3863 3864  14.05s user 0.17s system 99% cpu 14.224 total
defhlt
источник
1
Я запустил его с входом 5 40и получил результат: [11, "11**1", 21, "21**1", 31, "31**1", 41, "41**1"]. Никаких признаков 25там нет, и я думаю, что правильного решения (например, для 21) 2*1нет,21**1
Кристиан Лупаску
@ w0lf Спасибо! Я думаю, что я это исправил.
defhlt
Да, теперь это прекрасно работает.
Кристиан Лупаску,
@ w0lf добавил много символов для форматирования вывода по мере необходимости
defhlt
Вы можете получить 2 символа, заменив '+-*/'.chars.to_a+['','**']на["+","-","*","/","","**"]
Кристиан Лупаску
4

Python 2,7 - 380 378 372 371 367 363 357 354 352 348 336 символов

Просто простой перебор.

from itertools import*
s=lambda x:[x]['1'>x>'0':]+['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))]
def E(e):
 try:return eval(e.replace("^","**"))
 except:0
A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}
print len(A)
for v in A:print v,A[v]

Пример выполнения:

1
300
9
128 (2^(8-1))
289 ((9+8)^2)
216 (6^(1+2))
121 (11^2)
153 (3*51)
25 (5^2)
125 (5^(2+1))
126 (6*21)
127 ((2^7)-1)

Объяснение:

s(x) является функцией, которая принимает строку, содержащую последовательность цифр, и возвращает все выражения, использующие эти цифры в указанном порядке.

[x]['1'>x>'0':] вычисляет список, содержащий x, если x равен '0' или последовательность цифр не начинается с '0'; в противном случае он оценивается как пустой список. В основном это обрабатывает случай, когда я соединяю все цифры вместе.

['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))] в основном разбивает x на две части (обе имеют ненулевую длину), вызывает s () для каждой части и объединяет все результаты вместе с некоторым оператором между ними, используя product ().

E(e) в основном безопасный eval. Возвращает значение e, если e допустимо, и None в противном случае.

A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}

По сути, этот код пробует все числа в диапазоне, переставляет их цифры и проверяет каждое выражение, сгенерированное s () для этой перестановки, игнорируя первое выражение, если x не начинается с «0», потому что если x не начинается с « 0 'тогда первое выражение будет просто х.

Альтернативная версия - 397 символов

Вот мой код, если вам необходимо использовать дроби:

from fractions import*
from itertools import*
s=lambda x:["Fraction(%s)"%x]['1'>x>'0':]+['(%s%s%s)'%f for i in range(1,len(x))for f in product(s(x[:i]),'*/-+^',s(x[i:]))]
def E(e):
 try:return eval(e.replace("^","**"))
 except:0
A={i:e for i in range(input(),input()+1)for x in permutations(`i`)for e in s("".join(x))[x>='1':]if E(e)==i}
print len(A)
for v in A:print v,A[v].replace("Fraction","")
JPvdMerwe
источник
Я не думаю, if len(x)<2что когда-нибудь будет правдой в функции s. Кроме того , можно заменить formatс , "a[Fraction(%s)%s%s]='(%s%s%s)'"%(x[:i],o,v,x[:i],o,A)чтобы сохранить 4 символов.
beary605
@ beary605: Иногда это правда, когда i = len (x) -1, тогда при следующем вызове будет один символ. Что касается второго пункта, спасибо! :)
JPvdMerwe
да ... except:0умный .. очень умный. Я буду помнить
Ev_genus
Пожалуйста, включите некоторые иллюстративные результаты.
DavidC
1
Нет, все еще работает. Я должен переместить свой компьютер сейчас, но я позволю ему поработать несколько дней и посмотрим, завершится ли он.
JPvdMerwe
3

Python3 (436) (434) (443)

Это было сложно. Я могу сэкономить некоторые символы, если я сделаю вывод более родным.

from itertools import*
r={};k=product;m=map
q=lambda n,h=1:["("+i+c+j+")"for(i,j),c in k(chain(*[k(*m(q,f))for f in sum(([(x[:q],x[q:])for q in range(1,len(x))]for x in m("".join,permutations(n))),[])]),list("+-*/^")+[""]*h)]if 1<len(n)else[n]*h
a,b=m(int,m(input,"nm"))
for i,j in chain(*[k(q(str(n),0),[n])for n in range(a,b+1)]):
    try:exec("if eval(%r)==j:r[j]=i"%i.replace("^","**"))
    except:0
print(len(r))
for j,i in r.items():print(i,j)

Вывод

n100
m200
6
(2^(8-1)) 128
(3*(51)) 153
((11)^2) 121
(5^(1+2)) 125
(6*(21)) 126
((2^7)-1) 127
Ev_genus
источник
1
Таким образом, у вас есть много умных трюков; Тем не менее, я должен отметить, что вы неправильно обрабатываете от 1 до 9, и ваш вклад не является исчерпывающим. Вы можете удалить 2 символа, удалив пробел после "("+i+c+j+")"и заменив len(n)>1, 1<len(n)после которого вы можете удалить пробел после этого выражения.
JPvdMerwe
Справедливая. Исправлено все, +7 символов
Ev_genus
Вы можете заменить последнюю строку, for j in r:print(r[j],j)чтобы сохранить 7 символов.
JPvdMerwe
1

Mathematica 456 416 402 404 400 396 символов

<< Combinatorica`; l = Length; p = Permutations; f = Flatten; c = Cases;
u[d_, o_, s_] := 
 Fold[#2[[1]] @@ If[s == 1, {#1, #2[[-1]]}, {#2[[-1]], #1}] &, 
 d[[1]], Thread@{o, Rest@d}];
q[t_, r_] := {u[t, #, r], u[HoldForm /@ t, #, r]} & /@ 
p[{Plus, Subtract, Times, Divide, Power}, {l@t - 1}];
v[m_, n_] := (t = Table[Union@
  c[f[{#~q~1, #~q~0} & /@ 
     f[p /@ c[
        FromDigits /@ # & /@ 
         f[SetPartitions /@ p@IntegerDigits@j, 1], x_ /; l@x > 1],
       1], 2], {j, _}], {j, m, n}]~f~1; {l@t}~Join~t)

Пример :

v[1,300]//TableForm

Выход :

Фридман выход

DavidC
источник