# Python源码构造网络约束 ## 概述 MindSpore完成从用户源码到计算图的编译,用户源码基于Python语法编写,当前MindSpore支持将普通函数或者继承自nn.Cell的实例转换生成计算图,暂不支持将任意Python源码转换成计算图,所以对于用户源码支持的写法有所限制,主要包括语法约束和网络定义约束两方面。随着MindSpore的演进,这些约束可能会发生变化。 ## 语法约束 ### 支持的Python数据类型 * Number:包括`int`、`float`、`bool`,不支持复数类型。 * String * List:当前只支持append方法;List的更新会拷贝生成新的List。 * Tuple * Dictionary:当前`key`只支持String类型 ### MindSpore扩展数据类型 * Tensor:Tensor变量必须是已定义实例。 ### 表达式类型 | 操作名 | 具体操作 | :----------- |:-------- | 一元操作符 |`+`、`-`、`not`,其中`+`操作符只支持标量。 | 数学表达式 |`+`、`-`、`*`、`/`、`%`、`**`、`//` | `if`表达式 |例如`a = x if x < y else y`。 | 比较表达式 | `>`、`>=`、`<`、`<=`、`==`、`!=` | 逻辑表达式 | `and`、 `or` | `lambda`表达式 | 例如`lambda x, y: x + y`。 | 保留关键字类型 | `True`、`False`、`None` ### 语句类型 | 语句 | 与Python对比 | :----------- |:-------- | `for` | 迭代序列必须是Tuple/List,部分嵌套场景支持。 | `while` | 部分嵌套场景支持。 | `if` | 与Python使用原则一致,但if条件的输入只支持常量。 | `in` | 仅支持Dictionary | `not in` | 仅支持Dictionary | `def` | 相同。 | 赋值语句 | List和Dictionary的多重下标访问不支持作为左值。 ### 系统函数 * len * partial * map * zip * range ### 函数参数 * 参数默认值:目前不支持默认值设为`Tensor`类型数据,支持`int`、`float`、`bool`、`None`、`str`、`tuple`、`list`、`dict`类型数据。 * 可变参数:支持带可变参数网络的推理和训练。 * 键值对参数:目前不支持带键值对参数的函数求反向。 * 可变键值对参数:目前不支持带可变键值对的函数求反向。 ### 操作符 | 运算符 | 支持类型 | :----------- |:-------- | `+` |标量、`Tensor`、`tuple`、`string` | `-` |标量、`Tensor` | `*` |标量、`Tensor` | `/` |标量、`Tensor` | `**` |标量、`Tensor` | `//` |标量、`Tensor` | `%` |标量、`Tensor` | `[]` |操作对象类型支持`list`、`tuple`、`Tensor`,支持多重下标访问作为右值,但不支持多重下标访问作为左值,且索引类型不支持Tensor;Tuple、Tensor类型访问限制见切片操作中的说明。 ### 索引操作 索引操作包含`tuple`和`Tensor`的索引操作。下面重点介绍一下`Tensor`的索引取值和赋值操作,取值以`tensor_x[index]`为例,赋值以`tensor_x[index] = u`为例进行详细说明。其中tensor_x是一个`Tensor`,对其进行切片操作;index表示索引,u表示赋予的值,可以是`scalar`或者`Tensor(size=1)`。索引类型如下: * 切片索引:index为`slice` * 取值:`tensor_x[start:stop:step]`,其中Slice(start:stop:step)与Python的语法相同,这里不再赘述。 * 赋值:`tensor_x[start:stop:step]=u`。 * Ellipsis索引:index为`ellipsis` * 取值:`tensor_x[...]`。 * 赋值:`tensor_x[...]=u`。 * 布尔常量索引:index为`True`,index为`False`暂不支持。 * 取值:`tensor_x[True]`。 * 赋值:暂不支持。 * Tensor索引:index为`Tensor` * 取值:暂不支持。 * 赋值:`tensor_x[index]=u`,`index`仅支持`bool`类型的`Tensor`。 * None常量索引:index为`None` * 取值:`tensor_x[None]`,结果与numpy保持一致。 * 赋值:暂不支持。 * tuple索引:index为`tuple` * tuple元素为slice: * 取值:例如`tensor_x[::, :4, 3:0:-1]`。 * 赋值:例如`tensor_x[::, :4, 3:0:-1]=u`。 * tuple元素为Number: * 取值:例如`tensor_x[2,1]`。 * 赋值:例如`tensor_x[1,4]=u`。 * tuple元素为slice和ellipsis混合情况: * 取值:例如`tensor_x[..., ::, 1:]` * 赋值:例如`tensor_x[..., ::, 1:]=u` * 其他情况暂不支持 另外tuple也支持切片取值操作,`tuple_x[start:stop:step]`,与Python的效果相同,这里不再赘述。 ### 不支持的语法 目前在网络构造函数里面暂不支持以下语法: `break`、 `continue`、 `pass`、 `raise`、 `yield`、 `async for`、 `with`、 `async with`、 `assert`、 `import`、 `await`。 ## 网络定义约束 ### 整网实例类型 * 带[@ms_function](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.html#mindspore.ms_function)装饰器的普通Python函数。 * 继承自[nn.Cell](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.nn.html#mindspore.nn.Cell)的Cell子类。 ### 网络输入类型 * 整网的训练数据输入参数只能是Tensor类型。 * 生成的ANF图里面不能包含这几种常量节点:字符串类型常量、带有Tuple嵌套的常量、带有List嵌套的常量。 ### 网络图优化 在ME前端图优化过程中,会将DataClass类型、Dictionary、List、键值对操作转换为Tuple相关操作。 ### 网络构造组件 | 类别 | 内容 | :----------- |:-------- | `Cell`实例 |[mindspore/nn/*](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.nn.html)、自定义[Cell](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.nn.html#mindspore.nn.Cell)。 | `Cell`实例的成员函数 | Cell的construct中可以调用其他类成员函数。 | 函数 | 自定义Python函数、前文中列举的系统函数。 | dataclass实例 | 使用@dataclass装饰的类。 | Primitive算子 |[mindspore/ops/operations/*](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.ops.operations.html) | Composite算子 |[mindspore/ops/composite/*](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.ops.composite.html) | constexpr生成算子 |使用[@constexpr](https://www.mindspore.cn/api/zh-CN/0.2.0-alpha/api/python/mindspore/mindspore.ops.html#mindspore.ops.constexpr)生成的值计算算子。 ### 其他约束 整网construct函数输入的参数以及使用ms_function装饰器修饰的函数的参数在图编译过程中会进行泛化,不能作为常量输入传给算子使用,如下例所示: * 错误的写法如下: ```python class ExpandDimsTest(Cell): def __init__(self): super(ExpandDimsTest, self).__init__() self.expandDims = P.ExpandDims() def construct(self, input_x, input_axis): return self.expandDims(input_x, input_axis) expand_dim = ExpandDimsTest() input_x = Tensor(np.random.randn(2,2,2,2).astype(np.float32)) expand_dim(input_x, 0) ``` 在示例中,ExpandDimsTest是一个只有单算子的网络,网络的输入有input_x和input_axis两个。因为ExpandDims算子的第二个输入需要是常量,这是因为在图编译过程中推导ExpandDims算子输出维度的时候需要用到,而input_axis作为网络参数输入会泛化成变量,无法确定其值,从而无法推导算子的输出维度导致图编译失败。所以在图编译阶段需要值推导的输入都应该是常量输入。在API中,这类算子需要常量输入的参数会进行说明,标注"constant input is needed"。 * 正确的写法是在construct函数里面对算子的常量输入直接填入需要的值或者是一个类的成员变量,如下: ```python class ExpandDimsTest(Cell): def __init__(self, axis): super(ExpandDimsTest, self).__init__() self.expandDims = P.ExpandDims() self.axis = axis def construct(self, input_x): return self.expandDims(input_x, self.axis) axis = 0 expand_dim = ExpandDimsTest(axis) input_x = Tensor(np.random.randn(2,2,2,2).astype(np.float32)) expand_dim(input_x) ```