Ява 11, 1325, 1379, 1356, 1336, 1290 байт.
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 байта, чтобы исправить крайний случай 501*2.0
(давал результат 1002
раньше, но теперь исправлю 1000
).
Теперь я понимаю, почему этот вызов оставался без ответа в течение почти двух лет.>.> В этом вызове есть более особые случаи, чем в голландском языке, который что-то говорит.
Java, безусловно, не является подходящим языком для такого рода вызовов (или для любого программиста) вызов в этом отношении ...; р), но это единственный язык, который я знаю достаточно хорошо, чтобы даже попытаться сделать трудный вызов, как этот.
Формат ввода String
без пробелов (если это не разрешено, вы можете добавить s=s.replace(" ","")
(+19 байт) в начало метода).
Попробуйте онлайн.
Объяснение:
Извините, за длинную статью.
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)));}
Эта часть используется для ввода, содержащего круглые скобки. Он получит разделенные части и будет использовать рекурсивные вызовы.
0.4*(2*6)
становится 0.4*A
, где A
рекурсивный вызовc(2*6)
(8.3*0.02)+(1.*(9*4)+2.2)
становится A+B
, где A
рекурсивный вызов c(8.3*0.02)
и B
рекурсивный вызов c(1.*(9*4)+2.2)
→ который в свою очередь становится 1.*C+2.2
, где C
рекурсивный вызовc(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;
Этот первый цикл используется для заполнения значений M
и k
где M
наибольшая целочисленная длина относительно значащих цифр и k
наибольшая десятичная длина.
1200+3.0
становится M=2, k=1
( 12, .0
)
999+2.00
становится M=3, k=2
( 999, .00
)
300.+1-300.
становится 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];
Этот второй контур используется для заполнения массивов A
и b
так же , как значение q
, где A
это количество значащих цифр, b
провести целые числа с ведущими нулями , чтобы соответствовать M
, а q
самый низкий длина без учета точек.
1200+3.0
становится A=[2, 5] (12, 00030)
, b=[1200, 0003.0]
и q=2
( 30
)
999+2.00
становится A=[3, 5] (999, 00200)
, b=[999, 002.00]
и q=3
(как 999
и 200
)
300.+1-300.
становится A=[3, 3, 3] (300, 001, 300)
, b=[300., 001, 300.]
и q=1
( 1
)
501*2.0
становится A=[3, 4] (501, 0020)
, b=[501, 002.0]
и q=2
( 20
)
double R=new Double(new javax.script.ScriptEngineManager().getEngineByName("JS").eval(s)+"")
Использует движок JavaScript для оценки ввода, который будет сохранен R
как двойной.
1200+3.0
становится R=1203.0
999+2.00
становится R=1001.0
300.+1-300.
становится R=1.0
for(int x:A)
m=x<m?x:m;
Это устанавливает m
наименьшее значение в массиве A
.
A=[2, 5]
становится m=2
A=[3, 5]
становится m=3
A=[3, 3, 3]
становится 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
Это модифицируется m
на основе нескольких факторов.
999+2.00 = 1001.0
& m=3,q=3
становится m=4
(потому что m==M
(оба 3
) → R%1==0
( 1001.0
не имеет десятичных значений) → (int)R/10%10<1
( (int)1001.0/10
становится 100
→ 100%10<1
) → "1001".length()>m
( 4>3
) → "1001".length()-q<=1
( 4-3<=1
) → так m
становится длиной целочисленной части "1001"
( 4
))
3.839*4 = 15.356
& m=1,q=1
остается m=1
(потому что m==M
(оба 1
) → R%1!=0
( 15.356
имеет десятичные значения) → R<=99
→ R%10!=0
( 15.356%10==5.356
) → m!=0
→ m
остается неизменным ( 1
)
4*7*3 = 84.0
& m=1,q=1
остается m=1
(потому что m==M
(оба 1
) → R%1==0
( 84.0
не имеет десятичных значений) → (int)R/10%10>=1
( (int)84/10
становится 8
→ 8%10>=1
) → R<=99
→ R%10!=0
( 84%10==4
) → m!=0
→ m
остается неизменным ( 1
))
6.0+4.0 = 10.0
& m=2,q=2
становится m=3
(потому что m!=M
( m=2, M=1
) → R<=99
→ R%10==0
( 10%10==0
) → так m
становится общей длиной R
(минус точка) "10.0".length()-1
( 3
))
0-8.8 = -8.8
& m=0,q=1
становится m=1
(потому что m!=M
( m=0, M=1
) → R<=99
→ R%10!=0
( -8.8%10==-8.8
) → m<1
→ так m
становится 1
)
501*2.0 = 1001.0
& m=3,q=2
становится m=2
(потому что m==M
(оба 3
) → R%1==0
( 1001.0
не имеет десятичных значений) → (int)R/10%10<1
( (int)1001.0/10
становится 100
→ 100%10<1
) → "1001".length()>m
( 4>3
) → "1001".length()-q>1
( 4-2>1
) → так m
становится q
( 2
))
R=new BigDecimal(R).round(new MathContext((R<0?-R:R)<1?m-1:m)).doubleValue();
Теперь R
округляется на основе m
.
1001.0
& m=4
становится1001.0
0.258
& m=3
становится 0.26
(потому что abs(R)<1
, m-1
( 2
) вместо m=3
используется внутри MathContext
)
-8.8
& m=1
становится-9.0
1002.0
& m=2
становится1000.0
m<M&(p=Math.pow(10,M-m))/10>R?(int)(R/p)*p:R;
Это изменяет целую часть R
при необходимости.
300.+1-300. = 1.0
& m=3,M=3
остается 1.0
(потому что m>=M
→ так же R
остается ( 1.0
))
0.4*10 = 4.0
& m=1,M=2
остается 4.0
(потому что m<M
→ (10^(M-m))/10<=R
( (10^1)/10<=4.0
→ 10/10<=4.0
→ 1<=4.0
) → так же R
остается ( 4.0
)
300+1-300 = 1.0
& m=1,M=3
становится 0.0
(потому что m<M
→ (10^(M-m))/10>R
( (10^2)/10>1.0
→ 100/10>1.0
→ 10>1.0
) → так R
становится 0.0
из-за 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`
Это устанавливает R
в r
качестве строки и модифицирует его на основе нескольких факторов.
1203.0
& m=4,k=2
становится 1203.
(потому что k>=1
→ так r
становится 1001.000
; r.length()>=m
( 8>=4
) → r.contains(".")
→ r.length()>=m
( 8>=4
) → подстрока из индекса 0
в m+1
( 5
))
6.9
& m=2,k=2
остается 6.9
(потому что k>=1
→ так r
становится 6.900
; r.length()>=m
( 5>=2
) → r.contains(".")
→ r.length()>=m
( 5>=2
) → подстрока из индекса 0
в m+1
( 3
))
1.0
& m=3,k=0
становится 1
(потому что k<1
→ так r
становится 1
; r.length()<m
( 1<3
) → подстрока из индекса 0
в r.length()
( 1
))
25.0
& m=4,k=4
становится 25.00
(потому что k>=1
→ так r
становится 25.00000
; r.length()>=m
( 8>=4
) → r.contains(".")
→ r.length()>+m
( 8>=4
) → подстрока из индекса 0
в m+1
( 5
))
0
& m=1,k=0
остается 0
(потому что k<1
→ так r
остается 0
; r.length()>=m
( 1>=1
) → !r.contains(".")
→ подстрока из индекса 0
в m
( 1
))
for(i=r.length();i++<l;)
r+=0;
Это при необходимости возвращает конечные нули обратно к целочисленной части.
r="12"
& R=1200.0
становитсяr="1200"
r="1"
& R=10.0
становитсяr="10"
r="8"
& R=80.0
становитсяr="80"
return r.replaceAll(z+"$","");
И, наконец, мы возвращаем результат, после того как мы удалили все конечные точки.
1203.
становится 1203
5.
становится 5
Можно определенно сыграть пару сотен байтов, но я просто рад, что сейчас это работает. Потребовалось некоторое время, чтобы понять каждый из случаев и то, что было задано в задаче. А потом потребовалось много проб и ошибок, тестирование и повторное тестирование, чтобы получить результат выше. И при написании этого объяснения выше я смог удалить еще ± 50 байт неиспользуемого кода.
999 + 2.00
,.