lstsq#

metatensor.lstsq(X: TensorMap, Y: TensorMap, rcond: float | None, driver: str | None = None) TensorMap[source]#

Solve a linear system using two TensorMap.

The least-squares solution w_b for the linear system Xbwb=Yb is solved for all blocks b in X and Y. X and Y must have the same keys. The returned TensorMap w has the same keys as X and Y, and stores in each block the least-squares solutions wb.

If a block has multiple components, they are moved to the “samples” axis before solving the linear system.

If gradients are present, they must be present in both X and Y. Gradients are concatenated with the block values along the “samples” axis, Ab=[Xb,Xb], Bb=[Yb,Yb], and the linear system Abwb=Bb is solved for wb using least-squares.

Note

The solutions wb differ from the output of numpy or torch in that they are already transposed. Be aware of that if you want to manually access the values of blocks of w (see also the example below).

Parameters:
Returns:

a TensorMap with the same keys of Y and X, and where each TensorBlock has: the sample equal to the properties of Y; and the properties equal to the properties of X.

Return type:

TensorMap

>>>
>>> import numpy as np
>>> from metatensor import Labels, TensorBlock, TensorMap
>>> import metatensor
>>> values_X = np.array(
...     [
...         [1.0, 2.0],
...         [3.0, 1.0],
...     ]
... )
>>> values_Y = np.array(
...     [
...         [1.0, 0.0],
...         [0.0, 1.0],
...     ]
... )
>>> samples = Labels("structure", np.array([[0], [1]]))
>>> components = []
>>> properties = Labels("properties", np.array([[0], [1]]))
>>> keys = Labels(names="key", values=np.array([[0]]))
>>> block_X = TensorBlock(values_X, samples, components, properties)
>>> block_Y = TensorBlock(values_Y, samples, components, properties)
>>> X = TensorMap(keys, [block_X])
>>> Y = TensorMap(keys, [block_Y])
>>> w = metatensor.lstsq(X, Y, rcond=1e-10)

We take the transpose here

>>>
>>> y = X.block(0).values @ w.block(0).values.T

Set small entries in y to 0, they are numerical noise

>>>
>>> mask = np.abs(y) < 1e-15
>>> y[mask] = 0.0
>>> print(y)
[[1. 0.]
 [0. 1.]]