-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathREADME
executable file
·133 lines (89 loc) · 4.03 KB
/
README
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
This module wraps the double complex arithmetic functions defined in complex.h.
Consequently, you need a C compiler that provides complex.h.
Build in the usual way:
perl Makefile.PL
make
make test
make install
=====================
Which Math::Complex_C
=====================
There are three Math::Complex_C modules, each now in it's own separate distro:
1) Math::Complex_C
Accesses C's 'double' precision (53-bit) complex arithemtic operations;
2) Math::Complex_C::L
Accesses C's 'long double' precision (usually 64-bit) complex arithemtic operations;
3) Math::Complex_C::Q
Accesses C's '__float128' precision (113-bit) complex arithemtic operations;
You can use any/all of those 3 modules (assuming they build ok for you) on any perl - irrespective
of whether your NV-type is 'double', 'long double' or '__float128' - though you need to go about
things a little judiciously if the precision of your NV is less than the precision of the complex
arithemtic operations for the Math::Complex_C module that you're using.
For example, if your NV is double, and you're using Math::Complex_C::Q:
########################
use warnings;
use Math::Complex_C::Q qw(:all);
$obj = sqrt(MCQ(-2.123, 0));
$nv = imag_cq($obj); # returns NV
$str = imag_cq2str($obj);# returns string
print "$nv\n$str\n";
########################
This outputs:
1.45705181788432
1.45705181788431952124809953480556e+00
Clearly, you've lost the full __float128 precision by calling imag_cq(), whereas imag_cq2str()
allows you to capture the full precision.
But there's another loss of precision in the above. When the NV (bareword) -2.123 was assigned
it was assigned only with 'double' (53-bit) precision. Assigning the value with full 113-bit
precision is quite simple - we just have to quote the bareword '-2.123':
########################
use warnings;
use Math::Complex_C::Q qw(:all);
$obj = sqrt(MCQ('-2.123', 0));
$nv = imag_cq($obj); # returns NV
$str = imag_cq2str($obj);# returns string
print "$nv\n$str\n";
########################
This doesn't change the value of $nv, but $str changes to:
1.45705181788431944566113502812563e+00
which is now (hopefully) the correct 33-digit approximation of sqrt(2.123).
Another alternative is to assign to a Math::Float128 object instead of to a string:
########################
use warnings;
use Math::Complex_C::Q qw(:all);
use Math::Float128 qw(:all);
$obj = sqrt(MCQ('-2.123', 0));
# $obj = sqrt(MCQ(Math::Float128->new('-2.123'), 0)); # same result
$f128_obj = imag_cq2F($obj);# returns Math::Float128 object
print "$f128_obj\n";
########################
This also prints out:
1.45705181788431944566113502812563e+00
but this time the value is stored in a Math::Float128 object, not a string.
So ... just make sure that values are assigned/retrieved as either strings or Math::Float128 objects.
The capability for this is provided.
Similarly if you're using Math::Complex_C::L on a perl whose NV is 'double' - grab/assign
the values as either strings or Math::LongDouble objects.
For example, again with NV type of double:
##############################
use warnings;
use Math::Complex_C::L qw(:all);
use Math::LongDouble qw(:all);
$obj = sqrt(MCL('-2.123', 0));
# $obj = sqrt(MCL(Math::LongDouble->new('-2.123'), 0)); # same result
$nv = imag_cl($obj); # returns NV
$str = imag_cl2str($obj); # returns string
$ld_obj = imag_cl2LD($obj);# returns Math::LongDouble object
print "$nv\n$str\n$ld_obj\n";
##############################
This outputs:
1.45705181788432
1.45705181788431945e+000
1.45705181788431945e+000
If your NV precision matches the precision of the complex (real and imaginary) parts then
assigning/retrieving values as NVs is fine.
If your NV precision is greater than the precision of the complex components then you're simply
wasting the extra precision your NV has - which is something you are entirely free to do.
In short, I'd recommend using Math::Complex_C::Q - unless you have some reason to not do so.
Valid reasons would include not needing that level of precision or quadmath not being available
for your system.