Skip to main content

simple functional fuzzy rules implementation

Project description

Build Status Coverage Status

frules - simple functional fuzzy rules

Frules stands for fuzzy/funtional rules. It allows to work easily with fuzzy rules and variables.

Installation:

pip install frules

Linguistic variables and expressions

Expression is a core concept in frules. Expression class represents subrange of linguistic variable in fuzzy logic.

Variables in classical math take numerical values. in fuzzy logic, the linguistic variables are non-numeric and are described with expressions. Expressions map continuous variable like nemerical temperature to its linguistic counterpart. For example temperature can be described as cold, warm or hot. There is no strict boundary between cold and warm - this is why this expressions are fuzzy.

To create new expression we use function that takes numerical value of contiunous variable and returns truth value. Truth value ranges between 0 and 1 - it's a degree of membership of continous value to that linguistic variable.

from frules.expressions import Expression
#We know that anything over 50 degrees is hot and below 40 is't hot
hot = Expression(lambda x: min(1, max((x - 40) / 10., 0)))

This ugly lambda is representation of some fuzzy set. If we take a look how it behaves, we'll see that it in fact returns 1 for anything over 50, 0 for anything below 40 and some linear values between 40 and 50:

>>> map(lambda x: {x: min(1, max((x - 40) / 10., 0))}, xrange(35, 55, 2))
[{35: 0}, {37: 0}, {39: 0}, {41: 0.1}, {43: 0.3}, {45: 0.5}, {47: 0.7}, {49: 0.9}, {51: 1}, {53: 1}

Using a lot of lambdas in practice makes your code a mess. Fuzzy expressions described this way are additionally hard to write because of some value assertions they must satisfy.

This is why we ancapsulate don't use raw functions and encapsulate them with expressions. Moreover frules provides a bunch of helpers that eases definition of new expressions. Example of full set of expressions for temperature variable could look this way:

from frules.expressions import Expression as E
from frules.expressions import ltrapezoid, trapezoid, rtrapezoid

cold = E(ltrapezoid(10, 20), "cold")        # anything below 10, more is fuzzy
warm = E(trapezoid(10, 20, 30, 35), "warm") # anything between 20 and 30
hot = E(rtrapezoid(30, 35), "hot")          # anything over 35, less is fuzzy

Expressions can be reused/mixed using logical operators:

cold_or_hot = cold || warm
not_hot = !hot

Optional names will be helpful when we start to work with fuzzy rules.

Fuzzy rules

Although expressions define linguistic variables, they aren't strictly bound to any variable. They are rather the adjectives we use to describe something and their meaning depends strictly on context. Both person and data could be big but this particular adjective has slighlty different meaning in each case.

Rule objects bounds continous variable with expressions. Rules also can also be evaluated to see how true they are for given continous input.

>>> from frules.rules import Rule
>>> is_hot = Rule(temperature=hot)
>>> is_hot.eval(temperature=5)
0.8

Rules can be mixed using logical operators (& and |) to create more sophisticated rules that allow fuzzy reasoning:

from frules.expressions import Expression as E
from frules.rules import Rule as R
from frules.expressions import ltrapezoid, trapezoid, rtrapezoid

# car age expressions
old = E(ltrapezoid(2001, 2008), "old")
new = E(rtrapezoid(2013, 2014), "new")
not_so_old = - (old & new)

# power expressions
strong = E(rtrapezoid(50, 100), "strong")
weak = E(ltrapezoid(50, 100), "weak")

# price expression
expensive = E(rtrapezoid(25000, 30000), "expensive")
cheap = - expensive

# yes expression
yes = E(lambda yes: float(yes), "yes") # converts bool to float


# rules
is_attractive = R(production_year=not_so_old) & R(horsepower=strong)
should_buy = is_attractive & R(price=cheap)

Having such set of rules we can do some reasoning:

>>> should_buy
(((age = !(old & new) & horsepower = strong) & !None = None) & cost = !expensive)
>>> should_buy.eval(horsepower=70, production_year=2012, price=15000)
0.4
>>>
>>> candidates = {
...     "car1": {"horsepower": 70, "production_year": 2012, "price": 15000},
...     "car2": {"horsepower": 150, "production_year": 2010, "price": 30000},
...     "car3": {"horsepower": 90, "production_year": 2014, "price": 10000},
...     "car4": {"horsepower": 85, "production_year": 2009, "price": 35000},
... }
>>> max(candidates.iteritems(), key=lambda (key, inputs): is_hot.eval(**inputs))
('car3', {'horsepower': 90, 'price': 10000, 'production_year': 2014})

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distribution

frules-0.2.0-py3-none-any.whl (9.9 kB view details)

Uploaded Python 3

File details

Details for the file frules-0.2.0-py3-none-any.whl.

File metadata

  • Download URL: frules-0.2.0-py3-none-any.whl
  • Upload date:
  • Size: 9.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.2.0 pkginfo/1.6.0 requests/2.24.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.51.0 CPython/3.8.0

File hashes

Hashes for frules-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 7e2be080b0ddef594b88c851872483dc63c0edefd5ec8dfd6a845cc574a2737b
MD5 8a2f1fc6785b5e7c96c62a00bbb97fbc
BLAKE2b-256 dc3ad172dfecefafb7d1f810c923f7c400e47e00ec1d2c03addfe92db039a83f

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page