Java 11, 1325 1379 1356 1336 1290 bytes
import java.math.*;String c(String s)throws Exception{String r="",T=r,a[],b[],z="\\.";int i=0,l,A[],M=0,m=s.length(),j,f=0,q=m;if(s.contains("(")){for(;i<m;){var c=s.charAt(i++);if(f<1){if(c==40){f=1;continue;}r+=c;}else{if(c==41&T.replaceAll("[^(]","").length()==T.replaceAll("[^)]","").length()){r+="x"+s.substring(i);break;}T+=c;}}return c(r.replace("x",c(T)));}else{for(a=s.split("[\\+\\-\\*/]"),A=new int[l=a.length];i<l;f=b.length>1&&(j=b[1].length())>f?j:f)M=(j=(b=a[i++].split(z))[0].length())>M?j:M;for(b=a.clone(),i=0;i<l;A[i]=b[i].contains(".")?j=b[i].length()-1:b[i].replaceAll("0*$","").length(),i++)for(q=(j=b[i].replace(".","").length())<q?j:q,j=a[i].split(z)[0].length();j++<M;)b[i]=0+b[i];double R=new Double(new javax.script.ScriptEngineManager().getEngineByName("JS").eval(s)+""),p;for(int x:A)m=x<m?x:m;m=m==M&R%1==0&(int)R/10%10<1&(j=(r=R+"").split(z)[0].length())>m?j-q>1?q:j:R>99?m:R%10==0?r.length()-1:m<1?1:m;R=new BigDecimal(R).round(new MathContext((R<0?-R:R)<1?m-1:m)).doubleValue();r=(m<M&(p=Math.pow(10,M-m))/10>R?(int)(R/p)*p:R)+"";l=r.length()-2;r=(r=f<1?r.replaceAll(z+"0$",""):r+"0".repeat(f)).substring(0,(j=r.length())<m?j:r.contains(".")?(j=r.replaceAll("^0\\.0+","").length())<m?m-~j:m+1:m);for(i=r.length();i++<l;)r+=0;return r.replaceAll(z+"$","");}}
+54 bytes para corrigir o caso de borda 501*2.0(deu o resultado 1002antes, mas agora está correto 1000).
Agora eu entendo por que este desafio foi respondida por quase dois anos ..>.> Este desafio tem casos mais especial do que a língua holandesa, que está dizendo algo ..
Java não é certamente a linguagem certa para este tipo de desafios (ou qualquer codegolf desafio nesse sentido ..; p), mas é a única linguagem que conheço o suficiente para tentar um desafio difícil como esse.
Formato de entrada como Stringsem espaços (se isso não for permitido, você pode adicionar s=s.replace(" ","")(+19 bytes) na parte superior do método).
Experimente online.
Explicação:
Desculpe pelo longo post.
if(s.contains("(")){
for(;i<m;){
var c=s.charAt(i++);
if(f<1){
if(c==40){
f=1;
continue;}
r+=c;}
else{
if(c==41&T.replaceAll("[^(]","").length()==T.replaceAll("[^)]","").length()){
r+="x"+s.substring(i);
break;}
T+=c;}}
return c(r.replace("x",c(T)));}
Esta parte é usada para entrada contendo parênteses. Ele obterá as partes separadas e usará chamadas recursivas.
0.4*(2*6)torna-se 0.4*A, onde Aé uma chamada recursiva parac(2*6)
(8.3*0.02)+(1.*(9*4)+2.2)torna-se A+B, onde Aé uma chamada recursiva c(8.3*0.02)e Buma chamada recursiva para c(1.*(9*4)+2.2)→ que por sua vez se torna 1.*C+2.2, onde Cé uma chamada recursiva parac(9*4)
for(a=s.split("[\\+\\-\\*/]"),A=new int[l=a.length];
i<l;
f=b.length>1&&(j=b[1].length())>f?j:f)
M=(j=(b=a[i++].split(z))[0].length())>M?j:M;
Esse primeiro loop é usado para preencher os valores Me k, onde Mé o maior comprimento inteiro em relação a números significativos e ko maior comprimento decimal.
1200+3.0torna-se M=2, k=1( 12, .0)
999+2.00torna-se M=3, k=2( 999, .00)
300.+1-300.torna-se M=3, k=0( 300, .)
for(b=a.clone(),i=0;
i<l;
A[i]=b[i].contains(".")?j=b[i].length()-1:b[i].replaceAll("0*$","").length(),i++)
for(q=(j=b[i].replace(".","").length())<q?j:q,
j=a[i].split(z)[0].length();
j++<M;)
b[i]=0+b[i];
Esse segundo loop é usado para preencher as matrizes Ae btambém o valor q, onde Aé a quantidade de números significativos, bmantém os números inteiros com zeros à esquerda correspondentes Me qé o menor comprimento, independentemente dos pontos.
1200+3.0torna-se A=[2, 5] (12, 00030), b=[1200, 0003.0]e q=2( 30)
999+2.00torna-se A=[3, 5] (999, 00200), b=[999, 002.00]e q=3(ambos 999e 200)
300.+1-300.torna-se A=[3, 3, 3] (300, 001, 300), b=[300., 001, 300.]e q=1( 1)
501*2.0torna-se A=[3, 4] (501, 0020), b=[501, 002.0]e q=2( 20)
double R=new Double(new javax.script.ScriptEngineManager().getEngineByName("JS").eval(s)+"")
Usa um mecanismo JavaScript para avaliar a entrada, que será salva Rcomo o dobro.
1200+3.0 torna-se R=1203.0
999+2.00 torna-se R=1001.0
300.+1-300. torna-se R=1.0
for(int x:A)
m=x<m?x:m;
Isso define mo menor valor na matriz A.
A=[2, 5] torna-se m=2
A=[3, 5] torna-se m=3
A=[3, 3, 3] torna-se m=3
m=m==M // If `m` equals `M`
&R%1==0 // and `R` has no decimal values (apart from 0)
&(int)R/10%10<1 // and floor(int(R)/10) modulo-10 is 0
&(j=(r=R+"").split(z)[0].length())>m?
// and the integer-length of R is larger than `m`:
j-q>1? // If this integer-length of `R` minus `q` is 2 or larger:
q // Set `m` to `q` instead
: // Else:
j // Set `m` to this integer-length of `R`
:R>99? // Else-if `R` is 100 or larger:
m // Leave `m` the same
:R%10==0? // Else-if `R` modulo-10 is exactly 0:
r.length()-1 // Set `m` to the total length of `R` (minus the dot)
:m<1? // Else-if `m` is 0:
1 // Set `m` to 1
: // Else:
m; // Leave `m` the same
Isso é modificado com mbase em vários fatores.
999+2.00 = 1001.0& m=3,q=3torna - se m=4(porque m==M(ambos 3) → R%1==0( 1001.0não possui valores decimais) → (int)R/10%10<1( (int)1001.0/10torna-se 100→ 100%10<1) → "1001".length()>m( 4>3) → "1001".length()-q<=1( 4-3<=1) → mtorna-se o comprimento da parte inteira "1001"( 4))
3.839*4 = 15.356& m=1,q=1permanece m=1(porque m==M(ambos 1) → R%1!=0( 15.356tem valores decimais) → R<=99→ R%10!=0( 15.356%10==5.356) → m!=0→ mpermanece o mesmo ( 1))
4*7*3 = 84.0& m=1,q=1permanece m=1(porque m==M(ambos 1) → R%1==0( 84.0não possui valores decimais) → (int)R/10%10>=1( (int)84/10torna-se 8→ 8%10>=1) → R<=99→ R%10!=0( 84%10==4) → m!=0→ mpermanece o mesmo ( 1))
6.0+4.0 = 10.0& m=2,q=2torna - se m=3(porque m!=M( m=2, M=1) → R<=99→ R%10==0( 10%10==0) → mtorna-se o comprimento do total R(menos o ponto) "10.0".length()-1( 3))
0-8.8 = -8.8& m=0,q=1torna - se m=1(porque m!=M( m=0, M=1) → R<=99→ R%10!=0( -8.8%10==-8.8) → m<1→ mtorna - se 1)
501*2.0 = 1001.0& m=3,q=2torna - se m=2(porque m==M(ambos 3) → R%1==0( 1001.0não possui valores decimais) → (int)R/10%10<1( (int)1001.0/10torna-se 100→ 100%10<1) → "1001".length()>m( 4>3) → "1001".length()-q>1( 4-2>1) → mtorna - se q( 2))
R=new BigDecimal(R).round(new MathContext((R<0?-R:R)<1?m-1:m)).doubleValue();
Agora Ré arredondado com base em m.
1001.0e m=4se torna1001.0
0.258& m=3torna - se 0.26(porque abs(R)<1, m-1( 2) em vez de m=3é usado dentro MathContext)
-8.8e m=1se torna-9.0
1002.0e m=2se torna1000.0
m<M&(p=Math.pow(10,M-m))/10>R?(int)(R/p)*p:R;
Isso modifica a parte inteira, Rse necessário.
300.+1-300. = 1.0& m=3,M=3permanece 1.0(porque m>=M→ Rpermanece o mesmo ( 1.0))
0.4*10 = 4.0& m=1,M=2permanece 4.0(porque m<M→ (10^(M-m))/10<=R( (10^1)/10<=4.0→ 10/10<=4.0→ 1<=4.0) → Rpermanece o mesmo ( 4.0))
300+1-300 = 1.0& m=1,M=3torna - se 0.0(porque m<M→ (10^(M-m))/10>R( (10^2)/10>1.0→ 100/10>1.0→ 10>1.0) → Rtorna - se 0.0por causa de int(R/(10^(M-m)))*(10^(M-m))( int(1.0/(10^2))*(10^2)→ int(1.0/100)*100→ 0*100→ 0)
r=(...)+""; // Set `R` to `r` as String (... is the part explained above)
l=r.length()-2; // Set `l` to the length of `R` minus 2
r=(r=k<1? // If `k` is 0 (no decimal values in any of the input-numbers)
r.replaceAll(z+"0$","")
// Remove the `.0` at the end
: // Else:
r+"0".repeat(f)
// Append `k` zeroes after the current `r`
).substring(0, // Then take the substring from index `0` to:
(j=r.length())<m? // If the total length of `r` is below `m`:
j // Leave `r` the same
:r.contains(".")? // Else-if `r` contains a dot
(j=r.replaceAll("^0\\.0+","").length())<m?
// And `R` is a decimal below 1,
// and its rightmost decimal length is smaller than `m`
m-~j // Take the substring from index 0 to `m+j+1`
// where `j` is this rightmost decimal length
: // Else:
m+1 // Take the substring from index 0 to `m+1`
: // Else:
m); // Take the substring from index 0 to `m`
Isso define Ra rcomo corda, e modifica-lo com base em vários fatores.
1203.0& m=4,k=2torna - se 1203.(porque k>=1→ rtorna - se assim 1001.000; r.length()>=m( 8>=4) → r.contains(".")→ r.length()>=m( 8>=4) → substring do índice 0para m+1( 5))
6.9& m=2,k=2permanece 6.9(porque k>=1→ rse torna 6.900; r.length()>=m( 5>=2) → r.contains(".")→ r.length()>=m( 5>=2) → subcadeia do índice 0para m+1( 3))
1.0& m=3,k=0torna - se 1(porque k<1→ rtorna - se assim 1; r.length()<m( 1<3) → substring do índice 0para r.length()( 1))
25.0& m=4,k=4torna - se 25.00(porque k>=1→ rtorna - se assim 25.00000; r.length()>=m( 8>=4) → r.contains(".")→ r.length()>+m( 8>=4) → substring do índice 0para m+1( 5))
0& m=1,k=0permanece 0(porque k<1→ rpermanece assim 0; r.length()>=m( 1>=1) → !r.contains(".")→ substring do índice 0para m( 1))
for(i=r.length();i++<l;)
r+=0;
Isso coloca os zeros à direita novamente na parte inteira, se necessário.
r="12"e R=1200.0se tornar="1200"
r="1"e R=10.0se tornar="10"
r="8"e R=80.0se tornar="80"
return r.replaceAll(z+"$","");
E finalmente retornamos o resultado, depois que removemos os pontos à direita.
1203. torna-se 1203
5. torna-se 5
Definitivamente pode ser jogado por algumas centenas de bytes, mas estou feliz que esteja funcionando agora. Já demorou um pouco para entender cada um dos casos e o que estava sendo solicitado no desafio. E então foram necessárias muitas tentativas e erros, testes e retestes para chegar ao resultado acima. E enquanto escrevia essa explicação acima, eu era capaz de remover outros ± 50 bytes de código não utilizado.
999 + 2.00,.