Próbuję skomponować seq-m i error-m, aby wykonać listę rzeczy na temat rzeczy, które mogą zwracać błędy. Moje wyniki mają nieoczekiwane typy, ale poza tym wydaje się, że są sensowne. Mój kod został poniżej eksplodować, ale tutaj jest również working gist.łączenie być może i kolejnych monad: pomieszanie na wyjściu
tutaj jest mój monadycznego logika biznesowa
def get_loan(name):
m_qualified_amounts = (
bind(get_banks(name), lambda bank:
bind(get_accounts(bank, name), lambda account:
bind(get_balance(bank, account), lambda balance:
bind(get_qualified_amount(balance), lambda qualified_amount:
unit(qualified_amount))))))
return m_qualified_amounts
names = ["Irek", "John", "Alex", "Fred"]
for name, loans in zip(names, map(get_loan, names)):
print "%s: %s" % (name, loans)
wyjście
Irek: [None, 'Insufficient funds for loan, current balance is 35000', None, 'Insufficient funds for loan, current balance is 70000', None, 'Unable to get balance due to technical issue for Wells Fargo: 3']
John: [None, 'Insufficient funds for loan, current balance is 140000']
Alex: [[245000], None, [280000], None]
Fred: (None, 'No bank associated with name Fred')
spodziewam się zobaczyć listę krotek - lista jest wynikiem listowego, a każda pozycja na ostatecznej liście powinna być wartością w monadzie błędu (value, error
krotka). Dokładnie tak, jakby jeden zbyt duży poziom zagnieżdżenia został usunięty przez seq_bind
.
oto moja definicja monad, która jeśli nie jest poprawna, jest bardzo bliska, ponieważ obie monady działają w izolacji, po prostu nie łączone.
def success(val): return val, None
def error(why): return None, why
def get_value(m_val): return m_val[0]
def get_error(m_val): return m_val[1]
# error monad
def error_unit(x): return success(x)
def error_bind(mval, mf):
assert isinstance(mval, tuple)
error = get_error(mval)
if error: return mval
else: return mf(get_value(mval))
def flatten(listOfLists):
"Flatten one level of nesting"
return [x for sublist in listOfLists for x in sublist]
# sequence monad
def seq_unit(x): return [x]
def seq_bind(mval, mf):
assert isinstance(mval, list)
return flatten(map(mf, mval))
# combined monad !!
def unit(x): return error_unit(seq_unit(x))
def bind(m_error_val, mf):
return error_bind(m_error_val, lambda m_seq_val: seq_bind(m_seq_val, mf))
monadycznego API
def get_banks(name):
if name == "Irek": return success(["Bank of America", "Wells Fargo"])
elif name == "John": return success(["PNC Bank"])
elif name == "Alex": return success(["TD Bank"])
else: return error("No bank associated with name %s" % name)
def get_accounts(bank, name):
if name == "Irek" and bank == "Bank of America": return success([1, 2])
elif name == "Irek" and bank == "Wells Fargo": return success([3])
elif name == "John" and bank == "PNC Bank": return success([4])
elif name == "John" and bank == "Wells Fargo": return success([5, 6])
elif name == "Alex" and bank == "TD Bank": return success([7, 8])
else: return error("No account associated with (%s, %s)" % (bank, name))
def get_balance(bank, account):
if bank == "Wells Fargo":
return error("Unable to get balance due to technical issue for %s: %s" % (bank, account))
else:
return success([account * 35000]) #right around 200,000 depending on acct number
def get_qualified_amount(balance):
if balance > 200000:
return success([balance])
else:
return error("Insufficient funds for loan, current balance is %s" % balance)
również szuka sposobów, aby poprawić kod. otagował haskell i clojure, ponieważ jest to idiomatyczne w tych językach, społeczność Pythona nie jest tym zainteresowana.
Oto jedna pythonista, która jest zainteresowana. Co się stało z twoim blogiem, Dustinem? –