我不得不对MySQL查询的结果进行四舍五入。我没有找到一个正确的四舍五入方法(在我看来,截断不是正确的方法)。
我编写了这段代码,它对浮点数进行舍入,并以西班牙-墨西哥的方式格式化这些数字(使用逗号表示千位、百万位等)。(如1425 -〉1,1425)
我知道这是非常低效的,但它以正确的方式完成工作。如果你有任何建议,请让我知道,使它更有效率。
对不起,西班牙语变量和注解..我是墨西哥人!!!;)
sub formatea_numero # recibe una cifra sin formato y le agrega comas por cada 3 digitos
{
my($importe_orig,$dec_a_redondear) = @_;
my @digitos_corregido;
my $contador=0;
my $signo;
my $importex;
my $decimal_con_ceros_al_inicio = 0;
### SIGNO ######################
### Obteniendo el signo si lo tiene
if($importe_orig=~/([-|+])(.*)/) # SI TIENE SIGNO $1 EL NUM ES $2
{
$signo = $1;
$importex = substr $importe_orig, 1;
}
else # NO TIENE SIGNO EL NUM ES $importe_orig
{
$importex = $importe_orig;
}
### DECIMALES ################
### Si tiene decimales.. se obtiene la parte entera y la parte decimal
if($importex=~/\./) # Tiene decimales
{
if ($dec_a_redondear ne "" && $dec_a_redondear == 0) # Caso Especial de que son 0 decimales a redondear
{
$importex=~/(.*)\.(.)(.*)/;
$parte_entera = $1;
$primer_decimal = $2;
$resto_decimal = $3;
if ($primer_decimal >= 5)
{$parte_entera = $parte_entera+1;}
}
else
{
if($importex=~/(.*)\.(.+)/) # Parte entera $1 Parte decimal $2
{
$parte_entera = $1;
$parte_decimal = $2;
if ($parte_decimal=~/^(0+).*/ ) # Caso especial si el decimal inicia en 0 ejem 3.0015 1.0000003 1.04 etc
{
$decimal_con_ceros_al_inicio =1;
$parte_decimal = "1" . $parte_decimal;
}
}
########### REDONDEAR DECIMALES
if ($decimal_con_ceros_al_inicio)
{
$dec_a_redondear = 1 + $dec_a_redondear;
$num_decimales_original = 1+ length $parte_decimal;
}
else
{ $num_decimales_original = length $parte_decimal;}
if ($dec_a_redondear>=0 && $dec_a_redondear < $num_decimales_original)
{
$parte_decimal_1 = substr($parte_decimal,0,$dec_a_redondear); # se obtienen los digitos hasta el numero de decimales que se quiere redondear
$siguiente_decimal = substr($parte_decimal,$dec_a_redondear,1); # se obtiene el primer dígito a descartar .. si es mayor que 5 se le agrega uno al anterior digito
if($siguiente_decimal >=5)
{
$largo_inicial_parte_decimal_1 = length $parte_decimal_1;
$parte_decimal_1 = $parte_decimal_1 +1;
$largo_final_parte_decimal_1 = length $parte_decimal_1;
if($largo_final_parte_decimal_1 <= $largo_inicial_parte_decimal_1)
{
$parte_decimal = $parte_decimal_1;
}
else
{
$parte_entera = $parte_entera + 1;
$parte_decimal = $parte_decimal_1 - 1;
$parte_decimal = 0;
}
}
else
{
$parte_decimal = $parte_decimal_1;
}
} # cierra if ($dec_a_redondear>0 && $dec_a_redondear < $num_decimales_original)
if ($decimal_con_ceros_al_inicio)
{
$parte_decimal=~/^1(.+)/;
$parte_decimal = $1;
}
########### TERMINA DECIMALES (redondeando)
} # Cierra if ($dec_a_redondear ne "" && $dec_a_redondear == 0) # Caso Especial de que son 0 decimales a redondear
} # cierra if($importex=~/\./) # Tiene decimales
########### FORMATEANDO LOS MILES ###########
if($importex=~/\./) # Tiene decimales
{
@digitos = split(//,$parte_entera);
}
else # No tiene decimales
{
@digitos = split(//,$importex);
}
@digitos= reverse(@digitos);
foreach $digito(@digitos)
{
if ($contador ==3)
{
push (@digitos_corregido,",");
push (@digitos_corregido,$digito);
$contador=1;
}
else
{
push (@digitos_corregido,$digito);
$contador++;
}
}
@digitos_corregido = reverse(@digitos_corregido);
$importe2 = join('',@digitos_corregido);
### Termina de procesar la parte entera
### Se integra el signo, la parte entera formateada y la parte decimal
if($importex=~/\./)
{
if ($dec_a_redondear ne "" && $dec_a_redondear == 0)
{ $importe2 = $importe2;}
else
{
if ($parte_decimal >0)
{$importe2 = $importe2 . "." .$parte_decimal;}
else
{$importe2 = $importe2;}
}
}
if($importe_orig=~/([-|+])(.*)/)
{
$importe2 = $signo . $importe2 ;
}
return $importe2;
} # cierra sub formatea_numero # recibe una cifra sin formato y le agrega comas por cada 3 digitos
1条答案
按热度按时间brccelvz1#
这似乎是一个很大的代码四舍五入一个数字。我不能完全遵循所有的西班牙语细节,但如果你没有特殊的标准,那么基本的可能就足够了,与sprintf
(This我会把
.00
加到一个整数上,所以我的条件是.
,最好是一个合法的数字。)sprintf
使用舍入到偶数的一半,根据IEEE754舍入到最接近的(整数)关系到偶数规则。(Windows可能不同,因为它不遵守IEEE规范,但Strawberry Perl确实舍入到偶数,感谢ikegami的注解。)也有一些库,比如Math::Round或Math::BigFloat。请仔细研究这个主题,特别是如果你的应用程序对细节很敏感(比如金融或科学领域),因为使用浮点运算充满了棘手的点。另请参见perlfaq4中的舍入(等)
然后,您可以添加“千位分隔符”(根据需要使用
,
),方法是从后面(在子字符串中直到小数点)用逗号+本身替换每个三位数字。请参阅perlfaq5了解更多信息