⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 complex.pm

📁 MSYS在windows下模拟了一个类unix的终端
💻 PM
📖 第 1 页 / 共 3 页
字号:
	if (ref $self) {			# Called as an object method	    if (exists $self->{display_format}) {		my %obj = %{$self->{display_format}};		@display_format{keys %obj} = values %obj;	    }	}	if (@_ == 1) {	    $display_format{style} = shift;	} else {	    my %new = @_;	    @display_format{keys %new} = values %new;	}	if (ref $self) { # Called as an object method	    $self->{display_format} = { %display_format };	    return		wantarray ?		    %{$self->{display_format}} :		    $self->{display_format}->{style};	}        # Called as a class method	%DISPLAY_FORMAT = %display_format;	return	    wantarray ?		%DISPLAY_FORMAT :		    $DISPLAY_FORMAT{style};}## (stringify)## Show nicely formatted complex number under its cartesian or polar form,# depending on the current display format:## . If a specific display format has been recorded for this object, use it.# . Otherwise, use the generic current default for all complex numbers,#   which is a package global variable.#sub stringify {	my ($z) = shift;	my $style = $z->display_format;	$style = $DISPLAY_FORMAT{style} unless defined $style;	return $z->stringify_polar if $style =~ /^p/i;	return $z->stringify_cartesian;}## ->stringify_cartesian## Stringify as a cartesian representation 'a+bi'.#sub stringify_cartesian {	my $z  = shift;	my ($x, $y) = @{$z->cartesian};	my ($re, $im);	my %format = $z->display_format;	my $format = $format{format};	if ($x) {	    if ($x =~ /^NaN[QS]?$/i) {		$re = $x;	    } else {		if ($x =~ /^-?$Inf$/oi) {		    $re = $x;		} else {		    $re = defined $format ? sprintf($format, $x) : $x;		}	    }	} else {	    undef $re;	}	if ($y) {	    if ($y =~ /^(NaN[QS]?)$/i) {		$im = $y;	    } else {		if ($y =~ /^-?$Inf$/oi) {		    $im = $y;		} else {		    $im =			defined $format ?			    sprintf($format, $y) :			    ($y == 1 ? "" : ($y == -1 ? "-" : $y));		}	    }	    $im .= "i";	} else {	    undef $im;	}	my $str = $re;	if (defined $im) {	    if ($y < 0) {		$str .= $im;	    } elsif ($y > 0 || $im =~ /^NaN[QS]?i$/i)  {		$str .= "+" if defined $re;		$str .= $im;	    }	} elsif (!defined $re) {	    $str = "0";	}	return $str;}## ->stringify_polar## Stringify as a polar representation '[r,t]'.#sub stringify_polar {	my $z  = shift;	my ($r, $t) = @{$z->polar};	my $theta;	my %format = $z->display_format;	my $format = $format{format};	if ($t =~ /^NaN[QS]?$/i || $t =~ /^-?$Inf$/oi) {	    $theta = $t; 	} elsif ($t == pi) {	    $theta = "pi";	} elsif ($r == 0 || $t == 0) {	    $theta = defined $format ? sprintf($format, $t) : $t;	}	return "[$r,$theta]" if defined $theta;	#	# Try to identify pi/n and friends.	#	$t -= int(CORE::abs($t) / pit2) * pit2;	if ($format{polar_pretty_print} && $t) {	    my ($a, $b);	    for $a (2..9) {		$b = $t * $a / pi;		if ($b =~ /^-?\d+$/) {		    $b = $b < 0 ? "-" : "" if CORE::abs($b) == 1;		    $theta = "${b}pi/$a";		    last;		}	    }	}        if (defined $format) {	    $r     = sprintf($format, $r);	    $theta = sprintf($format, $theta) unless defined $theta;	} else {	    $theta = $t unless defined $theta;	}	return "[$r,$theta]";}1;__END__=pod=head1 NAMEMath::Complex - complex numbers and associated mathematical functions=head1 SYNOPSIS	use Math::Complex;	$z = Math::Complex->make(5, 6);	$t = 4 - 3*i + $z;	$j = cplxe(1, 2*pi/3);=head1 DESCRIPTIONThis package lets you create and manipulate complex numbers. By default,I<Perl> limits itself to real numbers, but an extra C<use> statement bringsfull complex support, along with a full set of mathematical functionstypically associated with and/or extended to complex numbers.If you wonder what complex numbers are, they were invented to be able to solvethe following equation:	x*x = -1and by definition, the solution is noted I<i> (engineers use I<j> instead sinceI<i> usually denotes an intensity, but the name does not matter). The numberI<i> is a pure I<imaginary> number.The arithmetics with pure imaginary numbers works just like you would expectit with real numbers... you just have to remember that	i*i = -1so you have:	5i + 7i = i * (5 + 7) = 12i	4i - 3i = i * (4 - 3) = i	4i * 2i = -8	6i / 2i = 3	1 / i = -iComplex numbers are numbers that have both a real part and an imaginarypart, and are usually noted:	a + biwhere C<a> is the I<real> part and C<b> is the I<imaginary> part. Thearithmetic with complex numbers is straightforward. You have tokeep track of the real and the imaginary parts, but otherwise therules used for real numbers just apply:	(4 + 3i) + (5 - 2i) = (4 + 5) + i(3 - 2) = 9 + i	(2 + i) * (4 - i) = 2*4 + 4i -2i -i*i = 8 + 2i + 1 = 9 + 2iA graphical representation of complex numbers is possible in a plane(also called the I<complex plane>, but it's really a 2D plane).The number	z = a + biis the point whose coordinates are (a, b). Actually, it wouldbe the vector originating from (0, 0) to (a, b). It follows that the additionof two complex numbers is a vectorial addition.Since there is a bijection between a point in the 2D plane and a complexnumber (i.e. the mapping is unique and reciprocal), a complex numbercan also be uniquely identified with polar coordinates:	[rho, theta]where C<rho> is the distance to the origin, and C<theta> the angle betweenthe vector and the I<x> axis. There is a notation for this using theexponential form, which is:	rho * exp(i * theta)where I<i> is the famous imaginary number introduced above. Conversionbetween this form and the cartesian form C<a + bi> is immediate:	a = rho * cos(theta)	b = rho * sin(theta)which is also expressed by this formula:	z = rho * exp(i * theta) = rho * (cos theta + i * sin theta)In other words, it's the projection of the vector onto the I<x> and I<y>axes. Mathematicians call I<rho> the I<norm> or I<modulus> and I<theta>the I<argument> of the complex number. The I<norm> of C<z> will benoted C<abs(z)>.The polar notation (also known as the trigonometricrepresentation) is much more handy for performing multiplications anddivisions of complex numbers, whilst the cartesian notation is bettersuited for additions and subtractions. Real numbers are on the I<x>axis, and therefore I<theta> is zero or I<pi>.All the common operations that can be performed on a real number havebeen defined to work on complex numbers as well, and are merelyI<extensions> of the operations defined on real numbers. This meansthey keep their natural meaning when there is no imaginary part, providedthe number is within their definition set.For instance, the C<sqrt> routine which computes the square root ofits argument is only defined for non-negative real numbers and yields anon-negative real number (it is an application from B<R+> to B<R+>).If we allow it to return a complex number, then it can be extended tonegative real numbers to become an application from B<R> to B<C> (theset of complex numbers):	sqrt(x) = x >= 0 ? sqrt(x) : sqrt(-x)*iIt can also be extended to be an application from B<C> to B<C>,whilst its restriction to B<R> behaves as defined above by usingthe following definition:	sqrt(z = [r,t]) = sqrt(r) * exp(i * t/2)Indeed, a negative real number can be noted C<[x,pi]> (the modulusI<x> is always non-negative, so C<[x,pi]> is really C<-x>, a negativenumber) and the above definition states that	sqrt([x,pi]) = sqrt(x) * exp(i*pi/2) = [sqrt(x),pi/2] = sqrt(x)*iwhich is exactly what we had defined for negative real numbers above.The C<sqrt> returns only one of the solutions: if you want the both,use the C<root> function.All the common mathematical functions defined on real numbers thatare extended to complex numbers share that same property of workingI<as usual> when the imaginary part is zero (otherwise, it would notbe called an extension, would it?).A I<new> operation possible on a complex number that isthe identity for real numbers is called the I<conjugate>, and is notedwith an horizontal bar above the number, or C<~z> here.	 z = a + bi	~z = a - biSimple... Now look:	z * ~z = (a + bi) * (a - bi) = a*a + b*bWe saw that the norm of C<z> was noted C<abs(z)> and was defined as thedistance to the origin, also known as:	rho = abs(z) = sqrt(a*a + b*b)so	z * ~z = abs(z) ** 2If z is a pure real number (i.e. C<b == 0>), then the above yields:	a * a = abs(a) ** 2which is true (C<abs> has the regular meaning for real number, i.e. standsfor the absolute value). This example explains why the norm of C<z> isnoted C<abs(z)>: it extends the C<abs> function to complex numbers, yetis the regular C<abs> we know when the complex number actually has noimaginary part... This justifies I<a posteriori> our use of the C<abs>notation for the norm.=head1 OPERATIONSGiven the following notations:	z1 = a + bi = r1 * exp(i * t1)	z2 = c + di = r2 * exp(i * t2)	z = <any complex or real number>the following (overloaded) operations are supported on complex numbers:	z1 + z2 = (a + c) + i(b + d)	z1 - z2 = (a - c) + i(b - d)	z1 * z2 = (r1 * r2) * exp(i * (t1 + t2))	z1 / z2 = (r1 / r2) * exp(i * (t1 - t2))	z1 ** z2 = exp(z2 * log z1)	~z = a - bi	abs(z) = r1 = sqrt(a*a + b*b)	sqrt(z) = sqrt(r1) * exp(i * t/2)	exp(z) = exp(a) * exp(i * b)	log(z) = log(r1) + i*t	sin(z) = 1/2i (exp(i * z1) - exp(-i * z))	cos(z) = 1/2 (exp(i * z1) + exp(-i * z))	atan2(z1, z2) = atan(z1/z2)The following extra operations are supported on both real and complexnumbers:	Re(z) = a	Im(z) = b	arg(z) = t	abs(z) = r	cbrt(z) = z ** (1/3)	log10(z) = log(z) / log(10)	logn(z, n) = log(z) / log(n)	tan(z) = sin(z) / cos(z)	csc(z) = 1 / sin(z)	sec(z) = 1 / cos(z)	cot(z) = 1 / tan(z)	asin(z) = -i * log(i*z + sqrt(1-z*z))	acos(z) = -i * log(z + i*sqrt(1-z*z))	atan(z) = i/2 * log((i+z) / (i-z))	acsc(z) = asin(1 / z)	asec(z) = acos(1 / z)	acot(z) = atan(1 / z) = -i/2 * log((i+z) / (z-i))	sinh(z) = 1/2 (exp(z) - exp(-z))	cosh(z) = 1/2 (exp(z) + exp(-z))	tanh(z) = sinh(z) / cosh(z) = (exp(z) - exp(-z)) / (exp(z) + exp(-z))	csch(z) = 1 / sinh(z)	sech(z) = 1 / cosh(z)	coth(z) = 1 / tanh(z)	asinh(z) = log(z + sqrt(z*z+1))	acosh(z) = log(z + sqrt(z*z-1))	atanh(z) = 1/2 * log((1+z) / (1-z))	acsch(z) = asinh(1 / z)	asech(z) = acosh(1 / z)	acoth(z) = atanh(1 / z) = 1/2 * log((1+z) / (z-1))I<arg>, I<abs>, I<log>, I<csc>, I<cot>, I<acsc>, I<acot>, I<csch>,I<coth>, I<acosech>, I<acotanh>, have aliases I<rho>, I<theta>, I<ln>,I<cosec>, I<cotan>, I<acosec>, I<acotan>, I<cosech>, I<cotanh>,I<acosech>, I<acotanh>, respectively.  C<Re>, C<Im>, C<arg>, C<abs>,C<rho>, and C<theta> can be used also also mutators.  The C<cbrt>returns only one of the solutions: if you want all three, use theC<root> function.The I<root> function is available to compute all the I<n>roots of some complex, where I<n> is a strictly positive integer.There are exactly I<n> such roots, returned as a list. Getting thenumber mathematicians call C<j> such that:	1 + j + j*j = 0;is a simple matter of writing:	$j = ((root(1, 3))[1];The I<k>th root for C<z = [r,t]> is given by:	(root(z, n))[k] = r**(1/n) * exp(i * (t + 2*k*pi)/n)The I<spaceship> comparison operator, E<lt>=E<gt>, is also defined. Inorder to ensure its restriction to real numbers is conform to what youwould expect, the comparison is run on the real part of the complexnumber first, and imaginary parts are compared only when the realparts match.=head1 CREATIONTo create a complex number, use either:	$z = Math::Complex->make(3, 4);	$z = cplx(3, 4);if you know the cartesian form of the number, or	$z = 3 + 4*i;if you like. To create a number using the polar form, use either:	$z = Math::Complex->emake(5, pi/3);	$x = cplxe(5, pi/3);instead. The first argument is the modulus, the second is the angle(in radians, the full circle is 2*pi).  (Mnemonic: C<e> is used as anotation for complex numbers in the polar form).It is possible to write:	$x = cplxe(-3, pi/4);but that will be silently converted into C<[3,-3pi/4]>, since themodulus must be non-negative (it represents the distance to the originin the complex plane).It is also possible to have a complex number as either argument ofeither the C<make> or C<emake>: the appropriate component ofthe argument will be used.	$z1 = cplx(-2,  1);	$z2 = cplx($z1, 4);=head1 STRINGIFICATIONWhen printed, a complex number is usually shown under its cartesianstyle I<a+bi>, but there are legitimate cases where the polar styleI<[r,t]> is more appropriate.By calling the class method C<Math::Complex::display_format> andsupplying either C<"polar"> or C<"cartesian"> as an argument, youoverride the default display style, which is C<"cartesian">. Notsupplying any argument returns the current settings.This default can be overridden on a per-number basis by calling theC<display_format> method instead. As before, not supplying any argumentreturns the current display style for this number. Otherwise whatever youspecify will be the new display style for I<this> particular number.For instance:	use Math::Complex;	Math::Complex::display_format('polar');	$j = (root(1, 3))[1];	print "j = $j\n";		# Prints "j = [1,2pi/3]"	$j->display_format('cartesian');	print "j = $j\n";		# Prints "j = -0.5+0.866025403784439i"The polar style attempts to emphasize arguments like I<k*pi/n>(where I<n> is a positive integer and I<k> an integer within [-9, +9]),this is called I<polar pretty-printing>.=head2 CHANGED IN PERL 5.6The C<display_format> class method and the correspondingC<display_format> object method can now be called usinga parameter hash instead of just a one parameter.The old display format style, which can have values C<"cartesian"> orC<"polar">, can be changed using the C<"style"> parameter.	$j->display_format(style => "polar");The one parameter calling convention also still works.	$j->display_format("polar");There are two new display parameters.The first one is C<"format">, which is a sprintf()-style format stringto be used for both numeric parts of the complex number(s).  The issomewhat system-dependent but most often it corresponds to C<"%.15g">.You can revert to the default by setting the C<format> to C<undef>.	# the $j from the above example	$j->display_format('format' => '%.5f');	print "j = $j\n";		# Prints "j = -0.50000+0.86603i"	$j->display_format('format' => undef);	print "j = $j\n";		# Prints "j = -0.5+0.86603i"Notice that this affects also the return values of theC<display_format> methods: in list context the whole parameter hashwill be returned, as opposed to only the style parameter value.This is a potential incompatibility with earlier versions if youhave been calling the C<display_format> method in list context.The second new display parameter is C<"polar_pretty_print">, which canbe set to true or false, the default being true.  See the previoussection for what this means.=head1 USAGEThanks to overloading, the handling of arithmetics with complex numbersis simple and almost transparent.Here are some examples:	use Math::Complex;	$j = cplxe(1, 2*pi/3);	# $j ** 3 == 1	print "j = $j, j**3 = ", $j ** 3, "\n";	print "1 + j + j**2 = ", 1 + $j + $j**2, "\n";	$z = -16 + 0*i;			# Force it to be a complex	print "sqrt($z) = ", sqrt($z), "\n";	$k = exp(i * 2*pi/3);	print "$j - $k = ", $j - $k, "\n";	$z->Re(3);			# Re, Im, arg, abs,	$j->arg(2);			# (the last two aka rho, theta)					# can be used also as mutators.=head1 ERRORS DUE TO DIVISION BY ZERO OR LOGARITHM OF ZEROThe division (/) and the following functions	log	ln	log10	logn	tan	sec	csc	cot	atan	asec	acsc	acot	tanh	sech	csch	coth	atanh	asech	acsch	acothcannot be computed for all arguments because that would mean dividingby zero or taking logarithm of zero. These situations cause fatalruntime errors looking like this	cot(0): Division by zero.	(Because in the definition of cot(0), the divisor sin(0) is 0)	Died at ...or	atanh(-1): Logarithm of zero.	Died at...For the C<csc>, C<cot>, C<asec>, C<acsc>, C<acot>, C<csch>, C<coth>,C<asech>, C<acsch>, the argument cannot be C<0> (zero).  For the thelogarithmic functions and the C<atanh>, C<acoth>, the argument cannotbe C<1> (one).  For the C<atanh>, C<acoth>, the argument cannot beC<-1> (minus one).  For the C<atan>, C<acot>, the argument cannot beC<i> (the imaginary unit).  For the C<atan>, C<acoth>, the argumentcannot be C<-i> (the negative imaginary unit).  For the C<tan>,C<sec>, C<tanh>, the argument cannot be I<pi/2 + k * pi>, where I<k>is any integer.Note that because we are operating on approximations of real numbers,these errors can happen when merely `too close' to the singularitieslisted above.=head1 ERRORS DUE TO INDIGESTIBLE ARGUMENTSThe C<make> and C<emake> accept both real and complex arguments.When they cannot recognize the arguments they will die with errormessages like the following    Math::Complex::make: Cannot take real part of ...    Math::Complex::make: Cannot take real part of ...    Math::Complex::emake: Cannot take rho of ...    Math::Complex::emake: Cannot take theta of ...=head1 BUGSSaying C<use Math::Complex;> exports many mathematical routines in thecaller environment and even overrides some (C<sqrt>, C<log>).This is construed as a feature by the Authors, actually... ;-)All routines expect to be given real or complex numbers. Don't attempt touse BigFloat, since Perl has currently no rule to disambiguate a '+'operation (for instance) between two overloaded entities.In Cray UNICOS there is some strange numerical instability that resultsin root(), cos(), sin(), cosh(), sinh(), losing accuracy fast.  Beware.The bug may be in UNICOS math libs, in UNICOS C compiler, in Math::Complex.Whatever it is, it does not manifest itself anywhere else where Perl runs.=head1 AUTHORSRaphael Manfredi <F<Raphael_Manfredi@pobox.com>> andJarkko Hietaniemi <F<jhi@iki.fi>>.Extensive patches by Daniel S. Lewart <F<d-lewart@uiuc.edu>>.=cut1;# eof

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -