Skip to content

hadamard

Hadamard

Bases: NumpyOp

Convert integer labels into hadamard code representations.

Parameters:

Name Type Description Default
inputs Union[str, List[str]]

Key of the input tensor(s) to be converted.

required
outputs Union[str, List[str]]

Key of the output tensor(s) in hadamard code representation.

required
n_classes int

How many classes are there in the inputs.

required
code_length Optional[int]

What code length to use. Will default to the smallest power of 2 which is >= the number of classes.

None
mode Union[None, str, Iterable[str]]

What mode(s) to execute this Op in. For example, "train", "eval", "test", or "infer". To execute regardless of mode, pass None. To execute in all modes except for a particular one, you can pass an argument like "!infer" or "!train".

None
ds_id Union[None, str, Iterable[str]]

What dataset id(s) to execute this Op in. To execute regardless of ds_id, pass None. To execute in all ds_ids except for a particular one, you can pass an argument like "!ds1".

None

Raises:

Type Description
ValueError

If an unequal number of inputs and outputs are provided, or if code_length is invalid.

Source code in fastestimator/fastestimator/op/numpyop/univariate/hadamard.py
@traceable()
class Hadamard(NumpyOp):
    """Convert integer labels into hadamard code representations.

    Args:
        inputs: Key of the input tensor(s) to be converted.
        outputs: Key of the output tensor(s) in hadamard code representation.
        n_classes: How many classes are there in the inputs.
        code_length: What code length to use. Will default to the smallest power of 2 which is >= the number of classes.
        mode: What mode(s) to execute this Op in. For example, "train", "eval", "test", or "infer". To execute
            regardless of mode, pass None. To execute in all modes except for a particular one, you can pass an argument
            like "!infer" or "!train".
        ds_id: What dataset id(s) to execute this Op in. To execute regardless of ds_id, pass None. To execute in all
            ds_ids except for a particular one, you can pass an argument like "!ds1".

    Raises:
        ValueError: If an unequal number of `inputs` and `outputs` are provided, or if `code_length` is invalid.
    """
    def __init__(self,
                 inputs: Union[str, List[str]],
                 outputs: Union[str, List[str]],
                 n_classes: int,
                 code_length: Optional[int] = None,
                 mode: Union[None, str, Iterable[str]] = None,
                 ds_id: Union[None, str, Iterable[str]] = None) -> None:
        super().__init__(inputs=inputs, outputs=outputs, mode=mode, ds_id=ds_id)
        self.in_list, self.out_list = True, True
        if len(self.inputs) != len(self.outputs):
            raise ValueError("Hadamard requires the same number of input and output keys.")
        self.n_classes = n_classes
        if code_length is None:
            code_length = 1 << (n_classes - 1).bit_length()
        if code_length <= 0 or (code_length & (code_length - 1) != 0):
            raise ValueError(f"code_length must be a positive power of 2, but got {code_length}.")
        if code_length < n_classes:
            raise ValueError(f"code_length must be >= n_classes, but got {code_length} and {n_classes}.")
        self.code_length = code_length
        labels = hadamard(self.code_length).astype(np.float32)
        labels[np.arange(0, self.code_length, 2), 0] = -1  # Make first column alternate
        labels = labels[:self.n_classes]
        self.labels = labels

    def forward(self, data: List[Union[int, np.ndarray]], state: Dict[str, Any]) -> List[np.ndarray]:
        # TODO - also support one hot with smoothed labels?
        return [gather(tensor=self.labels, indices=np.array(inp)) for inp in data]