Source code for mindquantum.parameterresolver.parameterresolver

# Copyright 2021 Huawei Technologies Co., Ltd
#
# 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.
# ============================================================================
"""Parameter resolver."""

from collections.abc import Iterable
from copy import deepcopy
import numpy as np
import sympy as sp


[docs]class ParameterResolver(dict): """ A ParameterRsolver can set the parameter of parameterized quantum gate or parameterized quantum circuit. By specific which part of parameters needs to calculate gradient, the PQC operator can only calculate gradient of these parameters. Args: data (dict): initial parameter names and its values. Default: None. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 0.3}) >>> pr['b'] = 0.5 >>> pr.no_grad_part('a') >>> pr *= 2 >>> pr {'a': 0.6, 'b': 1.0} >>> pr.no_grad_parameters {'a'} """ def __init__(self, data=None): if data is None: data = {} if not isinstance(data, (dict, ParameterResolver)): raise TypeError( "Require a dict or a ParameterResolver, but get {}!".format( type(data))) for k, v in data.items(): if not isinstance(k, str): raise TypeError( "Parameter name should be a string, but get {}!".format( type(k))) if not isinstance(v, _num_type): raise TypeError( "Require a number, but get {}, which is {}!".format( v, type(v))) super(ParameterResolver, self).__init__(data) self.no_grad_parameters = set() self.requires_grad_parameters = set(self.para_name) def __setitem__(self, keys, values): """ Set parameter or as list of parameters of this parameter resolver. By default, the parameter you set requires gradient. Args: keys (Union[str, list[str]]): The name of parameters. values (Union[number, list[number]]): The value of parameters. Raises: TypeError: If the key that you set is not a string or a iterable of string. """ if isinstance(keys, str): if not isinstance(values, _num_type): raise TypeError( "Parameter value should be a number, but get {}, which is {}!" .format(values, type(values))) super().__setitem__(keys, values) self.requires_grad_parameters.add(keys) elif isinstance(keys, Iterable): if not isinstance(values, Iterable): raise ValueError("Values should be iterable.") if len(values) != len(keys): raise ValueError("Size of keys and values do not match.") for i, k in enumerate(keys): self.__setitem__(k, values[i]) else: raise TypeError( "Parameter name should be a string, but get {}!".format( type(keys))) def __add__(self, pr): """ Add a parameter resolver with other parameter. Returns: ParameterResolver, parameter resolver after adding. Args: pr (ParameterResolver): The parameter resolver need to add. Examples: >>> from mindquantum import ParameterResolver >>> pr1 = ParameterResolver({'a': 1}) >>> pr2 = ParameterResolver({'a': 2, 'b': 3}) >>> (pr1 + pr2).expression() 3*a + 3*b """ if not isinstance(pr, ParameterResolver): raise ValueError( 'Require a parameter resolver, but get {}.'.format(type(pr))) res = self * 1 pr = pr * 1 for k, v in pr.items(): if k in res: res[k] += v pr[k] = res[k] res.update(pr) return res def __sub__(self, pr): """ Subtraction a parameter resolver with other parameter. Returns: :class:`mindquantum.parameterresolver.ParameterResolver` Args: pr (ParameterResolver): The parameter resolver need to subtract. Examples: >>> pr1 = ParameterResolver({'a': 1}) >>> pr2 = ParameterResolver({'a': 2, 'b': 3}) >>> (pr1 - pr2).expression() -a - 3*b """ return self + (-1 * pr) def __neg__(self): """ Get the negative version of this parameter resolver. Returns: ParameterResolver, the negative version. Examples: >>> pr1 = ParameterResolver({'a': 1}) >>> (-pr1).expression() -a """ return -1 * self def __imul__(self, num): """ Parameter support inplace multiply. Returns: :class:`mindquantum.parameterresolver.ParameterResolver` Args: num (number): Multiply factor. Examples: >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr *= 2 >>> pr {'a': 2, 'b': 4} """ no_grad_parameters = deepcopy(self.no_grad_parameters) requires_grad_parameters = deepcopy(self.requires_grad_parameters) for k in self.keys(): self[k] = self[k] * num self.no_grad_parameters = no_grad_parameters self.requires_grad_parameters = requires_grad_parameters return self def __mul__(self, num): """ Multiply num with every value of parameter resolver. Returns: :class:`mindquantum.parameterresolver.ParameterResolver` Args: num (number): Multiply factor. Examples: >>> pr1 = ParameterResolver({'a': 1, 'b': 2}) >>> pr2 = pr1 * 2 >>> pr2 {'a': 2, 'b': 4} """ no_grad_parameters = deepcopy(self.no_grad_parameters) requires_grad_parameters = deepcopy(self.requires_grad_parameters) out = deepcopy(self) out *= num out.no_grad_parameters = no_grad_parameters out.requires_grad_parameters = requires_grad_parameters return out def __rmul__(self, num): """ See :class:`mindquantum.parameterresolver.ParameterResolver.__mul__`. """ return self.__mul__(num) def __eq__(self, other): _check_pr_type(other) no_grad_eq = self.no_grad_parameters == other.no_grad_parameters requires_grad_eq = self.requires_grad_parameters == other.requires_grad_parameters return super().__eq__(other) and no_grad_eq and requires_grad_eq @property def para_name(self): """ Get the parameters name. Returns: list, a list of parameters name. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.para_name ['a', 'b'] """ return list(self.keys()) @property def para_value(self): """ Get the parameters value. Returns: list, a list of parameters value. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.para_value [1, 2] """ return list(self.values())
[docs] def requires_grad(self): """ Set all parameters of this parameter resolver to require gradient calculation. Inplace operation. Returns: ParameterResolver, the parameter resolver itself. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.no_grad_part('a') >>> pr.requires_grad() >>> pr.requires_grad_parameters {'a', 'b'} """ self.no_grad_parameters = set() self.requires_grad_parameters = set(self.para_name) return self
[docs] def no_grad(self): """ Set all parameters to not require gradient calculation. Inplace operation. Returns: ParameterResolver, the parameter resolver itself. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.no_grad() >>> pr.requires_grad_parameters set() """ self.no_grad_parameters = set(self.para_name) self.requires_grad_parameters = set() return self
[docs] def requires_grad_part(self, *names): """ Set part of parameters that requires grad. Inplace operation. Args: names (tuple[str]): Parameters that requires grad. Returns: ParameterResolver, the parameter resolver itself. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.no_grad() >>> pr.requires_grad_part('a') >>> pr.requires_grad_parameters {'a'} """ for name in names: if not isinstance(name, str): raise TypeError("name should be a string, but get {}!".format( type(name))) if name not in self: raise KeyError( "Parameter {} not in this parameter resolver!".format( name)) while name in self.no_grad_parameters: self.no_grad_parameters.remove(name) while name not in self.requires_grad_parameters: self.requires_grad_parameters.add(name) return self
[docs] def no_grad_part(self, *names): """ Set part of parameters that not requires grad. Args: names (tuple[str]): Parameters that not requires grad. Returns: ParameterResolver, the parameter resolver itself. Examples: >>> from mindquantum import ParameterResolver >>> pr = ParameterResolver({'a': 1, 'b': 2}) >>> pr.no_grad_part('a') >>> pr.requires_grad_parameters {'b'} """ for name in names: if not isinstance(name, str): raise TypeError("name should be a string, but get {}!".format( type(name))) if name not in self: raise KeyError( "Parameter {} not in this parameter resolver!".format( name)) while name not in self.no_grad_parameters: self.no_grad_parameters.add(name) while name in self.requires_grad_parameters: self.requires_grad_parameters.remove(name) return self
[docs] def update(self, others): """ Update this parameter resolver with other parameter resolver. Args: others (ParameterResolver): other parameter resolver. Raises: ValueError: If some parameters require grad and not require grad in other parameter resolver and vise versa. Examples: >>> from mindquantum import ParameterResolver >>> pr1 = ParameterResolver({'a': 1}) >>> pr2 = ParameterResolver({'b': 2}) >>> pr2.no_grad() >>> pr1.update(pr2) >>> pr1 {'a': 1, 'b': 2} >>> pr1.no_grad_parameters {'b'} """ _check_pr_type(others) super().update(others) conflict = (self.no_grad_parameters & others.requires_grad_parameters ) | (others.no_grad_parameters & self.requires_grad_parameters) if conflict: raise ValueError( "Parameter conflict, {} require grad in some parameter \ resolver and not require grad in other parameter resolver ".format(conflict)) self.no_grad_parameters.update(others.no_grad_parameters) self.requires_grad_parameters.update(others.requires_grad_parameters)
[docs] def mindspore_data(self): """ Generate data for PQC operator. Returns: Dict. """ m_data = { 'gate_params_names': [], 'gate_coeff': [], 'gate_requires_grad': [] } for k, v in self.items(): m_data['gate_params_names'].append(k) m_data['gate_coeff'].append(float(v)) m_data['gate_requires_grad'].append( k in self.requires_grad_parameters) return m_data
[docs] def expression(self): """ Get the expression of this parameter resolver. Returns: sympy.Expr, the symbol expression of this parameter resolver. Examples: >>> from mindquantum.parameterresolver import ParameterResolver as PR >>> pr = PR({'a' : 2, 'b' : 0.3}) >>> pr.expression() 2*a + 0.3*b """ res = 0 for k, v in self.items(): res += sp.Symbol(k) * v return res
[docs] def conjugate(self): """ Get the conjugate of the parameter resolver. Returns: ParameterResolver, the conjugate version of this parameter resolver. Examples: >>> from mindquantum.parameterresolver import ParameterResolver as PR >>> pr = PR({'a' : 1, 'b': 1j}) >>> pr.conjugate().expression() a - 1.0*I*b """ out = 1 * self for k, v in out.items(): out[k] = np.conj(v) return out
[docs] def combination(self, pr): """ Apply linear combination between this parameter resolver with input pr. Args: pr (Union[dict, ParameterResolver]): The parameter resolver you want to do linear combination. Returns: numbers.Number, the combination result. Examples: >>> from mindquantum import ParameterResolver >>> pr1 = ParameterResolver({'a': 1, 'b': 2}) >>> pr2 = ParameterResolver({'a': 2, 'b': 3}) >>> pr1.combination(pr2) """ if not isinstance(pr, (ParameterResolver, dict)): raise ValueError( 'Require a parameter resolver or a dict, but get {}.'.format( type(pr))) res = 0 for k, v in self.items(): if k not in pr: raise KeyError('{} not in input parameter resolver'.format(k)) res += v * pr[k] return res
@property def real(self): """ Get the real part of this parameter resolver Returns: ParameterResolver, the real part of this parameter resolver. Examples: >>> from mindquantum.parameterresolver import ParameterResolver as PR >>> pr = PR({'a': 1.2 + 1.3j}) >>> pr.real() {'a': 1.2} """ out = 1 * self for k, v in self.items(): out[k] = np.real(v) return out @property def imag(self): """ Get the real part of this parameter resolver Returns: ParameterResolver, the image part of this parameter resolver. Examples: >>> from mindquantum.parameterresolver import ParameterResolver as PR >>> pr = PR({'a': 1.2 + 1.3j}) >>> pr.imag() {'a': 1.3} """ out = 1 * self for k, v in self.items(): out[k] = np.imag(v) return out
def _check_pr_type(pr): if not isinstance(pr, ParameterResolver): raise TypeError("Require a ParameterResolver, but get {}".format( type(pr))) _num_type = (int, float, complex, np.int32, np.int64, np.float32, np.float64)