What Is The Recommended Way To Mix TensorFlow And TensorFlow Federated Code?
Solution 1:
Great question. Indeed, there are at least 3 ways to approach composition of TensorFlow code for use with TFF, each with its own merits.
- Using TensorFlow's compositional mechanism (defuns) is the recommended way, assuming it works for your specific situation. TensorFlow already has mechanisms for composing code, and we don't want to reinvent the wheel. The reason we've created our own compositional mechanism in TFF (@tff.tf_computation) was to deal with specific limitations (such as the lack of support for data sets at the interface level in TF, and the need for TF components to interoperate with the rest of TFF), and we'd ideally limit the use of this mechanism to just the situations that really call for it.
When possible, decorate TensorFlow components using @tf.function, and wrap the entire TensorFlow block as a @tff.tf_computation only at the top level, before embedding it in a @tff.federated_computation. One of the many benefits of this is that it allows you to test components outside of TFF, using standard TensorFlow tools.
So, the following is encouraged and preferred:
# here using TensorFlow's compositional mechanism (defuns)
# rather than TFF's to decorate "foo"
@tf.function(...)
def foo(...):
...
@tff.tf_computation(...)
def bar(...):
# here relying on TensorFlow to embed "foo" as a component of "bar"
...foo(...)...
- Using Python's compositional mechanism (plain undecorated Python functions) is also a good option, although it's less preferable than (1), since it just causes one body of code to be embedded within the other at the definition time as TFF traces through all the TFF-decorated Python functions to construct a serialized representation of the computation to execute, without offering you isolation or any other special benefits.
You may still want to use this pattern to allow your components to be tested outside of TFF, or in situations where neither (1) or (3) works.
So, the following is an alternative you should consider first if (1) doesn't work:
# here composing things in Python, no special TF or TFF mechanism employed
def foo(...):
# keep in mind that in this case, "foo" can access and tamper with
# the internal state of "bar" - you get no isolation benefits
...
@tff.tf_computation(...)
def bar(...):
# here effectively just executing "foo" within "bar" at the
# time "bar" is traced
...foo(...)...
- Using TFF's compositional mechanism (@tff.tf_computation) is not recommended, except - as noted above - in situations that require it, such as when a TensorFlow component needs to accept a data set as a parameter, or if it's going to be invoked only from a @tff.federated_computation. Keep in mind that TFF's support for data sets as parameters is still experimental, and while in some cases it may be the only solution, you may still run into problems. You can expect the implementation to evolve.
Not encouraged (although currently sometimes necessary):
# here using TFF's compositional mechanism
@tff.tf_computation(...)
def foo(...):
# here you do get isolation benefits - "foo" is traced and
# serialized by TFF, but you can expect that e.g., some
# tf.data.Dataset features won't work
...
@tff.tf_computation(...)
def bar(...):
# here relying on TFF to embed "foo" within "bar"
...foo(...)...
Post a Comment for "What Is The Recommended Way To Mix TensorFlow And TensorFlow Federated Code?"