unique_name.py 6.0 KB
Newer Older
Y
Yu Yang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14
#   Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

15 16
from __future__ import print_function

Y
Yu Yang 已提交
17
import collections
S
rename  
sneaxiy 已提交
18
from .wrapped_decorator import signature_safe_contextmanager
19
import six
Y
Yu Yang 已提交
20 21
import sys

Y
yuyang18 已提交
22
__all__ = ['generate', 'switch', 'guard']
Y
Yu Yang 已提交
23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56


class UniqueNameGenerator(object):
    """
    Generate unique name with prefix.

    Args:
        prefix(str): The generated name prefix. All generated name will be
                     started with this prefix.
    """

    def __init__(self, prefix=None):
        self.ids = collections.defaultdict(int)
        if prefix is None:
            prefix = ""
        self.prefix = prefix

    def __call__(self, key):
        """
        Generate unique names with prefix

        Args:
            key(str): The key of return string.

        Returns(str): A unique string with the prefix
        """
        tmp = self.ids[key]
        self.ids[key] += 1
        return self.prefix + "_".join([key, str(tmp)])


generator = UniqueNameGenerator()


Y
Yu Yang 已提交
57
def generate(key):
58
    """
59 60 61
    Generate unique name with prefix key. Currently, Paddle distinguishes the
    names of the same key by numbering it from zero. For example, when key=fc,
    it continuously generates fc_0, fc_1, fc_2, etc.
62

63 64
    Args: 
        key(str): The prefix of generated name.
65 66 67 68

    Returns: 
        str: A unique string with the prefix key.

69 70
    Examples: 

71 72 73 74 75
        .. code-block:: python

            import paddle.fluid as fluid
            name1 = fluid.unique_name.generate('fc')
            name2 = fluid.unique_name.generate('fc')
76
            print(name1, name2) # fc_0, fc_1
77
    """
Y
Yu Yang 已提交
78
    return generator(key)
Y
Yu Yang 已提交
79 80


81 82
# FIXME(zjl): The previous naming rule in static graph would
# cause memory leak in dygraph mode. It is because the previous
Z
Zeng Jinle 已提交
83
# naming rule would use `conv_0.tmp` as the key, and in dygraph
84 85 86 87 88 89 90
# mode, `conv_i` increases as batch increases. Thus, keys would
# increase in a way like `conv_0.tmp`, `conv_1.tmp`, .... 
# Not find a better way to fix this bug in dygraph mode. In TF,
# variable name is meaningless in eager execution mode, and in
# PyTorch, there is no variable name at all. Maybe we should
# discard variable name in dygraph mode.
#
Z
Zeng Jinle 已提交
91
# Another concern is that save/load interfaces. Usually, user
92 93 94 95 96 97 98 99 100 101 102 103
# would save model in static graph mode, and load it in dygraph
# mode. Therefore, we keep the variable name of Parameter currently.
# 
# Please fix me if a better method is found.        
def generate_with_ignorable_key(key):
    from .framework import in_dygraph_mode
    if in_dygraph_mode():
        key = "tmp"

    return generator(key)


Y
Yu Yang 已提交
104
def switch(new_generator=None):
105
    """
106 107 108 109
    Switch the namespace of in current context to a new namespace. Though
    :code:`switch()` and :code:`guard()` can both change namespace, 
    :code:`guard()` is recommended since it can manage the context better 
    together with :code:`with` statement.
110

111 112 113 114
    Args: 
        new_generator(UniqueNameGenerator, optional): A new UniqueNameGenerator, not
            required normally. Default is None, which means switch to a new anonymous
            namespace.
115 116 117 118

    Returns: 
        UniqueNameGenerator: The previous UniqueNameGenerator.

119 120
    Examples: 

121 122 123 124 125
        .. code-block:: python

            import paddle.fluid as fluid
            name1 = fluid.unique_name.generate('fc')
            name2 = fluid.unique_name.generate('fc')
126
            print(name1, name2) # fc_0, fc_1
127

128
            pre_generator = fluid.unique_name.switch() # switch to a new anonymous namespace.
129
            name2 = fluid.unique_name.generate('fc')
130 131 132 133 134 135
            print(name2) # fc_0

            fluid.unique_name.switch(pre_generator) # switch back to pre_generator.
            name3 = fluid.unique_name.generate('fc')
            print(name3) # fc_2, since pre_generator has generated fc_0, fc_1.

136
    """
Y
Yu Yang 已提交
137 138 139 140 141 142 143 144 145
    global generator
    old = generator
    if new_generator is None:
        generator = UniqueNameGenerator()
    else:
        generator = new_generator
    return old


S
rename  
sneaxiy 已提交
146
@signature_safe_contextmanager
Y
Yu Yang 已提交
147
def guard(new_generator=None):
148
    """
149 150 151 152 153 154 155 156 157
    Change the namespace of unique name with :code:`with` statement. After calling it,
    a new namespace in the context of :code:`with` will be created, and it will number
    names from zero again when calling :code:`generate()` with same key.

    Args: 
        new_generator(str|bytes, optional): New name of global namespace. Note that str
            in Python2 was spilted into str and bytes in Python3, so here are two 
            types. Default is None. If not None, new_generator will be added into 
            the prefix of unique name generated by :code:`generate()`.
158
    
159 160 161 162
    Returns:
        None.

    Examples: 
163 164 165 166 167 168 169 170

        .. code-block:: python

            import paddle.fluid as fluid
            with fluid.unique_name.guard():
              name_1 = fluid.unique_name.generate('fc')
            with fluid.unique_name.guard():
              name_2 = fluid.unique_name.generate('fc')
171
            print(name_1, name_2) # fc_0, fc_0
172 173 174 175

            with fluid.unique_name.guard('A'):
              name_1 = fluid.unique_name.generate('fc')
            with fluid.unique_name.guard('B'):
176 177
              name_2 = fluid.unique_name.generate('fc') 
            print(name_1, name_2) # Afc_0, Bfc_0
178
    """
179
    if isinstance(new_generator, six.string_types):
Y
Yu Yang 已提交
180
        new_generator = UniqueNameGenerator(new_generator)
M
minqiyang 已提交
181 182
    elif isinstance(new_generator, six.binary_type):
        new_generator = UniqueNameGenerator(new_generator.decode())
Y
Yu Yang 已提交
183 184 185
    old = switch(new_generator)
    yield
    switch(old)