[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
5.12 Arrays
5.12.1 Slices | Python-style array slices |
Appending []
to a built-in or user-defined type yields an array.
The array element i
of an array A
can be accessed as A[i]
.
By default, attempts to access or assign to an array element using a negative
index generates an error. Reading an array element with an index
beyond the length of the array also generates an error; however,
assignment to an element beyond the length of the array causes the
array to be resized to accommodate the new element.
One can also index an array A
with an integer array B
:
the array A[B]
is formed by indexing array A
with
successive elements of array B
.
A convenient Java-style shorthand exists for iterating over all elements of an
array; see array iteration.
The declaration
real[] A;
initializes A
to be an empty (zero-length) array. Empty arrays should be
distinguished from null arrays. If we say
real[] A=null;
then A
cannot be dereferenced at all (null arrays have no length
and cannot be read from or assigned to).
Arrays can be explicitly initialized like this:
real[] A={0,1,2};
Array assignment in Asymptote
does a shallow copy: only
the pointer is copied (if one copy if modified, the other will be too).
The copy
function listed below provides a deep copy of an array.
Every array A
of type T[]
has the virtual members
-
int length
, -
void cyclic(bool b)
, -
bool cyclicflag
, -
int[] keys
, -
T push(T x)
, -
void append(T[] a)
, -
T pop()
, -
void insert(int i ... T[] x)
, -
void delete(int i, int j=i)
, -
void delete()
, and -
bool initialized(int n)
.
The member A.length
evaluates to the length of the array.
Setting A.cyclic(true)
signifies that array indices should be reduced
modulo the current array length. Reading from or writing to a nonempty
cyclic array never leads to out-of-bounds errors or array resizing. The member
A.cyclicflag
returns the current setting of the cyclic
flag.
The member A.keys
evaluates to an array of integers containing the
indices of initialized entries in the array in ascending order. Hence, for an
array of length n
with all entries initialized, A.keys
evaluates
to the array of integers from 0
to n-1
inclusive. A new keys
array is produced each time A.keys
is evaluated.
The functions A.push
and A.append
append their
arguments onto the end of the array, while A.insert(int i ... T[] x)
inserts x
into the array at index i
.
For convenience A.push
returns the pushed item.
The function A.pop()
pops and returns the last element,
while A.delete(int i, int j=i)
deletes elements with indices in
the range [i
,j
], shifting the position of all higher-indexed
elements down. If no arguments are given, A.delete()
provides a
convenient way of deleting all elements of A
. The routine
A.initialized(int n)
can be used to examine whether the element
at index n
is initialized. Like all Asymptote
functions,
cyclic
, push
, append
, pop
, insert
,
delete
, and initialized
can be "pulled off" of the array
and used on their own. For example,
int[] A={1}; A.push(2); // A now contains {1,2}. A.append(A); // A now contains {1,2,1,2}. int f(int)=A.push; f(3); // A now contains {1,2,1,2,3}. int g()=A.pop; write(g()); // Outputs 3. A.delete(0); // A now contains {2,1,2}. A.delete(0,1); // A now contains {2}. A.insert(1,3); // A now contains {2,3}. A.insert(1 ... A); // A now contains {2,2,3,3} A.insert(2,4,5); // A now contains {2,2,4,5,3,3}.
The []
suffix can also appear after the variable name; this
is sometimes convenient for declaring a list of variables and arrays
of the same type:
real a,A[];
This declares a
to be real
and implicitly declares A
to
be of type real[]
.
In the following list of built-in array functions, T
represents a
generic type. Note that the internal functions alias
, array
,
copy
, concat
, sequence
, map
, and
transpose
, which depend on type T[]
, are defined only after the
first declaration of a variable of type T[]
.
-
new T[]
returns a new empty array of type
T[]
;-
new T[] {list}
returns a new array of type
T[]
initialized withlist
(a comma delimited list of elements).-
new T[n]
returns a new array of
n
elements of typeT[]
. Thesen
array elements are not initialized unless they are arrays themselves (in which case they are each initialized to empty arrays).-
T[] array(int n, T value, int depth=intMax)
returns an array consisting of
n
copies of value. By default, ifvalue
is itself an array, a deep copy of that array is made for each entry in the new array. Ifdepth
is specified, this deep copying only recurses to the number of levels specified.-
int[] sequence(int n)
if
n >= 1
returns the array{0,1,...,n-1}
(otherwise returns a null array);-
int[] sequence(int n, int m)
if
m >= n
returns an array{n,n+1,...,m}
(otherwise returns a null array);-
T[] sequence(T f(int), int n)
if
n >= 1
returns the sequence{f_i :i=0,1,...n-1}
given a functionT f(int)
and integerint n
(otherwise returns a null array);-
T[] map(T f(T), T[] a)
returns the array obtained by applying the function
f
to each element of the arraya
. This is equivalent tosequence(new T(int i) {return f(a[i]);},a.length)
.-
int[] reverse(int n)
if
n >= 1
returns the array{n-1,n-2,...,0}
(otherwise returns a null array);-
int[] complement(int[] a, int n)
returns the complement of the integer array
a
in{0,1,2,...,n-1}
, so thatb[complement(a,b.length)]
yields the complement ofb[a]
.-
real[] uniform(real a, real b, int n)
if
n >= 1
returns a uniform partition of[a,b]
inton
subintervals (otherwise returns a null array);-
int find(bool[], int n=1)
returns the index of the
n
thtrue
value or -1 if not found. Ifn
is negative, search backwards from the end of the array for the-n
th value;-
int search(T[] a, T key)
For built-in ordered types
T
, searches a sorted ordered arraya
ofn
elements to find an interval containingkey
, returning-1
ifkey
is less than the first element,n-1
ifkey
is greater than or equal to the last element, and otherwise the index corresponding to the left-hand (smaller) endpoint.-
T[] copy(T[] a)
returns a deep copy of the array
a
;-
T[][] copy(T[][] a)
returns a deep copy of the array
a
;-
T[][][] copy(T[][][] a)
returns a deep copy of the array
a
;-
T[] concat(... T[][] a)
returns a new array formed by concatenating the arrays given as arguments;
-
bool alias(T[] a, T[] b)
returns
true
if the arraysa
andb
are identical;-
T[] sort(T[] a)
For built-in ordered types
T
, returns a copy ofa
sorted in ascending order;-
T[][] sort(T[][] a)
For built-in ordered types
T
, returns a copy ofa
with the rows sorted by the first column, breaking ties with successively higher columns. For example:string[][] a={{"bob","9"},{"alice","5"},{"pete","7"}, {"alice","4"}}; // Row sort (by column 0, using column 1 to break ties): write(sort(a));
produces
alice 4 alice 5 bob 9 pete 7
-
T[] sort(T[] a, bool compare(T i, T j))
returns a copy of
a
stably sorted in ascending order such that elementi
precedes elementj
ifcompare(i,j)
is true.-
T[][] transpose(T[][] a)
returns the transpose of
a
.-
T[][][] transpose(T[][][] a, int[] perm)
returns the 3D transpose of
a
obtained by applying the permutationperm
ofnew int[]{0,1,2}
to the indices of each entry.-
T sum(T[] a)
For arithmetic types
T
, returns the sum ofa
. In the case whereT
isbool
, the number of true elements ina
is returned.-
T min(T[] a)
-
T min(T[][] a)
-
T min(T[][][] a)
For built-in ordered types
T
, returns the minimum element ofa
.-
T max(T[] a)
-
T max(T[][] a)
-
T max(T[][][] a)
For built-in ordered types
T
, returns the maximum element ofa
.-
T[] min(T[] a, T[] b)
For built-in ordered types
T
, and arraysa
andb
of the same length, returns an array composed of the minimum of the corresponding elements ofa
andb
.-
T[] max(T[] a, T[] b)
For built-in ordered types
T
, and arraysa
andb
of the same length, returns an array composed of the maximum of the corresponding elements ofa
andb
.-
pair[] pairs(real[] x, real[] y);
For arrays
x
andy
of the same length, returns the pair arraysequence(new pair(int i) {return (x[i],y[i]);},x.length)
.-
pair[] fft(pair[] a, int sign=1)
returns the Fast Fourier Transform of
a
(if the optionalFFTW
package is installed), using the givensign
. Here is a simple example:int n=4; pair[] f=sequence(n); write(f); pair[] g=fft(f,-1); write(); write(g); f=fft(g,1); write(); write(f/n);
-
real dot(real[] a, real[] b)
returns the dot product of the vectors
a
andb
.-
real[] tridiagonal(real[] a, real[] b, real[] c, real[] f);
Solve the periodic tridiagonal problem L
x
=f
and return the solutionx
, wheref
is an n vector and L is the n \times n matrix[ b[0] c[0] a[0] ] [ a[1] b[1] c[1] ] [ a[2] b[2] c[2] ] [ ... ] [ c[n-1] a[n-1] b[n-1] ]
For Dirichlet boundary conditions (denoted here by
u[-1]
andu[n]
), replacef[0]
byf[0]-a[0]u[-1]
andf[n-1]-c[n-1]u[n]
; then seta[0]=c[n-1]=0
.-
real[] solve(real[][] a, real[] b, bool warn=true)
Solve the linear equation
a
x=b
by LU decomposition and return the solution x, wherea
is an n \times n matrix andb
is an array of length n. For example:import math; real[][] a={{1,-2,3,0},{4,-5,6,2},{-7,-8,10,5},{1,50,1,-2}}; real[] b={7,19,33,3}; real[] x=solve(a,b); write(a); write(); write(b); write(); write(x); write(); write(a*x);
If
a
is a singular matrix andwarn
isfalse
, return an empty array. If the matrixa
is tridiagonal, the routinetridiagonal
provides a more efficient algorithm (see tridiagonal).-
real[][] solve(real[][] a, real[][] b, bool warn=true)
Solve the linear equation
a
x=b
and return the solution x, wherea
is an n \times n matrix andb
is an n \times m matrix. Ifa
is a singular matrix andwarn
isfalse
, return an empty matrix.-
real[][] identity(int n);
returns the n \times n identity matrix.
-
real[][] diagonal(... real[] a)
returns the diagonal matrix with diagonal entries given by a.
-
real[][] inverse(real[][] a)
returns the inverse of a square matrix
a
.-
real[] quadraticroots(real a, real b, real c);
This numerically robust solver returns the real roots of the quadratic equation ax^2+bx+c=0, in ascending order. Multiple roots are listed separately.
-
pair[] quadraticroots(explicit pair a, explicit pair b, explicit pair c);
This numerically robust solver returns the two complex roots of the quadratic equation ax^2+bx+c=0.
-
real[] cubicroots(real a, real b, real c, real d);
This numerically robust solver returns the real roots of the cubic equation ax^3+bx^2+cx+d=0. Multiple roots are listed separately.
Asymptote
includes a full set of vectorized array instructions for
arithmetic (including self) and logical operations. These
element-by-element instructions are implemented in C++ code for speed. Given
real[] a={1,2}; real[] b={3,2};
then a == b
and a >= 2
both evaluate to the vector
{false, true}
.
To test whether all components of a
and b
agree,
use the boolean function all(a == b)
. One can also use conditionals like
(a >= 2) ? a : b
, which returns the array {3,2}
, or
write((a >= 2) ? a : null
, which returns the array {2}
.
All of the standard built-in libm
functions of signature
real(real)
also take a real array as an argument, effectively like an
implicit call to map
.
As with other built-in types, arrays of the basic data types can be read in by assignment. In this example, the code
file fin=input("test.txt"); real[] A=fin;
reads real values into A
until the end-of-file is reached (or an
I/O error occurs). If line mode is set with line(file)
, then
reading will stop once the end of the line is reached instead
(line mode may be cleared with line(file,false)
):
file fin=input("test.txt"); real[] A=line(fin);
Since string reads by default read up to the end of line anyway, line mode
normally has no effect on string array reads.
However, there is a white-space delimiter mode for reading strings, set with
word(file)
and cleared with word(file,false)
, which causes
string reads to respect white-space delimiters, instead of the default
end-of-line delimiter:
file fin=word(line(input("test.txt"))); real[] A=fin;
Another useful mode is comma-separated-value mode, set with csv(file)
and cleared with csv(file,false)
, which causes reads to respect
comma delimiters:
file fin=csv(input("test.txt")); real[] A=fin;
To restrict the number of values read, use the dimension(file,int)
function:
file fin=input("test.txt"); real[] A=dimension(fin,10);
This reads 10 values into A, unless end-of-file (or end-of-line in line mode) occurs first. Attempting to read beyond the end of the file will produce a runtime error message. Specifying a value of 0 for the integer limit is equivalent to the previous example of reading until end-of-file (or end-of-line in line mode) is encountered.
Two- and three-dimensional arrays of the basic data types can be read in like this:
file fin=input("test.txt"); real[][] A=dimension(fin,2,3); real[][][] B=dimension(fin,2,3,4);
Again, an integer limit of zero means no restriction.
Sometimes the array dimensions are stored with the data as integer
fields at the beginning of an array. Such arrays can be read in with the
functions read1
, read2
, and read3
, respectively:
file fin=input("test.txt"); real[] A=read1(fin); real[][] B=read2(fin); real[][][] C=read3(fin);
One, two, and three-dimensional arrays of the basic data types can be
output with the functions write(file,T[])
,
write(file,T[][])
, write(file,T[][][])
, respectively.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |