一、界面
界面的实现在参照课本例子的基础上,做了部分改动。总体使用GridLayout布局,设置7行4列。两个TextView组件分别显示输入的算式和计算结果,以及一个清除键,这三个组件分别占据一整行即4列的空间。下方数字0~9、符号+ - * / = . 等Button组件,每行四个按钮平均分布。
二、功能
第一次做简易计算器,暂时只能完成两个数之间的加减乘除运算,不能进行负数的运算,多个数的混合运算需要考虑运算的优先级问题,有待之后完善。
界面实现后就是对计算器的逻辑实现。首先在MainActivity中对各个控件进行控件实例化,并设置监听事件,使用实现接口的方式来实现监听事件。
使用switch判断按钮,将字符添加到算式字符串中,按“=”调用一个getResult()方法计算结果并显示。
三、源代码
strings.xml
<resources>
<string name="app_name">计算器</string>
<string name="defaultTxt"></string>
<string name="resultTxt"></string>
<string name="allClear">AC</string>
<string name="zero">0</string>
<string name="one">1</string>
<string name="two">2</string>
<string name="three">3</string>
<string name="four">4</string>
<string name="five">5</string>
<string name="six">6</string>
<string name="seven">7</string>
<string name="eight">8</string>
<string name="nine">9</string>
<string name="point">.</string>
<string name="plus">+</string>
<string name="minus">-</string>
<string name="multiply">*</string>
<string name="divide">/</string>
<string name="equal">=</string>
</resources>
界面
<?xml version="1.0" encoding="utf-8"?>
<GridLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:rowCount="7"
android:columnCount="4">
<TextView
android:id="@+id/txt1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_columnSpan="4"
android:gravity="end"
android:text="@string/defaultTxt"
android:textSize="40sp"/>
<TextView
android:id="@+id/txt2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_columnSpan="4"
android:gravity="end"
android:text="@string/resultTxt"
android:textSize="50sp"/>
<Button
android:id="@+id/btnClear"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_columnSpan="4"
android:text="@string/allClear"
android:textSize="26sp"
/>
<Button
android:id="@+id/btn1"
android:text="@string/one"
android:textSize="26sp"
android:layout_columnWeight="1" />
<Button
android:id="@+id/btn2"
android:text="@string/two"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn3"
android:text="@string/three"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnPlus"
android:text="@string/plus"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn4"
android:text="@string/four"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn5"
android:text="@string/five"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn6"
android:text="@string/six"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnMinus"
android:text="@string/minus"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn7"
android:text="@string/seven"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn8"
android:text="@string/eight"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn9"
android:text="@string/nine"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnMultiply"
android:text="@string/multiply"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnPoint"
android:text="@string/point"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btn0"
android:text="@string/zero"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnEqual"
android:text="@string/equal"
android:textSize="26sp"
android:layout_columnWeight="1"/>
<Button
android:id="@+id/btnDivide"
android:text="@string/divide"
android:textSize="26sp"
android:layout_columnWeight="1"/>
</GridLayout>
MainActivity.java
package com.example.calculator;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
boolean end; //计算是否结束
TextView tv1,tv2;
Button [] buttons = new Button[17];
int [] btnID = new int[]{R.id.btn0,R.id.btn1,R.id.btn2,R.id.btn3,R.id.btn4,R.id.btn5,
R.id.btn6,R.id.btn7,R.id.btn8,R.id.btn9,R.id.btnClear,R.id.btnPlus,R.id.btnMinus,
R.id.btnMultiply,R.id.btnDivide,R.id.btnPoint,R.id.btnEqual};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv1 = (TextView)findViewById(R.id.txt1);
tv2 = (TextView)findViewById(R.id.txt2);
for (int i=0;i<btnID.length;i++) {
buttons[i] = (Button)findViewById(btnID[i]);
buttons[i].setOnClickListener(this);
}
}
public void onClick(View v){
String str = tv1.getText().toString();
switch (v.getId()){
case R.id.btn0:
case R.id.btn1:
case R.id.btn2:
case R.id.btn3:
case R.id.btn4:
case R.id.btn5:
case R.id.btn6:
case R.id.btn7:
case R.id.btn8:
case R.id.btn9:
case R.id.btnPoint:
if (end){
end = false;
str = "";
tv1.setText("");
}
tv1.setText(str+((Button)v).getText());
break;
case R.id.btnPlus:
case R.id.btnMinus:
case R.id.btnMultiply:
case R.id.btnDivide:
if (end){
end = false;
str = "";
tv1.setText("");
}
tv1.setText(str+" "+((Button)v).getText()+" ");
break;
case R.id.btnClear:
end = false;
str = "";
tv1.setText("");
tv2.setText("");
break;
case R.id.btnEqual:
getResult();
break;
}
}
void getResult(){
String exp = tv1.getText().toString();
if (exp.equals("")) return;//算式为空
if (!exp.contains(" ")) return;//无运算符
end = true;
double result = 0;
//仅完成两个数字间的运算,无混合运算
//运算符前的数字
String s1=exp.substring(0,exp.indexOf(" "));
//运算符
String op=exp.substring(exp.indexOf(" ")+1,exp.indexOf(" ")+2);
//运算符后的数字
String s2=exp.substring(exp.indexOf(" ")+3);
//有两个数字
if (!s1.equals("")&&!s2.equals("")) {
double n1 = Double.parseDouble(s1);
double n2 = Double.parseDouble(s2);
if(op.equals("+")){
result=n1+n2;
}else if(op.equals("-")){
result=n1-n2;
}else if(op.equals("*")){
result=n1*n2;
}else if(op.equals("/")) {
if (n2 == 0) tv2.setText("不能除以0");//如果被除数是0
else result = n1/n2;
}
}
//只有运算符
else if (s1.equals("")&&s2.equals("")) {
result = 0;
}
//运算符后无数字 有s1无s2
else if (!s1.equals("")&&s2.equals("")) {
double n1 = Double.parseDouble(s1);
result = n1;
}
//运算符前无数字
else if (s1.equals("")&&!s2.equals("")) {
double n2 = Double.parseDouble(s2);
if(op.equals("+")){
result=n2;
}else if(op.equals("-")) {
result=0-n2;
}else if(op.equals("*")||op.equals("/")){
result=0;
}
}
tv2.setText("= "+result);
}
}
四、总结
以下是开发过程中遇到的问题以及最后采取的解决办法:
1. 原本根据课本的布局示例,按键右方离屏幕边缘有很大的空白区域,按键没有平分屏幕宽度。
解决方法:每个Button组件加上 android:layout_columnWeight="1"
使Button的权重相同,就能平均分布。
2. 在源码MainActivity.java中,计算结果的方法getResult()内,从算式字符串中截取出两个数和一个运算符后(仍是字符串),将两个数的字符串分别通过Double.parseDouble()转换为double型数据。这一步原本在截取出字符串后和用if判断前,运行时一旦输入不满足两个数运算,计算时就会闪退。后将数据类型转换移动到if内部即解决。