svd.cpp:
#include <julia.h>
#include <iostream>
#include <vector>
void print_vector(jl_array_t *vec) {
double *data = (double *) jl_array_data(vec);
size_t n = jl_array_dim(vec, 0);
for(int i = 0; i < n; ++i)
std::cout << data[i] << " ";
std::cout << std::endl;
}
// from column major vector
void print_2d_matrix(jl_array_t *mat) {
double *data = (double *) jl_array_data(mat);
size_t m = jl_array_dim(mat, 0);
size_t n = jl_array_dim(mat, 1);
for(int i = 0; i < m; ++i) {
for(int j = 0; j < n; ++j)
std::cout << data[i+j*m] << " ";
std::cout << std::endl;
}
}
int main() {
// matrix represented as a vector in column-major order
std::vector<double> mat = {
1, 0, 0, 0, 0, 0, 0, 2, 0, 3, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0
};
// initialize Julia
jl_init(NULL);
jl_value_t* dims = (jl_value_t *) jl_eval_string("(4, 5)");
// make sure dims isn't cleaned up by the Julia gc till we're done with it.
JL_GC_PUSH(&dims);
// get the svd function
jl_function_t *svd = jl_get_function(jl_base_module, "svd");
// build a wrapper around the std::vector data to pass our matrix
// to the svd function
jl_value_t* array_type = jl_apply_array_type(jl_float64_type, 2);
jl_array_t *jl_mat = jl_ptr_to_array(array_type, mat.data(), dims, 0);
JL_GC_POP();
// call svd
jl_value_t *ret = jl_call1(svd, (jl_value_t*)jl_mat);
// make sure we don't lose our return data
JL_GC_PUSH1(&ret);
jl_array_t *jl_U = (jl_array_t*)(jl_fieldref(ret, 0));
jl_array_t *jl_S = (jl_array_t*)(jl_fieldref(ret, 1));
jl_array_t *jl_V = (jl_array_t*)(jl_fieldref(ret, 2));
std::cout << "M: " << std::endl;
print_2d_matrix(jl_mat);
std::cout << "U: " << std::endl;
print_2d_matrix(jl_U);
std::cout << "S: " << std::endl;
print_vector(jl_S);
std::cout << "V: " << std::endl;
print_2d_matrix(jl_V);
JL_GC_POP();
jl_atexit_hook(0);
return 0;
}
編譯:
g++ -std=c++14 -fPIC -I$HOME/local/include/julia svd.cpp -L$HOME/local/lib/julia -ljulia
運行:
LD_LIBRARY_PATH=$HOME/local/lib/julia JULIA_HOME=$HOME/local/bin ./a.out
輸出:
M:
1 0 0 0 2
0 0 3 0 0
0 0 0 0 0
0 2 0 0 0
U:
0 1 0 0
1 0 0 0
0 0 0 -1
0 0 1 0
S:
3 2.23607 2 0
V:
-0 0.447214 -0 0
0 0 1 0
1 0 0 0
-0 0 -0 1
0 0.894427 0 0
朱莉婭輸出:
_
_ _ _(_)_ | A fresh approach to technical computing
(_) | (_) (_) | Documentation: http://docs.julialang.org
_ _ _| |_ __ _ | Type "?help" for help.
| | | | | | |/ _` | |
| | |_| | | | (_| | | Version 0.4.6 (2016-06-19 17:16 UTC)
_/ |\__'_|_|_|\__'_| |
|__/ | x86_64-redhat-linux
julia> mat = [1 0 0 0 2; 0 0 3 0 0; 0 0 0 0 0; 0 2 0 0 0]
4x5 Array{Int64,2}:
1 0 0 0 2
0 0 3 0 0
0 0 0 0 0
0 2 0 0 0
julia> svd(mat)
(
4x4 Array{Float64,2}:
0.0 1.0 0.0 0.0
1.0 0.0 0.0 0.0
0.0 0.0 0.0 -1.0
0.0 0.0 1.0 0.0,
[3.0,2.23606797749979,2.0,0.0],
5x4 Array{Float64,2}:
-0.0 0.447214 -0.0 0.0
0.0 0.0 1.0 0.0
1.0 0.0 0.0 0.0
-0.0 0.0 -0.0 1.0
0.0 0.894427 0.0 0.0)
julia>
我無法弄清楚如何構建價值傳遞給'jl_ptr_to_array'沒有調用'jl_eval_string(「(4,5)」)',如果別人知道我將非常感謝幫助。 – Chisholm