第八章
1.这次把Account写成了抽象类,里面有几个虚抽象函数,在子类中实现了。
2.批量操作和统一接口,因为对象类型不一样,还要求统一实现,就用到虚函数,而函数接口设置为一样,在函数体额外加判断条件,来达到接口统一,更利于程序的封装性,安全性。
3.java里面的输出,也是调用类库,比如java.text.DecimalFormat可以控制输出小数点,java.io.IOException和java.util.*则是输入的专用库,每种类型的输入还不一样。
4.在Run里面定义输入变量cmd是char,cmd = (char)System.in.read()。 这时候可能发生异常所以有抛出声明throws IOException。

package ob5;

public class Date {
	//日期类
		private int year;		//年
		private	int month;		//月
		private	int day;		//日
		private	int totalDays;	//该日期是从公元元年1月1日开始的第几天
		
		
		public	int[] DAYS_BEFORE_MONTH =new int[]{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };
		public	void show(){
			System.out.println( getYear()+"-"+getMonth()+"-"+getDay());
		}		//输出当前日期
		public Date(int year, int month, int day){
			this.year=year;
			this.month=month;
			this.day=day;
			if (day <= 0 || day > getMaxDay()) {
				System.out.println( "Invalid date: ");
				show();
				System.out.println();
				System.exit(1);
			}
			int years = year - 1;
			totalDays = years * 365 + years / 4 - years / 100 + years / 400
				+ DAYS_BEFORE_MONTH[month - 1] + day;
			if (isLeapYear() && month > 2) totalDays++;
			
		}//用年、月、日构造日期
		public	int getYear() { return year; }
		public	int getMonth() { return month; }
		public	int getDay() { return day; }
		public	int getMaxDay(){
			if (isLeapYear() && month == 2)
				return 29;
			else
				return DAYS_BEFORE_MONTH[month]- DAYS_BEFORE_MONTH[month - 1];
		} //获得当月有多少天
		public	boolean isLeapYear()  {	//判断当年是否为闰年
				return year % 4 == 0 && year % 100 != 0 || year % 400 == 0;
			}
		
			//计算两个日期之间差多少天	
		public	int distance(Date date) {
				return totalDays - date.totalDays;
			}

}


package ob5;
import java.text.DecimalFormat;

abstract public class Account {//账户类

	private String id;	//帐号
	private double balance;	//余额
	private static double total = 0; //所有账户的总金额
	
		//供派生类调用的构造函数,id为账户
	protected Account(final Date date, final String id)
	{
		this.id=id;
		balance=0;
		date.show();
		System.out.println("\t#" + id + " created");
	}
		//记录一笔帐,date为日期,amount为金额,desc为说明
	protected void record(final Date date, double amount, final String desc)
	{
		amount = Math.floor(amount * 100 + 0.5) / 100;	//保留小数点后两位
		balance += amount;
		total += amount;
		date.show();
		System.out.println( "\t#" + id + "\t" + amount + "\t" + balance + "\t" + desc );
	}
		//报告错误信息
	protected void error(final String msg)
	{
		System.out.println("Error(#" + id + "):" + msg);
	}
	
	static DecimalFormat df = new DecimalFormat("0.0");//限制位数
	public final String getId() { return id; }
	public double getBalance() { return balance; }
	public static double getTotal() { return total; }
		//显示账户信息
	
	abstract void deposit(final Date date, double amount,final String desc);
	//取出现金,date为日期,amount为金额,desc为款项说明
	abstract void withdraw(final Date date, double amount,final String desc);
	//结算(计算利息、年费等),每月结算一次,date为结算日期
	abstract void settle(final Date date);
	//显示账户信息
	public void show()
	{
		System.out.print(id + "\tBalance: " + df.format(balance));
	}

}

public class CreditAccount extends Account{//信用账户类

	private Accumulator acc;	//辅助计算利息的累加器
	private double credit;		//信用额度
	private double rate;		//欠款的日利率
	private double fee;			//信用卡年费

	private double getDebt() {	//获得欠款额
			double balance = getBalance();
			return (balance < 0 ? balance : 0);
	}
	
		//构造函数
	public CreditAccount(final Date date,final String id, double credit, double rate, double fee)
	{
		super(date, id);
		this.credit=credit;
		this.rate=rate;
		this.fee=fee;
		acc = new Accumulator(date, 0);
		
	}
	public double getCredit() { return credit; }
	public double getRate() { return rate; }
	public double getFee() { return fee; }
	public double getAvailableCredit() {	//获得可用信用
			if (getBalance() < 0) 
				return credit + getBalance();
			else
				return credit;
		}
		//存入现金
	public void deposit(final Date date, double amount,final String desc)
	{
		record(date, amount, desc);
		acc.change(date, getDebt());
	}
		//取出现金
	public void withdraw(final Date date, double amount,final String desc)
	{
		if (amount - getBalance() > credit) {
			error("not enough credit");
		} else {
			record(date, -amount, desc);
			acc.change(date, getDebt());
		}
	}
		//结算利息和年费,每月1日调用一次该函数
	public void settle(final Date date)
	{
		double interest = acc.getSum(date) * rate;
		if (interest != 0)
			record(date, interest, "interest");
		if (date.getMonth() == 1)
			record(date, -fee, "annual fee");
		acc.reset(date, getDebt());
	}
	public void show()
	{
		super.show();
		System.out.print( "\t\tAvailable credit:" + getAvailableCredit());
	}
}

package ob5;

public class SavingsAccount extends Account {
	//储蓄账户类
	private Accumulator acc;	//辅助计算利息的累加器
	private double rate;		//存款的年利率
	
		//构造函数
	public SavingsAccount(final Date date,final String id, double rate)
	{
		super(date,id);
		this.rate = rate;
		acc=new Accumulator(date,0);
	}
	public double getRate() { return rate; }
		//存入现金
	public void deposit(final Date date, double amount,final String desc)
	{
		record(date, amount, desc);
		acc.change(date, getBalance());
	}
		//取出现金
	public void withdraw(final Date date, double amount,final String desc)
	{
		if (amount > getBalance()) {
			error("not enough money");
		} else {
			record(date, -amount, desc);
			acc.change(date, getBalance());
		}
	}
		//结算利息,每年1月1日调用一次该函数
	public void settle(final Date date)
	{
		if(date.getMonth() == 1) {
			double interest = acc.getSum(date) * rate / date.distance(new Date(date.getYear() - 1, 1, 1));	//计算年息
			//计算年息
		if (interest != 0)
			record(date, interest, "interest");
		acc.reset(date, getBalance());
	}
	}

}
package ob5;

public class Accumulator {//将某个数值按日累加

	private Date lastDate;	//上次变更数值的时期
	private double value;	//数值的当前值
	private double sum;		//数值按日累加之和
	
		//构造函数,date为开始累加的日期,value为初始值
	public Accumulator(final Date date, double value)
	{
		lastDate=date;
		this.value=value;
		sum=0;
	}
		//获得到日期date的累加结果
	public double getSum(final Date date){
			return sum + value * date.distance(lastDate);
		}

		//在date将数值变更为value
	public void change(final Date date, double value) {
			sum = getSum(date);
			lastDate = date;
			this.value = value;
		}

		//初始化,将日期变为date,数值变为value,累加器清零
	public void reset(final  Date date, double value) {
			lastDate = date;
			this.value = value;
			sum = 0;
		}

}

package ob5;
import java.text.DecimalFormat;
import java.util.*; 
import java.io.IOException;
public class Run {
	public static void main(String[] args) throws IOException  {
		Date date = new Date(2008, 11, 1);	//起始日期
		//建立几个账户
		SavingsAccount sa1 = new SavingsAccount(date, "S3755217", 0.015);
		SavingsAccount sa2 = new SavingsAccount(date, "02342342", 0.015);
		CreditAccount ca = new CreditAccount(date, "C5392394", 10000, 0.0005, 50);
		Account accounts[] = { sa1, sa2, ca };
		final int n = accounts.length;	//账户总数

		System.out.println("(d)deposit (w)withdraw (s)show (c)change day (n)next month (e)exit");
		char cmd;
		Scanner in=new Scanner(System.in); //使用Scanner类定义对象
		int index, day;
		double amount;
		String desc;
		DecimalFormat df = new DecimalFormat("0.0");
		do {
			//显示日期和总金额
			date.show();
			System.out.print("\tTotal: " + df.format(Account.getTotal()) + "\tcommand> ");

			cmd = (char)System.in.read();
			switch (cmd) {
			case 'd':	//存入现金
				index = in.nextInt();
				amount = in.nextDouble();
				desc = in.nextLine();
				accounts[index].deposit(date, amount, desc);
				break;
			case 'w':	//取出现金
				index = in.nextInt();
				amount = in.nextDouble();
				desc = in.nextLine();
				accounts[index].withdraw(date, amount, desc);
				break;
			case 's':	//查询各账户信息
				for (int i = 0; i < n; i++) {
					System.out.print("[" + i + "] ");
					accounts[i].show();
					System.out.println();
					if(i<(n-1)) {System.in.read();}
				}
				break;
			case 'c':	//改变日期
				day = in.nextInt();
				if (day < date.getDay())
					System.out.print("You cannot specify a previous day");
				else if (day > date.getMaxDay())
					System.out.print("Invalid day");
				else
					date = new Date(date.getYear(), date.getMonth(), day);
				break;
			case 'n':	//进入下个月
				if (date.getMonth() == 12)
					date = new Date(date.getYear() + 1, 1, 1);
				else
					date = new Date(date.getYear(), date.getMonth() + 1, 1);
				for (int i = 0; i < n; i++) {
					accounts[i].settle(date);
					if(i<(n-1)) {System.in.read();}
				}
				break;
			}
		} while (cmd != 'e');
		in.close();
	}
	

}