// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
import Std.Diagnostics.*;
import Std.Math.*;
/// # Summary
/// Given an array and a predicate that is defined
/// for the elements of the array, and checks if all elements of the
/// array satisfy the predicate.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## predicate
/// A function from `'T` to `Bool` that is used to check elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// A `Bool` value of the AND function of the predicate applied to all elements.
///
/// # Example
/// The following code checks whether all elements of the array are non-zero:
/// ```qsharp
/// let allNonZero = All(x -> x != 0, [1, 2, 3, 4, 5]);
/// ```
function All<'T>(predicate : ('T -> Bool), array : 'T[]) : Bool {
for element in array {
if not predicate(element) {
return false;
}
}
true
}
/// # Summary
/// Given an array and a predicate that is defined
/// for the elements of the array, checks if at least one element of
/// the array satisfies the predicate.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## predicate
/// A function from `'T` to `Bool` that is used to check elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// A `Bool` value of the OR function of the predicate applied to all elements.
///
/// # Example
/// ```qsharp
/// let anyEven = Any(x -> x % 2 == 0, [1, 3, 6, 7, 9]);
/// ```
function Any<'T>(predicate : ('T -> Bool), array : 'T[]) : Bool {
for element in array {
if predicate(element) {
return true;
}
}
false
}
/// # Summary
/// Splits an array into multiple parts of equal length.
///
/// # Input
/// ## chunkSize
/// The length of each chunk. Must be positive.
/// ## array
/// The array to be split in chunks.
///
/// # Output
/// A array containing each chunk of the original array.
///
/// # Remarks
/// Note that the last element of the output may be shorter
/// than `chunkSize` if `Length(array)` is not divisible by `chunkSize`.
function Chunks<'T>(chunkSize : Int, array : 'T[]) : 'T[][] {
Fact(chunkSize > 0, "`chunkSize` must be positive");
mutable output = [];
mutable remaining = array;
while (not IsEmpty(remaining)) {
let chunkSizeToTake = MinI(Length(remaining), chunkSize);
set output += [remaining[...chunkSizeToTake - 1]];
set remaining = remaining[chunkSizeToTake...];
}
output
}
/// # Summary
/// Shift an array circularly left or right by a specific step size.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## stepCount
/// The amount of positions by which the array elements will be shifted.
/// If this is positive, `array` is circularly shifted to the right.
/// If this is negative, `array` is circularly shifted to the left.
/// ## array
/// Array to be circularly shifted.
///
/// # Output
/// An array `output` that is the `array` circularly shifted to the right or left
/// by the specified step size.
///
/// # Example
/// ```qsharp
/// let array = [10, 11, 12];
/// // The following line returns [11, 12, 10].
/// let output = CircularlyShifted(2, array);
/// // The following line returns [12, 10, 11].
/// let output = CircularlyShifted(-2, array);
/// ```
function CircularlyShifted<'T>(stepCount : Int, array : 'T[]) : 'T[] {
let arrayLength = Length(array);
if arrayLength <= 1 {
return array;
}
// normalize circular shift count to be within the bounds of the array length
let normalizedShift = stepCount % arrayLength;
let effectiveShift = normalizedShift >= 0 ? arrayLength - normalizedShift | -normalizedShift;
// no shift needed
if effectiveShift == 0 {
return array;
}
let leftPart = array[...effectiveShift - 1];
let rightPart = array[effectiveShift..arrayLength - 1];
rightPart + leftPart
}
/// # Summary
/// Extracts a column from a matrix.
///
/// # Description
/// This function extracts a column in a matrix in row-wise order.
/// Extracting a row corresponds to element access of the first index
/// and therefore requires no further treatment.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `matrix`.
///
/// # Input
/// ## column
/// Column of the matrix
/// ## matrix
/// 2-dimensional matrix in row-wise order
///
/// # Example
/// ```qsharp
/// let matrix = [[1, 2, 3], [4, 5, 6], [7, 8, 9]];
/// let column = ColumnAt(0, matrix);
/// // same as: column = [1, 4, 7]
/// ```
///
/// # See Also
/// - [Std.Arrays.Transposed](xref:Qdk.Std.Arrays.Transposed)
/// - [Std.Arrays.Diagonal](xref:Qdk.Std.Arrays.Diagonal)
function ColumnAt<'T>(column : Int, matrix : 'T[][]) : 'T[] {
Fact(IsRectangularArray(matrix), "`matrix` is not a rectangular array");
mutable columnValues = [];
for row in matrix {
set columnValues += [row[column]];
}
columnValues
}
/// # Summary
/// Given an array and a predicate that is defined
/// for the elements of the array, returns the number of elements
/// an array that consists of those elements that satisfy the predicate.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## predicate
/// A function from `'T` to Boolean that is used to filter elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// The number of elements in `array` that satisfy the predicate.
///
/// # Example
/// ```qsharp
/// let evensCount = Count(x -> x % 2 == 0, [1, 3, 6, 7, 9]);
/// // evensCount is 1.
/// ```
function Count<'T>(predicate : ('T -> Bool), array : 'T[]) : Int {
mutable count = 0;
for element in array {
if predicate(element) {
set count += 1;
}
}
count
}
/// # Summary
/// Returns an array of diagonal elements of a 2-dimensional array
///
/// # Description
/// If the 2-dimensional array has not a square shape, the diagonal over
/// the minimum over the number of rows and columns will be returned.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `matrix`.
///
/// # Input
/// ## matrix
/// 2-dimensional matrix in row-wise order.
///
/// # Example
/// ```qsharp
/// let matrix = [[1, 2, 3], [4, 5, 6], [7, 8, 9]];
/// let diagonal = Diagonal(matrix);
/// // same as: column = [1, 5, 9]
/// ```
///
/// # See Also
/// - [Std.Arrays.Transposed](xref:Qdk.Std.Arrays.Transposed)
function Diagonal<'T>(matrix : 'T[][]) : 'T[] {
Fact(IsRectangularArray(matrix), "`matrix` is not a rectangular array");
let rows = Length(matrix);
let columns = rows == 0 ? 0 | Length(Head(matrix));
let rangeLimit = MinI(rows, columns) - 1;
mutable diagonal = [];
for index in 0..rangeLimit {
set diagonal += [matrix[index][index]];
}
diagonal
}
/// # Summary
/// Repeats an operation for a given number of samples, collecting its outputs
/// in an array.
///
/// # Input
/// ## op
/// The operation to be called repeatedly.
/// ## nSamples
/// The number of samples of calling `op` to collect.
/// ## input
/// The input to be passed to `op`.
///
/// # Type Parameters
/// ## TInput
/// The type of input expected by `op`.
/// ## TOutput
/// The type of output returned by `op`.
///
/// # Example
/// The following samples an alternating array of results.
/// ```qsharp
/// use qubit = Qubit();
/// let results = Std.Arrays.DrawMany(q => {X(q); M(q)}, 3, qubit);
/// ```
operation DrawMany<'TInput, 'TOutput>(op : ('TInput => 'TOutput), nSamples : Int, input : 'TInput) : 'TOutput[] {
mutable outputs = [];
for _ in 1..nSamples {
set outputs += [op(input)];
}
outputs
}
/// # Summary
/// Given an array, returns a new array containing elements of the original
/// array along with the indices of each element.
///
/// # Type Parameters
/// ## 'TElement
/// The type of elements of the array.
///
/// # Input
/// ## array
/// An array whose elements are to be enumerated.
///
/// # Output
/// A new array containing elements of the original array along with their
/// indices.
///
/// # Example
/// The following `for` loops are equivalent:
/// ```qsharp
/// for (idx in IndexRange(array)) {
/// let element = array[idx];
/// ...
/// }
/// for ((idx, element) in Enumerated(array)) { ... }
/// ```
function Enumerated<'TElement>(array : 'TElement[]) : (Int, 'TElement)[] {
MappedByIndex((index, element) -> (index, element), array)
}
/// # Summary
/// Returns an array containing the elements of another array,
/// excluding elements at a given list of indices.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## remove
/// An array of indices denoting which elements should be excluded.
/// from the output.
/// ## array
/// Array of which the values in the output array are taken.
///
/// # Output
/// An array `output` such that `output[0]` is the first element
/// of `array` whose index does not appear in `remove`,
/// such that `output[1]` is the second such element, and so
/// forth.
///
/// # Example
/// ```qsharp
/// let array = [10, 11, 12, 13, 14, 15];
/// // The following line returns [10, 12, 15].
/// let subarray = Excluding([1, 3, 4], array);
/// ```
function Excluding<'T>(remove : Int[], array : 'T[]) : 'T[] {
let arrayLength = Length(array);
mutable toKeep = Repeated(true, arrayLength);
for indexToRemove in remove {
set toKeep w/= indexToRemove <- false;
}
mutable output = [];
for index in 0..arrayLength - 1 {
if toKeep[index] {
set output += [array[index]];
}
}
output
}
/// # Summary
/// Given an array and a predicate that is defined
/// for the elements of the array, returns an array that consists of
/// those elements that satisfy the predicate.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## predicate
/// A function from `'T` to Boolean that is used to filter elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// An array `'T[]` of elements that satisfy the predicate.
///
/// # Example
/// The following code creates an array that contains only even numbers.
/// ```qsharp
/// Filtered(x -> x % 2 == 0, [0, 1, 2, 3, 4])
/// ```
function Filtered<'T>(predicate : ('T -> Bool), array : 'T[]) : 'T[] {
mutable filtered = [];
for element in array {
if predicate(element) {
set filtered += [element];
}
}
filtered
}
/// # Summary
/// Given an array and a function that maps an array element to some output
/// array, returns the concatenated output arrays for each array element.
///
/// # Type Parameters
/// ## 'TInput
/// The type of `array` elements.
/// ## 'TOutput
/// The `mapper` function returns arrays of this type.
///
/// # Input
/// ## mapper
/// A function from `'TInput` to `'TOutput[]` that is used to map array elements.
/// ## array
/// An array of elements.
///
/// # Output
/// An array of `'TOutput[]` which is the concatenation of all arrays generated by
/// the mapping function.
///
/// # Example
/// The following code creates an array with each element of the input array repeated twice.
/// ```qsharp
/// let repeatedPairs = FlatMapped(x -> Repeated(x, 2), [1, 2, 3]);
/// // repeatedPairs is [1, 1, 2, 2, 3, 3].
/// ```
function FlatMapped<'TInput, 'TOutput>(mapper : ('TInput -> 'TOutput[]), array : 'TInput[]) : 'TOutput[] {
mutable output = [];
for element in array {
set output += mapper(element);
}
output
}
/// # Summary
/// Given an array of arrays, returns the concatenation of all arrays.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## arrays
/// Array of arrays.
///
/// # Output
/// Concatenation of all arrays.
///
/// # Example
/// ```qsharp
/// let flattened = Flattened([[1, 2], [3], [4, 5, 6]]);
/// // flattened = [1, 2, 3, 4, 5, 6]
/// ```
function Flattened<'T>(arrays : 'T[][]) : 'T[] {
mutable output = [];
for array in arrays {
set output += array;
}
output
}
/// # Summary
/// Iterates a function `f` through an array `array`, returning
/// `f(...f(f(initialState, array[0]), array[1]), ...)`.
///
/// # Type Parameters
/// ## 'State
/// The type of states the `folder` function operates on, i.e., accepts as its first
/// argument and returns.
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## folder
/// A function to be folded over the array.
/// ## state
/// The initial state of the folder.
/// ## array
/// An array of values to be folded over.
///
/// # Output
/// The final state returned by the folder after iterating over
/// all elements of `array`.
///
/// # Example
/// ```qsharp
/// let sum = Fold((x, y) -> x + y, 0, [1, 2, 3, 4, 5]); // `sum` is 15.
/// ```
function Fold<'State, 'T>(folder : (('State, 'T) -> 'State), state : 'State, array : 'T[]) : 'State {
mutable current = state;
for element in array {
set current = folder(current, element);
}
current
}
/// # Summary
/// Given an array and an operation that is defined
/// for the elements of the array, returns a new array that consists
/// of the images of the original array under the operation.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
/// ## 'U
/// The result type of the `action` operation.
///
/// # Input
/// ## action
/// An operation from `'T` to `'U` that is applied to each element.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// An array `'U[]` of elements that are mapped by the `action` operation.
///
/// # See Also
/// - [Std.Arrays.Mapped](xref:Qdk.Std.Arrays.Mapped)
operation ForEach<'T, 'U>(action : ('T => 'U), array : 'T[]) : 'U[] {
mutable output = [];
for element in array {
set output += [action(element)];
}
output
}
/// # Summary
/// Returns the first element of the array.
///
/// # Type Parameters
/// ## 'A
/// The type of the array elements.
///
/// # Input
/// ## array
/// Array of which the first element is taken. Array must have at least 1 element.
///
/// # Output
/// The first element of the array.
function Head<'A>(array : 'A[]) : 'A {
Fact(Length(array) > 0, "Array must have at least 1 element");
array[0]
}
/// # Summary
/// Returns a tuple of first and all remaining elements of the array.
///
/// # Type Parameters
/// ## 'A
/// The type of the array elements.
///
/// # Input
/// ## array
/// An array with at least one element.
///
/// # Output
/// A tuple of first and all remaining elements of the array.
function HeadAndRest<'A>(array : 'A[]) : ('A, 'A[]) {
(Head(array), Rest(array))
}
/// # Summary
/// Returns the first index of the first element in an array that satisfies
/// a given predicate. If no such element exists, returns -1.
///
/// # Input
/// ## predicate
/// A predicate function acting on elements of the array.
/// ## array
/// An array to be searched using the given predicate.
///
/// # Output
/// Either the smallest index of an element for which `predicate(array[index])` is true,
/// or -1 if no such element exists.
///
/// # Example
/// The following code gets the index of the first even number in the input array.
/// ```qsharp
/// let indexOfFirstEven = IndexOf(x -> x % 2 == 0, [1, 3, 17, 2, 21]);
/// // `indexOfFirstEven` is 3.
/// ```
function IndexOf<'T>(predicate : ('T -> Bool), array : 'T[]) : Int {
for index in 0..Length(array) - 1 {
if predicate(array[index]) {
return index;
}
}
-1
}
/// # Summary
/// Given an array, returns a range over the indices of that array, suitable
/// for use in a for loop.
///
/// # Type Parameters
/// ## 'TElement
/// The type of elements of the array.
///
/// # Input
/// ## array
/// An array for which a range of indices should be returned.
///
/// # Output
/// A range over all indices of the array.
///
/// # Example
/// The following `for` loops are equivalent:
/// ```qsharp
/// for idx in IndexRange(array) { ... }
/// for idx in 0 .. Length(array) - 1 { ... }
/// ```
function IndexRange<'TElement>(array : 'TElement[]) : Range {
0..Length(array) - 1
}
/// # Summary
/// Interleaves two arrays of (almost) same size.
///
/// # Description
/// This function returns the interleaving of two arrays, starting
/// with the first element from the first array, then the first
/// element from the second array, and so on.
///
/// The first array must either be
/// of the same length as the second one, or can have one more element.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `first` and `second`.
///
/// # Input
/// ## first
/// The first array to be interleaved.
///
/// ## second
/// The second array to be interleaved.
///
/// # Output
/// Interleaved array
///
/// # Example
/// ```qsharp
/// // same as interleaved = [1, -1, 2, -2, 3, -3]
/// let interleaved = Interleaved([1, 2, 3], [-1, -2, -3])
/// ```
function Interleaved<'T>(first : 'T[], second : 'T[]) : 'T[] {
let firstLength = Length(first);
let secondLength = Length(second);
Fact(
firstLength == secondLength or firstLength == secondLength + 1,
"Array `first` must either be of same size as `second` or have one more element"
);
let interleavedLength = firstLength + secondLength;
mutable interleaved = [];
for index in 0..interleavedLength - 1 {
let originalIndex = index / 2;
let value = if index % 2 == 0 { first[originalIndex] } else { second[originalIndex] };
set interleaved += [value];
}
interleaved
}
/// # Summary
/// Returns true if and only if an array is empty.
///
/// # Input
/// ## array
/// The array to be checked.
///
/// # Output
/// `true` if and only if the array is empty (has length 0).
function IsEmpty<'T>(array : 'T[]) : Bool {
Length(array) == 0
}
/// # Summary
/// Returns whether a 2-dimensional array has a rectangular shape
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `array`.
///
/// # Input
/// ## array
/// A 2-dimensional array of elements.
///
/// # Output
/// `true` if the array is rectangular, `false` otherwise.
///
/// # Example
/// ```qsharp
/// IsRectangularArray([[1, 2], [3, 4]]); // true
/// IsRectangularArray([[1, 2, 3], [4, 5, 6]]); // true
/// IsRectangularArray([[1, 2], [3, 4, 5]]); // false
/// ```
///
/// # See Also
/// - [Std.Arrays.IsSquareArray](xref:Qdk.Std.Arrays.IsSquareArray)
function IsRectangularArray<'T>(array : 'T[][]) : Bool {
if (Length(array) > 0) {
let columnCount = Length(Head(array));
for index in 1..Length(array) - 1 {
if Length(array[index]) != columnCount {
return false;
}
}
}
true
}
/// # Summary
/// Given an array, returns whether that array is sorted as defined by
/// a given comparison function.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `array`.
///
/// # Input
/// ## comparison
/// A function that compares two elements such that `a` is considered to
/// be less than or equal to `b` if `comparison(a, b)` is `true`.
/// ## array
/// The array to be checked.
///
/// # Output
/// `true` if and only if for each pair of elements `a` and `b` of
/// `array` occurring in that order, `comparison(a, b)` is `true`.
///
/// # Remarks
/// The function `comparison` is assumed to be transitive, such that
/// if `comparison(a, b)` and `comparison(b, c)`, then `comparison(a, c)`
/// is assumed.
function IsSorted<'T>(comparison : (('T, 'T) -> Bool), array : 'T[]) : Bool {
for index in 1..Length(array) - 1 {
if not comparison(array[index - 1], array[index]) {
return false;
}
}
true
}
/// # Summary
/// Returns whether a 2-dimensional array has a square shape
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `array`.
///
/// # Input
/// ## array
/// A 2-dimensional array of elements.
///
/// # Example
/// ```qsharp
/// IsSquareArray([[1, 2], [3, 4]]); // true
/// IsSquareArray([[1, 2, 3], [4, 5, 6]]); // false
/// IsSquareArray([[1, 2], [3, 4], [5, 6]]); // false
/// ```
///
/// # Output
/// `true` if the array is square, `false` otherwise.
///
/// # See Also
/// - [Std.Arrays.IsRectangularArray](xref:Qdk.Std.Arrays.IsRectangularArray)
function IsSquareArray<'T>(array : 'T[][]) : Bool {
if (Length(array) > 0) {
let columnCount = Length(array);
for column in array {
if Length(column) != columnCount {
return false;
}
}
}
true
}
/// # Summary
/// Given an array and a function that is defined
/// for the elements of the array, returns a new array that consists
/// of the images of the original array under the function.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
/// ## 'U
/// The result type of the `mapper` function.
///
/// # Input
/// ## mapper
/// A function from `'T` to `'U` that is used to map elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// An array `'U[]` of elements that are mapped by the `mapper` function.
///
/// # See Also
/// - [Std.Arrays.ForEach](xref:Qdk.Std.Arrays.ForEach)
function Mapped<'T, 'U>(mapper : ('T -> 'U), array : 'T[]) : 'U[] {
mutable mapped = [];
for element in array {
set mapped += [mapper(element)];
}
mapped
}
/// # Summary
/// Given an array and a function that is defined
/// for the indexed elements of the array, returns a new array that consists
/// of the images of the original array under the function.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
/// ## 'U
/// The result type of the `mapper` function.
///
/// # Input
/// ## mapper
/// A function from `(Int, 'T)` to `'U` that is used to map elements
/// and their indices.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// An array `'U[]` of elements that are mapped by the `mapper` function.
///
/// # Example
/// The following two lines are equivalent:
/// ```qsharp
/// let array = MappedByIndex(f, [x0, x1, x2]);
/// ```
/// and
/// ```qsharp
/// let array = [f(0, x0), f(1, x1), f(2, x2)];
/// ```
///
/// # See Also
/// - [Std.Arrays.Mapped](xref:Qdk.Std.Arrays.Mapped)
function MappedByIndex<'T, 'U>(mapper : ((Int, 'T) -> 'U), array : 'T[]) : 'U[] {
mutable mapped = [];
for index in 0..Length(array) - 1 {
set mapped += [mapper(index, array[index])];
}
mapped
}
/// # Summary
/// Given a range and a function that takes an integer as input,
/// returns a new array that consists
/// of the images of the range values under the function.
///
/// # Type Parameters
/// ## 'T
/// The result type of the `mapper` function.
///
/// # Input
/// ## mapper
/// A function from `Int` to `'T` that is used to map range values.
/// ## range
/// A range of integers.
///
/// # Output
/// An array `'T[]` of elements that are mapped by the `mapper` function.
///
/// # Example
/// This example adds 1 to a range of even numbers:
/// ```qsharp
/// let numbers = MappedOverRange(x -> x + 1, 0..2..10);
/// // numbers = [1, 3, 5, 7, 9, 11]
/// ```
///
/// # See Also
/// - [Std.Arrays.Mapped](xref:Qdk.Std.Arrays.Mapped)
function MappedOverRange<'T>(mapper : (Int -> 'T), range : Range) : 'T[] {
mutable output = [];
for element in range {
set output += [mapper(element)];
}
output
}
/// # Summary
/// Creates an array that is equal to an input array except that the last array
/// element is dropped.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## array
/// An array whose first to second-to-last elements are to form the output array.
///
/// # Output
/// An array containing the elements `array[0..Length(array) - 2]`.
function Most<'T>(array : 'T[]) : 'T[] {
array[...Length(array) - 2]
}
/// # Summary
/// Returns a tuple of all but one and the last element of the array.
///
/// # Type Parameters
/// ## 'A
/// The type of the array elements.
///
/// # Input
/// ## array
/// An array with at least one element.
///
/// # Output
/// A tuple of all but one and the last element of the array.
function MostAndTail<'A>(array : 'A[]) : ('A[], 'A) {
(Most(array), Tail(array))
}
/// # Summary
/// Returns an array padded at with specified values up to a
/// specified length.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## paddedLength
/// The length of the padded array. If this is positive, `array`
/// is padded at the head. If this is negative, `array` is padded
/// at the tail.
/// ## defaultElement
/// Default value to use for padding elements.
/// ## array
/// Array to be padded.
///
/// # Output
/// An array `output` that is the `array` padded at the head or the tail
/// with `defaultElement`s until `output` has length `paddedLength`
///
/// # Example
/// ```qsharp
/// let array = [10, 11, 12];
/// // The following line returns [10, 12, 15, 2, 2].
/// let output = Padded(-5, 2, array);
/// // The following line returns [2, 2, 10, 12, 15].
/// let output = Padded(5, 2, array);
/// ```
function Padded<'T>(paddedLength : Int, defaultElement : 'T, inputArray : 'T[]) : 'T[] {
let nElementsInitial = Length(inputArray);
let nAbsElementsTotal = AbsI(paddedLength);
if nAbsElementsTotal < nElementsInitial {
fail "Specified output array length must be at least as long as `inputArray` length.";
}
let nElementsPad = nAbsElementsTotal - nElementsInitial;
let padArray = Repeated(defaultElement, nElementsPad);
if (paddedLength >= 0) {
padArray + inputArray // Padded at head.
} else {
inputArray + padArray // Padded at tail.
}
}
/// # Summary
/// Splits an array into multiple parts.
///
/// # Input
/// ## partitionSizes
/// Number of elements in each split part of array.
/// ## array
/// Input array to be split.
///
/// # Output
/// Multiple arrays where the first array is the first `partitionSizes[0]` of `array`
/// and the second array are the next `partitionSizes[1]` of `array` etc. The last array
/// will contain all remaining elements. If the array is split exactly, the
/// last array will be the empty array, indicating there are no remaining elements.
/// In other words, `Tail(Partitioned(...))` will always return the remaining
/// elements, while `Most(Partitioned(...))` will always return the complete
/// partitions of the array.
///
/// # Example
/// ```qsharp
/// // The following returns [[2, 3], [5], [7]];
/// let split = Partitioned([2, 1], [2, 3, 5, 7]);
/// // The following returns [[2, 3], [5, 7], []];
/// let split = Partitioned([2, 2], [2, 3, 5, 7]);
/// ```
function Partitioned<'T>(partitionSizes : Int[], array : 'T[]) : 'T[][] {
mutable output = Repeated([], Length(partitionSizes) + 1);
mutable partitionStartIndex = 0;
for index in IndexRange(partitionSizes) {
let partitionEndIndex = partitionStartIndex + partitionSizes[index] - 1;
if partitionEndIndex >= Length(array) {
fail "Partitioned argument out of bounds.";
}
set output w/= index <- array[partitionStartIndex..partitionEndIndex];
set partitionStartIndex = partitionEndIndex + 1;
}
set output w/= Length(partitionSizes) <- array[partitionStartIndex..Length(array) - 1];
output
}
/// # Summary
/// Creates an array that is equal to an input array except that the first array
/// element is dropped.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## array
/// An array whose second to last elements are to form the output array.
///
/// # Output
/// An array containing the elements `array[1..Length(array) - 1]`.
function Rest<'T>(array : 'T[]) : 'T[] {
array[1...]
}
/// # Summary
/// Create an array that contains the same elements as an input array but in reversed
/// order.
///
/// # Type Parameters
/// ## 'T
/// The type of the array elements.
///
/// # Input
/// ## array
/// An array whose elements are to be copied in reversed order.
///
/// # Output
/// An array containing the elements `array[Length(array) - 1]` .. `array[0]`.
function Reversed<'T>(array : 'T[]) : 'T[] {
array[...-1...]
}
/// # Summary
/// Get an array of integers in a given interval.
///
/// # Input
/// ## from
/// An inclusive start index of the interval.
/// ## to
/// An inclusive end index of the interval that is not smaller than `from`.
///
/// # Output
/// An array containing the sequence of numbers `from`, `from + 1`, ...,
/// `to`.
///
/// # Example
/// ```qsharp
/// let arr1 = SequenceI(0, 3); // [0, 1, 2, 3]
/// let arr2 = SequenceI(23, 29); // [23, 24, 25, 26, 27, 28, 29]
/// let arr3 = SequenceI(-5, -2); // [-5, -4, -3, -2]
///
/// let numbers = SequenceI(0, _); // function to create sequence from 0 to `to`
/// let naturals = SequenceI(1, _); // function to create sequence from 1 to `to`
/// ```
function SequenceI(from : Int, to : Int) : Int[] {
Fact(to >= from, "`to` must be larger than `from`.");
mutable array = [];
for index in from..to {
set array += [index];
}
array
}
/// # Summary
/// Get an array of integers in a given interval.
///
/// # Input
/// ## from
/// An inclusive start index of the interval.
/// ## to
/// An inclusive end index of the interval that is not smaller than `from`.
///
/// # Output
/// An array containing the sequence of numbers `from`, `from + 1`, ...,
/// `to`.
///
/// # Remarks
/// The difference between `from` and `to` must fit into an `Int` value.
///
/// # Example
/// ```qsharp
/// let arr1 = SequenceL(0L, 3L); // [0L, 1L, 2L, 3L]
/// let arr2 = SequenceL(23L, 29L); // [23L, 24L, 25L, 26L, 27L, 28L, 29L]
/// let arr3 = SequenceL(-5L, -2L); // [-5L, -4L, -3L, -2L]
/// ```
function SequenceL(from : BigInt, to : BigInt) : BigInt[] {
Fact(to >= from, "`to` must be larger than `from`");
mutable array = [];
mutable current = from;
while current <= to {
set array += [current];
set current += 1L;
}
array
}
/// # Summary
/// Given an array, returns the elements of that array sorted by a given
/// comparison function.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `array`.
///
/// # Input
/// ## comparison
/// A function that compares two elements such that `a` is considered to
/// be less than or equal to `b` if `comparison(a, b)` is `true`.
/// ## array
/// The array to be sorted.
///
/// # Output
/// An array containing the same elements as `array`, such that for all
/// elements `a` occurring earlier than elements `b`, `comparison(a, b)`
/// is `true`.
///
/// # Example
/// The following snippet sorts an array of integers to occur in ascending
/// order:
/// ```qsharp
/// let sortedArray = Sorted(LessThanOrEqualI, [3, 17, 11, -201, -11]);
/// ```
///
/// # Remarks
/// The function `comparison` is assumed to be transitive, such that
/// if `comparison(a, b)` and `comparison(b, c)`, then `comparison(a, c)`
/// is assumed. If this property does not hold, then the output of this
/// function may be incorrect.
function Sorted<'T>(comparison : (('T, 'T) -> Bool), array : 'T[]) : 'T[] {
if Length(array) <= 1 {
return array;
}
let pivotIndex = Length(array) / 2;
let left = array[...pivotIndex - 1];
let right = array[pivotIndex...];
// Sort each sublist, then merge them back into a single combined
// list and return.
SortedMerged(
comparison,
Sorted(comparison, left),
Sorted(comparison, right)
)
}
/// # Summary
/// Given two sorted arrays, returns a single array containing the
/// elements of both in sorted order. Used internally by `Sorted`.
internal function SortedMerged<'T>(comparison : (('T, 'T) -> Bool), left : 'T[], right : 'T[]) : 'T[] {
mutable output = [];
mutable remainingLeft = left;
mutable remainingRight = right;
while (not IsEmpty(remainingLeft)) and (not IsEmpty(remainingRight)) {
if comparison(Head(remainingLeft), Head(remainingRight)) {
set output += [Head(remainingLeft)];
set remainingLeft = Rest(remainingLeft);
} else {
set output += [Head(remainingRight)];
set remainingRight = Rest(remainingRight);
}
}
// Note that at this point, either or both of `remainingLeft` and `remainingRight` are empty,
// such that we can simply append both to our output to get the whole merged array.
output + remainingLeft + remainingRight
}
/// # Summary
/// Takes an array and a list of locations and
/// produces a new array formed from the elements of the original
/// array that match the given locations.
///
/// # Remarks
/// If `locations` contains repeated elements, the corresponding elements
/// of `array` will likewise be repeated.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## locations
/// A list of locations in the input array that is used to define the subarray.
/// ## array
/// An array from which a subarray will be generated.
///
/// # Output
/// An array `out` of elements whose locations correspond to the subarray,
/// such that `out[index] == array[locations[index]]`.
///
/// # Example
///
/// ```qsharp
/// let array = [1, 2, 3, 4];
/// let permutation = Subarray([3, 0, 2, 1], array); // [4, 1, 3, 2]
/// let duplicates = Subarray([1, 2, 2], array); // [2, 3, 3]
/// ```
function Subarray<'T>(locations : Int[], array : 'T[]) : 'T[] {
mutable subarray = [];
for location in locations {
set subarray += [array[location]];
}
subarray
}
/// # Summary
/// Applies a swap of two elements in an array.
///
/// # Input
/// ## firstIndex
/// Index of the first element to be swapped.
///
/// ## secondIndex
/// Index of the second element to be swapped.
///
/// ## array
/// Array with elements to be swapped.
///
/// # Output
/// The array with the in place swap applied.
///
/// # Example
/// ```qsharp
/// // The following returns [0, 3, 2, 1, 4]
/// Swapped(1, 3, [0, 1, 2, 3, 4]);
/// ```
function Swapped<'T>(firstIndex : Int, secondIndex : Int, array : 'T[]) : 'T[] {
array
w/ firstIndex <- array[secondIndex]
w/ secondIndex <- array[firstIndex]
}
/// # Summary
/// Returns the transpose of a matrix represented as an array
/// of arrays.
///
/// # Description
/// Input as an r x c matrix with r rows and c columns. The matrix
/// is row-based, i.e., `matrix[i][j]` accesses the element at row `i` and column `j`.
///
/// This function returns the c x r matrix that is the transpose of the
/// input matrix.
///
/// # Type Parameters
/// ## 'T
/// The type of each element of `matrix`.
///
/// # Input
/// ## matrix
/// Row-based r x c matrix.
///
/// # Output
/// Transposed c x r matrix.
///
/// # Example
/// ```qsharp
/// // same as [[1, 4], [2, 5], [3, 6]]
/// let transposed = Transposed([[1, 2, 3], [4, 5, 6]]);
/// ```
function Transposed<'T>(matrix : 'T[][]) : 'T[][] {
let rowCount = Length(matrix);
Fact(rowCount > 0, "Matrix must have at least 1 row");
let columnCount = Length(Head(matrix));
Fact(columnCount > 0, "Matrix must have at least 1 column");
Fact(IsRectangularArray(matrix), "Matrix is not a rectangular array");
mutable transposed = [];
for columnIndex in 0..columnCount - 1 {
mutable newRow = [];
for rowIndex in 0..rowCount - 1 {
set newRow += [matrix[rowIndex][columnIndex]];
}
set transposed += [newRow];
}
transposed
}
/// # Summary
/// Returns the last element of the array.
///
/// # Type Parameters
/// ## 'A
/// The type of the array elements.
///
/// # Input
/// ## array
/// Array of which the last element is taken. Array must have at least 1 element.
///
/// # Output
/// The last element of the array.
function Tail<'A>(array : 'A[]) : 'A {
let size = Length(array);
Fact(size > 0, "Array must have at least 1 element");
array[size - 1]
}
/// # Summary
/// Given an array of 2-tuples, returns a tuple of two arrays, each containing
/// the elements of the tuples of the input array.
///
/// # Type Parameters
/// ## 'T
/// The type of the first element in each tuple.
/// ## 'U
/// The type of the second element in each tuple.
///
/// # Input
/// ## array
/// An array containing 2-tuples.
///
/// # Output
/// Two arrays, the first one containing all first elements of the input
/// tuples, the second one containing all second elements of the input tuples.
///
/// # Example
/// ```qsharp
/// // split is same as ([5, 4, 3, 2, 1], [true, false, true, true, false])
/// let split = Unzipped([(5, true), (4, false), (3, true), (2, true), (1, false)]);
/// ```
///
/// # See Also
/// - [Std.Arrays.Zipped](xref:Qdk.Std.Arrays.Zipped)
function Unzipped<'T, 'U>(array : ('T, 'U)[]) : ('T[], 'U[]) {
mutable first = [];
mutable second = [];
for index in 0..Length(array) - 1 {
let (left, right) = array[index];
set first += [left];
set second += [right];
}
return (first, second);
}
/// # Summary
/// Given a predicate and an array, returns the indices of that
/// array where the predicate is true.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## predicate
/// A function from `'T` to Boolean that is used to filter elements.
/// ## array
/// An array of elements over `'T`.
///
/// # Output
/// An array of indices where `predicate` is true.
function Where<'T>(predicate : ('T -> Bool), array : 'T[]) : Int[] {
mutable indexes = [];
for index in 0..Length(array) - 1 {
if predicate(array[index]) {
set indexes += [index];
}
}
indexes
}
/// # Summary
/// Returns all consecutive subarrays of length `size`.
///
/// # Description
/// This function returns all `n - size + 1` subarrays of
/// length `size` in order, where `n` is the length of `array`.
/// The first subarrays are `array[0..size - 1], array[1..size], array[2..size + 1]`
/// until the last subarray `array[n - size..n - 1]`.
///
/// # Type Parameters
/// ## 'T
/// The type of `array` elements.
///
/// # Input
/// ## size
/// Length of the subarrays.
///
/// ## array
/// An array of elements.
///
/// # Example
/// ```qsharp
/// // same as [[1, 2, 3], [2, 3, 4], [3, 4, 5]]
/// let windows = Windows(3, [1, 2, 3, 4, 5]);
/// ```
///
/// # Remarks
/// The size of the window must be a positive integer no greater than the size of the array
function Windows<'T>(size : Int, array : 'T[]) : 'T[][] {
let arrayLength = Length(array);
Fact(
size > 0 or size <= arrayLength,
"The size of the window must be a positive integer no greater than the size of the array"
);
mutable windows = [];
for index in 0..arrayLength - size {
set windows += [array[index..index + size - 1]];
}
windows
}
/// # Summary
/// Given two arrays, returns a new array of pairs such that each pair
/// contains an element from each original array.
///
/// # Type Parameters
/// ## 'T
/// The type of the left array elements.
/// ## 'U
/// The type of the right array elements.
///
/// # Input
/// ## left
/// An array containing values for the first element of each tuple.
/// ## right
/// An array containing values for the second element of each tuple.
///
/// # Output
/// An array containing pairs of the form `(left[index], right[index])` for
/// each `index`. If the two arrays are not of equal length, the output will
/// be as long as the shorter of the inputs.
///
/// # Example
/// ```qsharp
/// let left = [1, 3, 71];
/// let right = [false, true];
/// let pairs = Zipped(left, right); // [(1, false), (3, true)]
/// ```
///
/// # See Also
/// - [Std.Arrays.Unzipped](xref:Qdk.Std.Arrays.Unzipped)
function Zipped<'T, 'U>(left : 'T[], right : 'U[]) : ('T, 'U)[] {
let arrayLength = MinI(Length(left), Length(right));
mutable zipped = [];
for index in 0..arrayLength - 1 {
set zipped += [(left[index], right[index])];
}
zipped
}
export
All,
Any,
Chunks,
CircularlyShifted,
ColumnAt,
Count,
Diagonal,
DrawMany,
Enumerated,
Excluding,
Filtered,
FlatMapped,
Flattened,
Fold,
ForEach,
Head,
HeadAndRest,
IndexOf,
IndexRange,
Interleaved,
IsEmpty,
IsRectangularArray,
IsSorted,
IsSquareArray,
Mapped,
MappedByIndex,
MappedOverRange,
Most,
MostAndTail,
Padded,
Partitioned,
Rest,
Reversed,
SequenceI,
SequenceL,
Sorted,
Subarray,
Swapped,
Transposed,
Tail,
Unzipped,
Where,
Windows,
Zipped;microsoft/qdk
Publicmirrored fromhttps://github.com/microsoft/qdkAvailable
library/std/src/Std/Arrays.qs
1486lines · modepreview