Getting started
This short guide explains how to get started with xtensor once you have installed it with one of the methods described in the installation section.
First example
#include <iostream>
#include <xtensor/xarray.hpp>
#include <xtensor/xio.hpp>
#include <xtensor/xview.hpp>
int main(int argc, char* argv[])
{
xt::xarray<double> arr1
{{1.0, 2.0, 3.0},
{2.0, 5.0, 7.0},
{2.0, 5.0, 7.0}};
xt::xarray<double> arr2
{5.0, 6.0, 7.0};
xt::xarray<double> res = xt::view(arr1, 1) + arr2;
std::cout << res << std::endl;
return 0;
}
This example simply adds the second row of a 2-dimensional array with a 1-dimensional array.
Compiling the first example
xtensor is a header-only library, so there is no library to link with. The only constraint
is that the compiler must be able to find the headers of xtensor (and xtl), this is usually done
by having the directory containing the headers in the include path. With G++, use the -I
option
to achieve this. Assuming the first example code is located in example.cpp
, the compilation command
is:
g++ -I /path/to/xtensor/ -I /path/to/xtl/ example.cpp -o example
Note that if you installed xtensor and xtl with Cmake, their headers will be located in the same
directory, so you will need to provide only one path with the -I
option.
When you run the program, it produces the following output:
{7, 11, 14}
Building with cmake
A better alternative for building programs using xtensor is to use Cmake, especially if you are developing for several platforms. Assuming the following folder structure:
first_example
|- src
| |- example.cpp
|- CMakeLists.txt
The following minimal CMakeLists.txt
is enough to build the first example:
cmake_minimum_required(VERSION 3.1)
project(first_example)
find_package(xtl REQUIRED)
find_package(xtensor REQUIRED)
add_executable(first_example src/example.cpp)
if(MSVC)
set(CMAKE_EXE_LINKER_FLAGS /MANIFEST:NO)
endif()
target_link_libraries(first_example xtensor xtensor::optimize xtensor::use_xsimd)
Note
target_link_libraries(... xtensor::optimize)
set the following compiler flags, if supported by the target compiler:
Unix:
-march=native
;Windows:
/EHsc /MP /bigobj
.
This may speed-up your code, but renders it hardware dependent.
Note
target_link_libraries(... xtensor::use_xsimd)
enables xsimd: an optional dependency of xtensor that enables simd acceleration, i.e. executing a same operation on a batch of data in a single CPU instruction. This is well-suited to improve performance when operating on tensors, but renders it hardware dependent.
cmake has to know where to find the headers, this is done through the CMAKE_INSTALL_PREFIX
variable. Note that CMAKE_INSTALL_PREFIX
is usually the path to a folder containing the following
subfolders: include
, lib
and bin
, so you don’t have to pass any additional option for linking.
Examples of valid values for CMAKE_INSTALL_PREFIX
on Unix platforms are /usr/local
, /opt
.
The following commands create a directory for building (avoid building in the source folder), builds the first example with cmake and then runs the program:
mkdir build
cd build
cmake -DCMAKE_INSTALL_PREFIX=your_prefix ..
make
./first_program
See Build and configuration for more details about the build options.
Second example: reshape
This second example initializes a 1-dimensional array and reshapes it in-place:
#include <iostream>
#include <xtensor/xarray.hpp>
#include <xtensor/xio.hpp>
int main(int argc, char* argv[])
{
xt::xarray<int> arr
{1, 2, 3, 4, 5, 6, 7, 8, 9};
arr.reshape({3, 3});
std::cout << arr;
return 0;
}
When compiled and run, this produces the following output:
{{1, 2, 3},
{4, 5, 6},
{7, 8, 9}}
Tip
To print the shape to the standard output you can use either:
const auto& s = arr.shape();
std::copy(s.cbegin(), s.cend(), std::ostream_iterator<double>(std::cout, " "));
Or:
std::cout << xt::adapt(arr.shape()); // with: #include <xtensor/xadapt.hpp>
Third example: index access
#include <iostream>
#include <xtensor/xarray.hpp>
#include <xtensor/xio.hpp>
int main(int argc, char* argv[])
{
xt::xarray<double> arr1
{{1.0, 2.0, 3.0},
{2.0, 5.0, 7.0},
{2.0, 5.0, 7.0}};
std::cout << arr1(0, 0) << std::endl;
xt::xarray<int> arr2
{1, 2, 3, 4, 5, 6, 7, 8, 9};
std::cout << arr2(0);
return 0;
}
Outputs:
1.0
1
Fourth example: broadcasting
This last example shows how to broadcast the xt::pow()
universal function:
#include <iostream>
#include <xtensor/xarray.hpp>
#include <xtensor/xmath.hpp>
#include <xtensor/xio.hpp>
int main(int argc, char* argv[])
{
xt::xarray<double> arr1
{1.0, 2.0, 3.0};
xt::xarray<unsigned int> arr2
{4, 5, 6, 7};
arr2.reshape({4, 1});
xt::xarray<double> res = xt::pow(arr1, arr2);
std::cout << res;
return 0;
}
Outputs:
{{1, 16, 81},
{1, 32, 243},
{1, 64, 729},
{1, 128, 2187}}