Copyright © 2004 W3C® (MIT, ERCIM, Keio), All Rights Reserved. W3C liability, trademark, document use and software licensing rules apply.
Representing part-whole relations is a very common issue for those
developing ontologies for the Semantic Web. OWL does not provide any
built-in primitives for part-whole relations (as it does for the
subclass relation), but contains sufficient expressive
power
to capture most, but not all, of the common cases. The study of
part-whole
relations is an entire field in itself - "mereology" - this note is
intended
only to deal with straightforward cases for defining classes involving
part-whole relations.
This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at http://www.w3.org/TR/.
This document will be a part of a larger document that will provide an introduction and overview of all ontology design patterns produced by the Semantic Web Best Practices and Deployment Working Group.
This document is a W3C Working Draft and is expected to change. The SWBPD WG does not expect this document to become a Recommendation. Rather, after further development, review and refinement, it will be published and maintained as a WG Note.
As a candidate Public Working Draft, we encourage public comments. Please send comments to public-swbp-wg@w3.org
Open issues, todo items:Publication as a draft does not imply endorsement by the W3C Membership. This document is a draft and may be updated, replaced or made obsolete by other documents at any time. It is inappropriate to cite this document as other than work in progress.
Many applications require representation of part-whole relations - catalogues of parts, fault diagnosis, anatomy, geography, etc. The study of part-whole relations is a large field in its own right - "mereology" and "mereotopology" and has been the topic of many papers, see the references section for a useful list.
RDF Schema and OWL does not contain specific primitives for part-whole relations (as it does for the subclass relation, for example), but it does support sufficient machinery to express most of what one may want to represent about part-whole relations. Where it does not, there are a number of "work-arounds" that suffice in most situations. This note will provide basic schemas for expressing part-whole relations in RDF Schema and OWL.
In many applications, what is needed is not a list of all parts but
rather
a list of the next level breakdown of parts, the "direct parts" of a
given
entity. It is therfore often useful to use the property hierarchy to
define
a subproperty of partOf
that
is not
transitive and
links each subpart just to the next level. For these examples we shall
call
this suproperty partOf_directly.
partOf
,
say hasPart
. For any two
individuals I1
and I2
, if
"I1 partOf I2"
then "I2
hasPart I1"
. However, care must be taken when using inverses
in
restrictions on classes. To say that "All As are parts of some B" does
not
imply that "All Bs have some As as parts", i.e. the restrictionTherefore, if we want to say both that "all As are parts of Bs" and "all Bs have part some A", we have to assert each statement separately. Such pairs of statements are sometimes called "reciprocals".
Unfortunately, all current OWL reasoners scale very badly for large
part-whole hierarchies connected by both
hasPart and partOf. Therefore,
if
reasoners are to be used, it is usually necessary to choose to use
either partOf
or hasPart
but not both.
Almost
always it is preferable to use
partOf because the most common queries and class definitions
are
for the parts of things, e.g. the class of all parts of a car.
Parts and wholes are ubiquitous in many applications:
We can define hasPart
as any other property in RDF Schema or OWL.
Just as an individual car (car123) can have a model (Porsche 911) and an owner
(Chris), it can have parts (motor, headlights, wheels). In summary, we perform
the following steps:
partOf
. partOf
(e.g.,
a class Car
and a class CarPart
respectively).
partOf
transitive. Consider a (over simplified) description of an individual car (an instance
of the class Car
):
id: car123
owner: Chris (instance of class
Person
)parts: motor123 (instance of class
Motor
), headlight123a and headlight123b (instances of classHeadlight
), wheel123a, wheel123b, wheel123c, wheel123d (instances of classWheel
)
The classes Motor
, Headlight
, Wheel
are subclasses of a generic class CarPart
.
Note the the representation of individuals in this pattern is the same for RDF Schema and OWL. The definition of the classes and properties is different for the two languages. Therefore, we present the two version of the definitions for classes and properties and then show the definition for individuals.
partOf a rdf:Property ; rdfs:domain CarPart ; rdfs:range Car . Car a rdfs:Class ; rdfs:subClassOf rdfs:Resource . Motor a rdfs:Class ; rdfs:subClassOf CarPart . ...
partOf
a owl:TransitiveProperty , owl:ObjectProperty ;
owl:inverseOf hasPart . Car
a owl:Class ;
rdfs:subClassOf owl:Thing ;
rdfs:subClassOf
[ a owl:Restriction ;
owl:onProperty hasPart ;
owl:allValuesFrom CarPart
] .
CarPart
a owl:Class ;
rdfs:subClassOf owl:Thing ;
rdfs:subClassOf
[ a owl:Restriction ;
owl:onProperty partOf ;
owl:allValuesFrom Car
] .
...
car123 a Car ; hasModel "Porsche 911" ; hasOwner Chris . motor123
a Motor ; partOf car123 . headlight123a a Headlight ; partOf car123 . ...
In the example above, we describe individual parts of an individual car. Note
that we do not say anything about whether cars in general have motors or wheels.
If we need to describe composition at the level of a whole class of objects
(e.g., all cars have motors), we need to use the partOf
relation
at the class level, as in the following patterns.
partOf
with an inverse hasPart
(Choose your own naming conventions to suit). partOf_directly
with an inverse hasPart_directly.
partOf
or hasPart
relation
as the basic relation amongst classes. If in doubt, choose partOf
.hasValue()
with partOf_directly
(if defined, otherwise with partOf
).hasSomeValuesFrom()
with partOf_directly
allValuesFrom
) constraints, add
those.There should now be sufficient information to make basic inferences about parts, e.g. to define a class of all parts of the car, car door, etc
Consider a (over simplified) catalog of Vehicle parts, all subsumed by the class Item
Cars have parts Motor, Headlight, Wheel
Motors have parts Crankcase, Carburetor
Headlights have parts head_light_bulb, reflector
A fragment of the N3 for the above example would then be:
partOf
a owl:TransitiveProperty , owl:ObjectProperty ;
owl:inverseOf hasPart .
partOf_directly
a owl:ObjectProperty ;
rdfs:subPropertyOf partOf ;
owl:inverseOf hasPart_directly .
Car
a owl:Class ;
rdfs:subClassOf Item
Motor
a owl:Class ;
rdfs:subClassOf Item ;
rdfs:subClassOf
[ a owl:Restriction ;
owl:onProperty partOf_directly ;
owl:someValuesFrom Car
] .
Crankcase
a owl:Class ;
rdfs:subClassOf Item ;
rdfs:subClassOf
[ a owl:Restriction ;
owl:onProperty partOf_directly ;
owl:someValuesFrom Motor
] .
...
etc.
Several issues arise even from such a simple example. To begin with, the representation using existential restrictions (i.e. owl:someValuesFrom) does not clearly communicate all of the semantics we may want for car parts. For example, a strict reading of the definition of the Crankcase class above is that a crankcase is part of at least one motor. In point of fact, a crankcase cannot be part of more than one motor. We may be tempted to add a cardinality restriction (e.g. maxCardinality 1) on partOf to the definition of crankcase, but this would be a mistake; since partOf is transitive, a crankcase is also part of the car the motor is part of. Note also that OWL-DL does not allow transitive properties to have any cardinality restrictions. In general it is best to avoid placing restrictions (including range restrictions) on transitive properties at all.
It would make more sense to add a restriction on the partOf_directly
property in the definition of these classes, when it is
appropriate. A single crankcase cannot be a direct part of more than one
motor, a motor cannot be a direct part
of more than one car, etc., so in these cases a maxCardinality
restriction would make the semantics more clear. On the other
hand, there is always a tradeoff when employing a reasoner between how
precise your semantics are and how much information the reasoner has to
consider. In this case, adding a cardinality restriction on all
the partOf_directly properties would significantly increase the amount
of information handed to a reasoner. One must consider precisely
what the ontology will be used for to determine which is more important
(enforcing semantic constraints vs. classification). The examples in this note are aimed primarily at
use-cases in which no instances of the classes are present.
When considering restrictions on the partOf_directly property for
different kinds of parts, the issue of using a universal
(owl:allValuesFrom) vs. an existential restriction arises. Many
different kinds of things have motors (boats, planes, etc.), and
in fact even car motors can exist without being part of a car.
This indicates that, ontologically, the existential restriction is
simply not true We will discuss this further in the subsequent
sections.
Extending the ontology in pattern 1, we can define the classes:
CarPart
and CarPart_directly
.
Informally:
CarPart = partOf someValuesFrom(Car)
CarPart_directly = partOf_directly someValuesFrom(Car)
In N3:
CarPart_directly
a owl:Class ;
owl:equivalentClass
[ a owl:Restriction ;
owl:onProperty partOf_directly ;
owl:someValuesFrom Car
] .
CarPart
a owl:Class ;
owl:equivalentClass
[ a owl:Restriction ;
owl:onProperty partOf ;
owl:someValuesFrom Car
] .
A classifier could then infer that CarPart_directly
subsumes
Motor
Headlight
Wheel
and that CarPart
subsumes
Motor
Crankcase
Carburator
Headlight
Headlight_bulb
Reflector
Wheel
This simple list may not be what we want, in which case it is necessary systematically to define a class for the parts of each part, e.g.
MotorPart a owl:Class ; owl:equivalentClass [ a owl:Restriction ; owl:onProperty partOf ; owl:someValuesFrom Motor ] .If all are defined in this way we get a hierarchy from the classifier:
CarPart Motor MotorPart Headlight HeadlightPart Headlight_bulb ...
These classes exemplify one of the main reasons to choose
existential restrictions on the direct part properties over universal
restrictions (as discussed in the previous pattern). A classifier
would not be able to infer the hierarchy above using universal
restrictions on the partOf_direct property in the first pattern, unless
there were minimum cardinality restrictions on the property as well.
Ontologically, these classes by themselves are reasonable, a "car
part" is indeed anything that is part of a car, however when combined
with the existential restrictions on the direct properties, a
classifier would infer the hierarchy above. These kinds of
hierarchies seem harmless at first glance, but in some contexts are
completely wrong: not all motors are car parts, some are boat motors,
etc. On the other hand, a motor for a 1969 Porsche 911E is
generally considered a "car part" regardless of whether it is in a car
or not (it may be for sale). When this is not obvious from the scope
of the ontology, it is good practice to reflect these issues in the names
of the class, e.g. CarHeadlight.
subclassOf
generate
hierarchies, it is important not to confuse the part-whole hierarchy
with the
subclassOf
hierarchy. This is easily done because in
many
library and related applications, part-whole and subclass relations are
deliberately conflated into a single "broader than / narrower than"
axis. For
example consider the following:
Vehicle
Car
Wheel
Tire
Pneumatic tire
"Car" is a kind of "Vehicle", but "Wheel" is a part of a "Car", "Tire" is a part of a "Wheel", but "Pneumatic tire" is a kind of "Tire". Such hierarchies serve well for navigation, however they are not in general true. Statements about "all vehicles" do not necessarily, or even probably, hold for "all tires".
However, such hierarchies do need to be recreated in situations that
obey
the rule "A fault of the part is a kind of fault of the whole". For
example
you can call for assistance with a fault in your
car if
you puncture a pneumatic tire. The following hierarchy is a correct
subclassOf
or "kind of" hierarchy of a type that we need
to
reproduce often in OWL:
Fault in CarThe easy way to say that what we really mean when we talk about a "fault in a car" is a "fault in a car or any of its parts" [1]: If we use the property
Fault in Wheel
Fault in Tire
Fault in Pneumatic tire
hasLocus
to locate the fault in a particular
part of the car, then:This may look tedious, but
can
actually be achieved quite simply with scripting tools or the ability
to
"clone and edit" classes easily.
In certain domains, most notably medicine, we generally understand
that while body parts (e.g. a heart) can exist outside of a body, they do
not normally do so. Thus it makes sense to say, in general, "A
fault in the heart is a fault in the body," without having a particular
heart or body in mind, and it makes sense to reason over classes
defined that way. For other domains, most notably manufacturing,
it is more common for parts to exist outside of some whole, and so it
may not generally be true that a fault in an engine is a fault in a car
(if the engine is not in a car), just as it may not be generally true
that an engine is a car part. In these cases, the capability to
reason over classes may not be that useful, and again the existential
restriction on the direct properties may not make sense.
CarPart_reflexive = Car OR CarPart
In N3:
CarPart_reflexiveWhen reflexive parts are defined, it simplifies the definition of faults (in fact, this is often used as a logical argument for why the partOf relation is reflexive):
a owl:Class ;
owl:equivalentClass
[ a owl:Class ;
owl:unionOf (Car [ a owl:Restriction ;
owl:onProperty partOf ;
owl:someValuesFrom Car
])
] .
Fault_in_car = Fault AND hasLocus someValuesFrom(CarPart_reflexive)
In N3:
Fault_in_car
a owl:Class ;
owl:equivalentClass
[ a owl:Class ;
owl:intersectionOf (Fault [ a owl:Restriction ;
owl:onProperty hasLocus ;
owl:someValuesFrom CarPart_reflexive
])
].
Logically these classes do not give us reflexivity at all, a
reflexive property is one that holds between an object and itself, not
between an object and something in the same class. It is not
possible in OWL to state such a restriction or inference, however, and
when reasoning only over the classes and properties in an ontology this
will suffice.
A number of other relations follow the same pattern as faults, e.g. "Repairs on a part are kinds of repairs on the whole". However, not all relations follow this pattern, e.g. "Purchase of a part is not purchase of the whole"
The classic study of parts and wholes, mereology, has three
axioms:
the part-of relation is
Furthermore, in mereology, since everything is a part of itself, we have to define "proper parts" as "parts not equal to the whole". Whereas in OWL we have to do the reverse: i.e. define "parts" (analogous to "proper parts") and then define "reflexive parts" in terms of "parts".
There are a number of relations easily confused with part-whole relations. Interested readers should consult [Flavours of part of]. However, a brief list includes:
partOf
and hasPart
In some contexts it is "more universal" to use partOf,
in
others to use hasPart.
For example, all cars have
wheels, but
not all wheels are parts of cars. On the other hand, all leaves are
parts of
plants (at least at some time), but not all plants have leaves. The
inability
of existing classifiers to cope with ontologies mixing partOf
and hasPart
is a significant
limitation. In the usual case where partOf
is used,
the best
option may then be not to enter a saying that "all wheels are parts of
cars"
but rather to define notions of as "Wheel of a car".
Wheel_of_car
a owl:Class ;
owl:equivalentClass
[a owl:Class ;
owl:intersectionOf (Wheel [a owl:Restriction ;
owl:onProperty partOf ;
owl:someValuesFrom Car ])
]).
By defintion, all Wheel_of_car
s are parts of cars,
and any
Wheel
that is a part of a car is a Wheel_of_car.
This is a work around, and not ideal, but at least all statements in
the
ontology are logically true.
partOf,
but the various flavours of part-whole
relation are
beyond the scope of this note. See [Flavours
of part of].