第三次博客作业
前言:
在上一个月的课程学习中,我吸收掌握了很多具有拓展性、延申性的知识,其中包含类与对象的构造、组成,以及在封装、继承、多态原则优化后的代码结构基础上,进行抽象类、抽象方法的联系与应用。在第六次至第八次PTA大作业中,我们接触并实现了以类与对象为基础的电信计费系列的编程,三次的题目分别从座机计费、手机座机计费、短信计费出发,对关于构造类知识的掌握、类图的分析与理解做出了考察。三次大作业中,电信计费系列的题目难度偏大,但主要原因在于代码量大、涉及的类与继承关系复杂。不过只要我们能够读懂类图,分析清楚题目的要求,加之对不同信息格式的严格判断,写出符合要求的代码是完全可行的。
设计与分析:
7-1 电信计费系列1-座机计费
实现一个简单的电信计费程序:
假设南昌市电信分公司针对市内座机用户采用的计费方式:
月租20元,接电话免费,市内拨打电话0.1元/分钟,省内长途0.3元/分钟,国内长途拨打0.6元/分钟。不足一分钟按一分钟计。
南昌市的区号:0791,江西省内各地市区号包括:0790~0799以及0701。
输入格式:
输入信息包括两种类型
1、逐行输入南昌市用户开户的信息,每行一个用户,
格式:u-号码 计费类型 (计费类型包括:0-座机 1-手机实时计费 2-手机A套餐)
例如:u-079186300001 0
座机号码除区号外由是7-8位数字组成。
本题只考虑计费类型0-座机计费,电信系列2、3题会逐步增加计费类型。
2、逐行输入本月某些用户的通讯信息,通讯信息格式:
座机呼叫座机:t-主叫号码 接听号码 起始时间 结束时间
t-079186330022 058686330022 2022.1.3 10:00:25 2022.1.3 10:05:11
以上四项内容之间以一个英文空格分隔,
时间必须符合"yyyy.MM.dd HH:mm:ss"格式。提示:使用SimpleDateFormat类。
以上两类信息,先输入所有开户信息,再输入所有通讯信息,最后一行以“end”结束。
注意:
本题非法输入只做格式非法的判断,不做内容是否合理的判断(时间除外,否则无法计算),比如:
1、输入的所有通讯信息均认为是同一个月的通讯信息,不做日期是否在同一个月还是多个月的判定,直接将通讯费用累加,因此月租只计算一次。
2、记录中如果同一电话号码的多条通话记录时间出现重合,这种情况也不做判断,直接 计算每条记录的费用并累加。
3、用户区号不为南昌市的区号也作为正常用户处理。
输出格式:
根据输入的详细通讯信息,计算所有已开户的用户的当月费用(精确到小数点后2位,
单位元)。假设每个用户初始余额是100元。
每条通讯信息单独计费后累加,不是将所有时间累计后统一计费。
格式:号码+英文空格符+总的话费+英文空格符+余额
每个用户一行,用户之间按号码字符从小到大排序。
错误处理:
输入数据中出现的不符合格式要求的行一律忽略。
建议类图:
参见图1、2、3,可根据理解自行调整:
图1中User是用户类,包括属性:
userRecords (用户记录)、balance(余额)、chargeMode(计费方式)、number(号码)。
ChargeMode是计费方式的抽象类:
chargeRules是计费方式所包含的各种计费规则的集合,ChargeRule类的定义见图3。
getMonthlyRent()方法用于返回月租(monthlyRent)。
UserRecords是用户记录类,保存用户各种通话、短信的记录,
各种计费规则将使用其中的部分或者全部记录。
其属性从上到下依次是:
市内拨打电话、省内(不含市内)拨打电话、省外拨打电话、
市内接听电话、省内(不含市内)接听电话、省外接听电话的记录
以及发送短信、接收短信的记录。
图2中CommunicationRecord是抽象的通讯记录类:
包含callingNumber拨打号码、answerNumber接听号码两个属性。
CallRecord(通话记录)、MessageRecord(短信记录)是它的子类。
CallRecord(通话记录类)包含属性:
通话的起始、结束时间以及
拨号地点的区号(callingAddressAreaCode)、接听地点的区号(answerAddressAreaCode)。
区号用于记录在哪个地点拨打和接听的电话。座机无法移动,就是本机区号,如果是手机号,则会有差异。
图3是计费规则的相关类,这些类的核心方法是:
calCost(ArrayList<CallRecord> callRecords)。
该方法针根据输入参数callRecords中的所有记录计算某用户的某一项费用;如市话费。
输入参数callRecords的约束条件:必须是某一个用户的符合计费规则要求的所有记录。
LandPhoneInCityRule、LandPhoneInProvinceRule、LandPhoneInLandRule三个类分别是
座机拨打市内、省内、省外电话的计费规则类,用于实现这三种情况的费用计算。
(提示:可以从UserRecords类中获取各种类型的callRecords)。
后续扩展说明:
后续题目集将增加手机用户,手机用户的计费方式中除了与座机计费类似的主叫通话费之外,还包含市外接听电话的漫游费以及发短信的费用。在本题的设计时可统一考虑。
通话记录中,手机需要额外记录拨打/接听的地点的区号,比如:
座机打手机:t-主叫号码 接听号码 接听地点区号 起始时间 结束时间
t-079186330022 13305862264 020 2022.1.3 10:00:25 2022.1.3 10:05:11
手机互打:t-主叫号码 拨号地点 接听号码 接听地点区号 起始时间 结束时间
t-18907910010 0791 13305862264 0371 2022.1.3 10:00:25 2022.1.3 10:05:11
短信的格式:m-主叫号码,接收号码,短信内容
m-18907910010 13305862264 welcome to jiangxi
m-13305862264 18907910010 thank you
输入样例:
在这里给出一组输入。例如:
u-079186300001 0
t-079186300001 058686330022 2022.1.3 10:00:25 2022.1.3 10:05:25
end
输出样例:
在这里给出相应的输出。例如:
079186300001 3.0 77.0
其余参考样例详见附件,未尽事宜以附件样例为准:
全码展示:

import java.text.DecimalFormat; import java.util.*; import java.util.ArrayList; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); String str = sc.nextLine(); ArrayList<User> Array = new ArrayList<>(); boolean flag = false; while (!str.equals("end")) { if (str.matches("u\\-0791\\d{7,8}\\s0")) { String str1[] = str.split("-"); String str2[] = str1[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).number.equals(str2[0])) { flag = true; break; } } if (flag == false) { Array.add(new User(str2[0])); } } if (str.matches("t\\-(\\d{10,12})\\s(\\d{10,12})\\s(\\d{4}.([1-9]{1}|1[0-2]).([1-9]{1}|1[0-9]|2[0-9]|3[0-1])\\s(0[0-9]|1[0-9]|2[0-3]):([0-5]{1}[0-9]):([0-5]{1}[0-9]))\\s([0-9]{4}.([1-9]{1}|1[0-2]).([1-9]{1}|1[0-9]|2[0-9]|3[0-1])\\s(0[0-9]|1[0-9]|2[0-3]):([0-5]{1}[0-9]):([0-5]{1}[0-9]))")) { String str3[] = str.split("-"); String str4[] = str3[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).number.equals(str4[0])) { if (str4[1].substring(0, 4).matches("0791")) { Array.get(i).getUserRecords().addgetCallingInCityRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); Array.get(i).setUserRecords(Array.get(i).getUserRecords()); } else if (str4[1].substring(0, 4).matches("079\\d|0701")) { Array.get(i).getUserRecords().addgetCallingInProvinceRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); Array.get(i).setUserRecords(Array.get(i).getUserRecords()); } else { Array.get(i).getUserRecords().addgetCallingInLandRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); Array.get(i).setUserRecords(Array.get(i).getUserRecords()); } } } } str = sc.nextLine(); } Collections.sort(Array,new Comparator<User>(){ public int compare(User u1,User u2){ int num = Math.toIntExact(Long.parseLong(u1.getNumber()) - Long.parseLong(u2.getNumber())); return num; } }); for (int i = 0; i < Array.size(); i++) { System.out.println(Array.get(i).number + " " + new DecimalFormat("0.0#").format(Array.get(i).calCost()) + " " + new DecimalFormat("0.0#").format(Array.get(i).getBalance() - Array.get(i).calCost() - 20)); } } } abstract class CallChargeRule extends ChargeRule{ public abstract double calCost(ArrayList<CallRecord> callRecords); } class CallRecord extends CommunicationRecord{ private Date startTime; private Date endTime; private String callingAddressAreaCode; private String answerAddressAreaCode; public CallRecord(String startTime, String endTime) { SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy.MM.dd HH:mm:ss"); try { this.startTime = simpleDateFormat.parse(startTime); } catch (ParseException e) { } try { this.endTime = simpleDateFormat.parse(endTime); } catch (ParseException e) { } } public Date getStartTime() { return startTime; } public void setStartTime(Date startTime) { this.startTime = startTime; } public String getCallingAddressAreaCode() { return callingAddressAreaCode; } public void setCallingAddressAreaCode(String callingAddressAreaCode) { this.callingAddressAreaCode = callingAddressAreaCode; } public Date getEndTime() { return endTime; } public void setEndTime(Date endTime) { this.endTime = endTime; } public String getAnswerAddressAreaCode() { return answerAddressAreaCode; } public void setAnswerAddressAreaCode(String answerAddressAreaCode) { this.answerAddressAreaCode = answerAddressAreaCode; } } abstract class ChargeMode { ArrayList<ChargeRule> chargeRules = new ArrayList<>(); public ArrayList<ChargeRule> getChargeRules() { return this.chargeRules; } public void setChargeRules(ArrayList<ChargeRule> chargeRules) { this.chargeRules = chargeRules; } public abstract double calCost(UserRecords userRecords); public abstract double getMonthlyRent(); } abstract class ChargeRule { } abstract class CommunicationRecord { protected String callingNumber; protected String answerNumber; public String getCallingNumber() { return callingNumber; } public void setCallingNumber(String callingNumber) { this.callingNumber = callingNumber; } public String getAnswerNumber() { return answerNumber; } public void setAnswerNumber(String answerNumber) { this.answerNumber = answerNumber; } } class LandlinePhoneCharging extends ChargeMode{ private double monthlyRent = 20; @Override public double calCost(UserRecords userRecords) { return 0; } @Override public double getMonthlyRent() { return monthlyRent; } } class LandPhoneInCityRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.1 * minutes; } return cost; } } class LandPhoneInLandRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes; for(CallRecord e:callRecords){ long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.6*minutes; } return cost; } } class LandPhoneInProvinceRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.3 * minutes; } return cost; } } class MessageRecord extends CommunicationRecord{ private String message; public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } } class User { private UserRecords userRecords = new UserRecords(); double balance = 100; String number; ChargeMode chargeMode; public User(String number) { this.number = number; } public double CalBalance(){ //return 0; return balance - calCost(); } public double calCost(){ LandPhoneInCityRule landPhoneInCityRule = new LandPhoneInCityRule(); LandPhoneInLandRule landPhoneInLandRule = new LandPhoneInLandRule(); LandPhoneInProvinceRule landPhoneInProvinceRule = new LandPhoneInProvinceRule(); return landPhoneInLandRule.calCost(userRecords.getCallingInLandRecords())+landPhoneInCityRule.calCost(userRecords.getCallingInCityRecords())+landPhoneInProvinceRule.calCost(userRecords.getCallingInProvinceRecords()); } public UserRecords getUserRecords() { return userRecords; } public void setUserRecords(UserRecords userRecords) { this.userRecords = userRecords; } public double getBalance() { return balance; } public String getNumber() { return number; } public void setNumber(String number) { this.number = number; } public ChargeMode getChargeMode() { return chargeMode; } public void setChargeMode(ChargeMode chargeMode) { this.chargeMode = chargeMode; } } class UserRecords { private ArrayList<CallRecord> callingInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> callingInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> callingInLandRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> answerInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> answerInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> answerInLandRecords = new ArrayList<CallRecord>(); private ArrayList<MessageRecord> sendMessageRecords = new ArrayList<MessageRecord>(); private ArrayList<MessageRecord> receiveMessageRecords = new ArrayList<MessageRecord>(); public ArrayList<CallRecord> getCallingInCityRecords() { return callingInCityRecords; } public ArrayList<CallRecord> getCallingInProvinceRecords() { return callingInProvinceRecords; } public ArrayList<CallRecord> getCallingInLandRecords() { return callingInLandRecords; } public ArrayList<CallRecord> getAnswerInCityRecords() { return answerInCityRecords; } public ArrayList<CallRecord> getAnswerInProvinceRecords() { return answerInProvinceRecords; } public ArrayList<CallRecord> getAnswerInLandRecords() { return answerInLandRecords; } public ArrayList<MessageRecord> getSendMessageRecords() { return sendMessageRecords; } public ArrayList<MessageRecord> getReceiveMessageRecords() { return receiveMessageRecords; } public void addgetCallingInCityRecords(CallRecord callRecord) { getCallingInCityRecords().add(callRecord); } public void addgetCallingInProvinceRecords(CallRecord callRecord) { getCallingInProvinceRecords().add(callRecord); } public void addgetCallingInLandRecords(CallRecord callRecord) { getCallingInLandRecords().add(callRecord); } public void addAnswerInCityRecords(CallRecord answerRecord) { answerInCityRecords.add(answerRecord); } public void addAnswerInProvinceRecords(CallRecord answerRecord) { answerInProvinceRecords.add(answerRecord); } public void addAnswerInLandRecords(CallRecord answerRecord) { answerInLandRecords.add(answerRecord); } public void addSendMessageRecords(MessageRecord sendMessageRecord) { sendMessageRecords.add(sendMessageRecord); } public void addReceiveMessageRecords(MessageRecord receiveMessageRecord) { receiveMessageRecords.add(receiveMessageRecord); } }
SourceMonitor分析图:
IDEA生成程序类图:
总结:
本题作为电信计费系列的第一道题,需要我们将题干中给出的类图进行详细的分析,从中提取关键信息组成完成的可运行代码。对于本题我的思路和解题步骤可总结如下:首先构造出电信计费系列的基本框架,即弄清楚都有些什么具体的类,类与类之间的继承关系是怎样的。比如通过第一张类图,我们可以认为它是用来讲用户的,其讲清楚了三个类:用户类User,用户记录类UserRecords,以及计费方式ChargeMode。其中ChargeMode是抽象类,我们可以同时确定继承ChargeMode的是哪些类,以及三种类之间的联系。按照同样的方式分析类图2与类图3,我们可以认为它们分别是通讯记录类和计费规则相关类,之再后向类中填入方法,同时注意抽象类的抽象方法,以及继承时对方法的引用做出相应的改变。
7-2 电信计费系列2-手机+座机计费
实现南昌市电信分公司的计费程序,假设该公司针对手机和座机用户分别采取了两种计费方案,分别如下:
1、针对市内座机用户采用的计费方式(与电信计费系列1内容相同):
月租20元,接电话免费,市内拨打电话0.1元/分钟,省内长途0.3元/分钟,国内长途拨打0.6元/分钟。不足一分钟按一分钟计。
假设本市的区号:0791,江西省内各地市区号包括:0790~0799以及0701。
2、针对手机用户采用实时计费方式:
月租15元,市内省内接电话均免费,市内拨打市内电话0.1元/分钟,市内拨打省内电话0.2元/分钟,市内拨打省外电话0.3元/分钟,省内漫游打电话0.3元/分钟,省外漫游接听0.3元/分钟,省外漫游拨打0.6元/分钟;
注:被叫电话属于市内、省内还是国内由被叫电话的接听地点区号决定,比如以下案例中,南昌市手机用户13307912264在区号为020的广州接听了电话,主叫号码应被计算为拨打了一个省外长途,同时,手机用户13307912264也要被计算省外接听漫游费:
u-13307912264 1
t-079186330022 13307912264 020 2022.1.3 10:00:25 2022.1.3 10:05:11
输入:
输入信息包括两种类型
1、逐行输入南昌市用户开户的信息,每行一个用户,含手机和座机用户
格式:u-号码 计费类型 (计费类型包括:0-座机 1-手机实时计费 2-手机A套餐)
例如:u-079186300001 0
座机号码由区号和电话号码拼接而成,电话号码包含7-8位数字,区号最高位是0。
手机号码由11位数字构成,最高位是1。
本题在电信计费系列1基础上增加类型1-手机实时计费。
手机设置0或者座机设置成1,此种错误可不做判断。
2、逐行输入本月某些用户的通讯信息,通讯信息格式:
座机呼叫座机:t-主叫号码 接听号码 起始时间 结束时间
t-079186330022 058686330022 2022.1.3 10:00:25 2022.1.3 10:05:11
以上四项内容之间以一个英文空格分隔,
时间必须符合"yyyy.MM.dd HH:mm:ss"格式。提示:使用SimpleDateFormat类。
输入格式增加手机接打电话以及收发短信的格式,手机接打电话的信息除了号码之外需要额外记录拨打/接听的地点的区号,比如:
座机打手机:
t-主叫号码 接听号码 接听地点区号 起始时间 结束时间
t-079186330022 13305862264 020 2022.1.3 10:00:25 2022.1.3 10:05:11
手机互打:
t-主叫号码 拨号地点 接听号码 接听地点区号 起始时间 结束时间
t-18907910010 0791 13305862264 0371 2022.1.3 10:00:25 2022.1.3 10:05:11
注意:以上两类信息,先输入所有开户信息,再输入所有通讯信息,最后一行以“end”结束。
输出:
根据输入的详细通讯信息,计算所有已开户的用户的当月费用(精确到小数点后2位,单位元)。假设每个用户初始余额是100元。
每条通讯、短信信息均单独计费后累加,不是将所有信息累计后统一计费。
格式:号码+英文空格符+总的话费+英文空格符+余额
每个用户一行,用户之间按号码字符从小到大排序。
错误处理:
输入数据中出现的不符合格式要求的行一律忽略。
本题只做格式的错误判断,无需做内容上不合理的判断,比如同一个电话两条通讯记录的时间有重合、开户号码非南昌市的号码等,此类情况都当成正确的输入计算。但时间的输入必须符合要求,比如不能输入2022.13.61 28:72:65。
建议类图:
参见图1、2、3:
图1中User是用户类,包括属性:
userRecords (用户记录)、balance(余额)、chargeMode(计费方式)、number(号码)。
ChargeMode是计费方式的抽象类:
chargeRules是计费方式所包含的各种计费规则的集合,ChargeRule类的定义见图3。
getMonthlyRent()方法用于返回月租(monthlyRent)。
UserRecords是用户记录类,保存用户各种通话、短信的记录,
各种计费规则将使用其中的部分或者全部记录。
其属性从上到下依次是:
市内拨打电话、省内(不含市内)拨打电话、省外拨打电话、
市内接听电话、省内(不含市内)接听电话、省外接听电话的记录
以及发送短信、接收短信的记录。
图2中CommunicationRecord是抽象的通讯记录类:
包含callingNumber拨打号码、answerNumber接听号码两个属性。
CallRecord(通话记录)、MessageRecord(短信记录)是它的子类。CallRecord(通话记录类)包含属性:
通话的起始、结束时间以及
拨号地点的区号(callingAddressAreaCode)、接听地点的区号(answerAddressAreaCode)。
区号用于记录在哪个地点拨打和接听的电话。座机无法移动,就是本机区号,如果是手机号,则会有差异。
图3是计费规则的相关类,这些类的核心方法是:
calCost(ArrayList<CallRecord> callRecords)。
该方法针根据输入参数callRecords中的所有记录计算某用户的某一项费用;如市话费。
输入参数callRecords的约束条件:必须是某一个用户的符合计费规则要求的所有记录。
SendMessageRule是发送短信的计费规则类,用于计算发送短信的费用。
LandPhoneInCityRule、LandPhoneInProvinceRule、LandPhoneInLandRule三个类分别是座机拨打市内、省内、省外电话的计费规则类,用于实现这三种情况的费用计算。
(提示:可以从UserRecords类中获取各种类型的callRecords)。
注意:以上图中所定义的类不是限定要求,根据实际需要自行补充或修改。
输入样例:
在这里给出一组输入。例如:
u-13811111111 1
t-13811111111 0791 13811111110 020 2022.1.3 08:00:00 2022.1.3 08:09:20
end
输出样例:
在这里给出相应的输出。例如:
13811111111 3.0 82.0
更多内容详见附件:
全码如下:

import java.text.DecimalFormat; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.Scanner; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); String str = sc.nextLine(); ArrayList<User> Array = new ArrayList<>(); boolean flag = false; while (!str.equals("end")) { flag = false; if (str.matches("u\\-0791\\d{7,8}\\s0") || str.matches("u\\-1\\d{10}\\s1")) { String str1[] = str.split("-"); String str2[] = str1[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str2[0])) { flag = true; break; } } if (flag == false) { User user = new User(str2[0]); if (str.matches("u\\-0791\\d{7,8}\\s0")) { LandLinePhoneCharging landLinePhoneCharging = new LandLinePhoneCharging(); user.setChargeMode(landLinePhoneCharging); } else if (str.matches("u\\-1\\d{10}\\s1")) { MobilePhoneCharging mobilePhoneCharging = new MobilePhoneCharging(); user.setChargeMode(mobilePhoneCharging); } Array.add(user); } } //座机打座机 if (str.matches("t-0\\d{9,11}\\s0\\d{9,11}\\s([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}\\.(1[0-2]|[1-9])\\.([1-9]|(1|2)[0-9]|(30|31)) ([0|1][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9] ?){2}")) { String str3[] = str.split("-"); String str4[] = str3[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str4[0])) { if (str4[1].substring(0, 4).matches("0791")) { Array.get(i).getUserRecords().addCallingInCityRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); } else if (str4[1].substring(0, 4).matches("079\\d|0701")) { Array.get(i).getUserRecords().addCallingInProvinceRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); } else { Array.get(i).getUserRecords().addCallingInLandRecords(new CallRecord(str4[2] + " " + str4[3], str4[4] + " " + str4[5])); } } }//座机打手机 } else if (str.matches("t-\\d{10,12}\\s1\\d{10}\\s\\d{3,4}\\s([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}\\.(1[0-2]|[1-9])\\.([1-9]|(1|2)[0-9]|(30|31)) ([0|1][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9] ?){2}")) { String str3[] = str.split("-"); String str4[] = str3[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str4[0])) { if (str4[2].matches("0791")) { Array.get(i).getUserRecords().addCallingInCityRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } else if (str4[2].matches("079\\d|0701")) { Array.get(i).getUserRecords().addCallingInProvinceRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } else { Array.get(i).getUserRecords().addCallingInLandRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } } if (Array.get(i).getNumber().equals(str4[1])) { if (!str4[2].matches("0791|079\\d|0701")) { Array.get(i).getUserRecords().addanswerInLandRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } } }//手机打手机 } else if (str.matches("t-\\d{11}\\s\\d{3,4}\\s\\d{11}\\s\\d{3,4}\\s([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}\\.(1[0-2]|[1-9])\\.([1-9]|(1|2)[0-9]|(30|31)) ([0|1][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9] ?){2}")) { String str3[] = str.split("-"); String str4[] = str3[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str4[0])) { if (str4[1].matches("0791")) { if (str4[3].matches("0791")) { Array.get(i).getUserRecords().addCityCallingInCityRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } else if (str4[3].matches("079\\d|0701")) { Array.get(i).getUserRecords().addCityCallingInProvinceRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } else { Array.get(i).getUserRecords().addCityCallingInLandRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } } else if (str4[1].matches("079\\d|0701")) { Array.get(i).getUserRecords().addProvinceCallingRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } else { Array.get(i).getUserRecords().addLandCallingRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } } if (Array.get(i).getNumber().equals(str4[2])) { if (!str4[3].matches("0791|079\\d|0701")) { Array.get(i).getUserRecords().addanswerInLandRecords(new CallRecord(str4[4] + " " + str4[5], str4[6] + " " + str4[7])); } } }//手机打座机 } else if (str.matches("t-\\d{11}\\s\\d{3,4}\\s\\d{10,12}\\s([0-9]{3}[1-9]|[0-9]{2}[1-9][0-9]{1}|[0-9]{1}[1-9][0-9]{2}|[1-9][0-9]{3}\\.(1[0-2]|[1-9])\\.([1-9]|(1|2)[0-9]|(30|31)) ([0|1][0-9]|2[0-3]):[0-5][0-9]:[0-5][0-9] ?){2}")) { String str3[] = str.split("-"); String str4[] = str3[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str4[0])) { if (str4[1].matches("0791")) { if (str4[2].substring(0, 4).matches("0791")) { Array.get(i).getUserRecords().addCityCallingInCityRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } else if (str4[2].substring(0, 4).matches("079\\d|0701")) { Array.get(i).getUserRecords().addCityCallingInProvinceRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } else { Array.get(i).getUserRecords().addCityCallingInLandRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } } else if (str4[1].matches("079\\d|0701")) { Array.get(i).getUserRecords().addProvinceCallingRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } else { Array.get(i).getUserRecords().addLandCallingRecords(new CallRecord(str4[3] + " " + str4[4], str4[5] + " " + str4[6])); } } } } str = sc.nextLine(); } //按照号码顺序排列 Collections.sort(Array, new Comparator<User>() { @Override public int compare(User o1, User o2) { return o1.getNumber().compareTo(o2.getNumber()); } }); for (int i = 0; i < Array.size(); i++) { System.out.println(Array.get(i).getNumber() + " " + new DecimalFormat("0.0#").format(Array.get(i).getChargeMode().calCost(Array.get(i).getUserRecords())) + " " + new DecimalFormat("0.0#").format(Array.get(i).getBalance() - Array.get(i).getChargeMode().calCost(Array.get(i).getUserRecords()) - Array.get(i).getChargeMode().getMonthlyRent())); } } } abstract class CallChargeRule extends ChargeRule { public abstract double calCost(ArrayList<CallRecord> callRecords); } class CallRecord extends CommunicationRecord { private Date startTime; private Date endTime; private String callingAddressAreaCode; private String answerAddressAreaCode; public CallRecord() { } public CallRecord(String startTime, String endTime) { SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy.MM.dd HH:mm:ss"); try { this.startTime = simpleDateFormat.parse(startTime); } catch (ParseException e) { } try { this.endTime = simpleDateFormat.parse(endTime); } catch (ParseException e) { } } public Date getStartTime() { return startTime; } public void setStartTime(Date startTime) { this.startTime = startTime; } public String getCallingAddressAreaCode() { return callingAddressAreaCode; } public void setCallingAddressAreaCode(String callingAddressAreaCode) { this.callingAddressAreaCode = callingAddressAreaCode; } public Date getEndTime() { return endTime; } public void setEndTime(Date endTime) { this.endTime = endTime; } public String getAnswerAddressAreaCode() { return answerAddressAreaCode; } public void setAnswerAddressAreaCode(String answerAddressAreaCode) { this.answerAddressAreaCode = answerAddressAreaCode; } } abstract class ChargeMode { ArrayList<ChargeRule> chargeRules = new ArrayList<>(); public ArrayList<ChargeRule> getChargeRules() { return this.chargeRules; } public void setChargeRules(ArrayList<ChargeRule> chargeRules) { this.chargeRules = chargeRules; } public abstract double calCost(UserRecords userRecords); public abstract double getMonthlyRent(); } abstract class ChargeRule { } abstract class CommunicationRecord { protected String callingNumber; protected String answerNumber; public CommunicationRecord(){ } public CommunicationRecord(String callingNumber, String answerNumber) { this.callingNumber = callingNumber; this.answerNumber = answerNumber; } public String getCallingNumber() { return callingNumber; } public void setCallingNumber(String callingNumber) { this.callingNumber = callingNumber; } public String getAnswerNumber() { return answerNumber; } public void setAnswerNumber(String answerNumber) { this.answerNumber = answerNumber; } } class LandLinePhoneCharging extends ChargeMode { private double monthlyRent = 20; public LandLinePhoneCharging() { } @Override public double calCost(UserRecords userRecords) { double cost1 = new LandPhoneInCityRule().calCost(userRecords.getCallingInCityRecords()); double cost2 = new LandPhoneInProvinceRule().calCost(userRecords.getCallingInProvinceRecords()); double cost3 = new LandPhoneInLandRule().calCost(userRecords.getCallingInLandRecords()); return cost1 + cost2 + cost3; } @Override public double getMonthlyRent() { return monthlyRent; } } class LandPhoneInCityRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.1 * minutes; } return cost; } } class LandPhoneInLandRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes; for(CallRecord e:callRecords){ long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); // System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.6*minutes; } return cost; } } class LandPhoneInProvinceRule extends CallChargeRule { @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.3 * minutes; } return cost; } } class MobilePhoneAnswerInLand extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.3 * minutes; } return cost; } } class MobilePhoneCallingInLand extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.6 * minutes; } return cost; } } class MobilePhoneCallingInProvince extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.3 * minutes; } return cost; } } class MobilePhoneCharging extends ChargeMode { private double MonthlyRent = 15; @Override public double calCost(UserRecords userRecords) { double cost1 = new MobilePhoneCityToCity().calCost(userRecords.getCityCallingInCityRecords()); double cost2 = new MobilePhoneCityToProvince().calCost(userRecords.getCityCallingInProvinceRecords()); double cost3 = new MobilePhoneCityToLand().calCost(userRecords.getCityCallingInLandRecords()); double cost4 = new MobilePhoneCallingInProvince().calCost(userRecords.getProvinceCallingRecords()); double cost5 = new MobilePhoneCallingInLand().calCost(userRecords.getLandCallingRecords()); double cost6 = new MobilePhoneAnswerInLand().calCost(userRecords.getAnswerInLandRecords()); return cost1 + cost2 + cost3 + cost4 + cost5 + cost6; } @Override public double getMonthlyRent() { return MonthlyRent; } } class MobilePhoneCityToCity extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.1 * minutes; } return cost; } } class MobilePhoneCityToLand extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.3 * minutes; } return cost; } } class MobilePhoneCityToProvince extends CallChargeRule{ @Override public double calCost(ArrayList<CallRecord> callRecords) { double cost = 0; double minutes = 0; for (CallRecord e : callRecords) { long diff = e.getEndTime().getTime() - e.getStartTime().getTime(); //System.out.println(diff); minutes = (int) Math.ceil((double) diff / (double) 60000); cost += 0.2 * minutes; } return cost; } } class User { private UserRecords userRecords = new UserRecords(); private double balance = 100; private String number; private ChargeMode chargeMode; public User(String number) { this.number = number; } public double CalBalance() { //return 0; return balance - calCost() - chargeMode.getMonthlyRent(); } public double calCost() { LandPhoneInCityRule landPhoneInCityRule = new LandPhoneInCityRule(); LandPhoneInLandRule landPhoneInLandRule = new LandPhoneInLandRule(); LandPhoneInProvinceRule landPhoneInProvinceRule = new LandPhoneInProvinceRule(); return landPhoneInLandRule.calCost(userRecords.getCallingInLandRecords()) + landPhoneInCityRule.calCost(userRecords.getCallingInCityRecords()) + landPhoneInProvinceRule.calCost(userRecords.getCallingInProvinceRecords()); } public UserRecords getUserRecords() { return userRecords; } public void setUserRecords(UserRecords userRecords) { this.userRecords = userRecords; } public double getBalance() { return balance; } public String getNumber() { return number; } public void setNumber(String number) { this.number = number; } public ChargeMode getChargeMode() { return chargeMode; } public void setChargeMode(ChargeMode chargeMode) { this.chargeMode = chargeMode; } } class UserRecords { private ArrayList<CallRecord> CallingInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CallingInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CallingInLandRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInLandRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> ProvinceCallingRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> LandCallingRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> answerInLandRecords = new ArrayList<CallRecord>(); public void addCityCallingInCityRecords(CallRecord callRecord){ CityCallingInCityRecords.add(callRecord); } public void addCityCallingInProvinceRecords(CallRecord callRecord){ CityCallingInProvinceRecords.add(callRecord); } public void addCityCallingInLandRecords(CallRecord callRecord){ CityCallingInLandRecords.add(callRecord); } public ArrayList<CallRecord> getCityCallingInCityRecords() { return CityCallingInCityRecords; } public ArrayList<CallRecord> getCityCallingInProvinceRecords() { return CityCallingInProvinceRecords; } public ArrayList<CallRecord> getCityCallingInLandRecords() { return CityCallingInLandRecords; } public void addCallingInCityRecords(CallRecord callRecord) { CallingInCityRecords.add(callRecord); } public void addCallingInProvinceRecords(CallRecord callRecord) { CallingInProvinceRecords.add(callRecord); } public void addCallingInLandRecords(CallRecord callRecord) { CallingInLandRecords.add(callRecord); } public void addProvinceCallingRecords(CallRecord callRecord) { ProvinceCallingRecords.add(callRecord); } public void addLandCallingRecords(CallRecord callRecord) { LandCallingRecords.add(callRecord); } public void addanswerInLandRecords(CallRecord callRecord) { answerInLandRecords.add(callRecord); } public ArrayList<CallRecord> getCallingInCityRecords() { return CallingInCityRecords; } public ArrayList<CallRecord> getCallingInProvinceRecords() { return CallingInProvinceRecords; } public ArrayList<CallRecord> getCallingInLandRecords() { return CallingInLandRecords; } public ArrayList<CallRecord> getProvinceCallingRecords() { return ProvinceCallingRecords; } public ArrayList<CallRecord> getLandCallingRecords() { return LandCallingRecords; } public ArrayList<CallRecord> getAnswerInLandRecords() { return answerInLandRecords; } }
SourceMonitor分析图:
IDEA生成程序类图:
总结:
本题是电信计费系列的第二题,在上一题座机计费的基础上添加了手机计费的功能和计费准则,本题关键点在于添加新继承在原抽象类下的子类。举个例子,CallChargeRule是电信计费的抽象规则类,但是在手机座机并列计费的情况下,对于抽象规则类我们要添加继承在它之下的子类。本题中还加入了更多非常复杂的格式判定,对正则表达式的理解和掌握程度有非常大的考察力度。
7-3 电信计费系列3-短信计费
实现一个简单的电信计费程序,针对手机的短信采用如下计费方式:
1、接收短信免费,发送短信0.1元/条,超过3条0.2元/条,超过5条0.3元/条。
2、如果一次发送短信的字符数量超过10个,按每10个字符一条短信进行计算。
输入:
输入信息包括两种类型
1、逐行输入南昌市手机用户开户的信息,每行一个用户。
格式:u-号码 计费类型 (计费类型包括:0-座机 1-手机实时计费 2-手机A套餐 3-手机短信计费)
例如:u-13305862264 3
座机号码由区号和电话号码拼接而成,电话号码包含7-8位数字,区号最高位是0。
手机号码由11位数字构成,最高位是1。
本题只针对类型3-手机短信计费。
2、逐行输入本月某些用户的短信信息,短信的格式:
m-主叫号码,接收号码,短信内容 (短信内容只能由数字、字母、空格、英文逗号、英文句号组成)
m-18907910010 13305862264 welcome to jiangxi.
m-13305862264 18907910010 thank you.
注意:以上两类信息,先输入所有开户信息,再输入所有通讯信息,最后一行以“end”结束。
输出:
根据输入的详细短信信息,计算所有已开户的用户的当月短信费用(精确到小数点后2位,单位元)。假设每个用户初始余额是100元。
每条短信信息均单独计费后累加,不是将所有信息累计后统一计费。
格式:号码+英文空格符+总的话费+英文空格符+余额
每个用户一行,用户之间按号码字符从小到大排序。
错误处理:
输入数据中出现的不符合格式要求的行一律忽略。
本题只做格式的错误判断,无需做内容上不合理的判断,比如同一个电话两条通讯记录的时间有重合、开户号码非南昌市的号码、自己给自己打电话等,此类情况都当成正确的输入计算。但时间的输入必须符合要求,比如不能输入2022.13.61
28:72:65。
本题只考虑短信计费,不考虑通信费用以及月租费。
建议类图:
参见图1、2、3:
图1中User是用户类,包括属性:
userRecords (用户记录)、balance(余额)、chargeMode(计费方式)、number(号码)。
ChargeMode是计费方式的抽象类:
chargeRules是计费方式所包含的各种计费规则的集合,ChargeRule类的定义见图3。
getMonthlyRent()方法用于返回月租(monthlyRent)。
UserRecords是用户记录类,保存用户各种通话、短信的记录,
各种计费规则将使用其中的部分或者全部记录。
其属性从上到下依次是:
市内拨打电话、省内(不含市内)拨打电话、省外拨打电话、
市内接听电话、省内(不含市内)接听电话、省外接听电话的记录
以及发送短信、接收短信的记录。
图2中CommunicationRecord是抽象的通讯记录类:
包含callingNumber拨打号码、answerNumber接听号码两个属性。
CallRecord(通话记录)、MessageRecord(短信记录)是它的子类。
图3是计费规则的相关类,这些类的核心方法是:
calCost(ArrayList callRecords)。
该方法针根据输入参数callRecords中的所有记录计算某用户的某一项费用;如市话费。
输入参数callRecords的约束条件:必须是某一个用户的符合计费规则要求的所有记录。
SendMessageRule是发送短信的计费规则类,用于计算发送短信的费用。
LandPhoneInCityRule、LandPhoneInProvinceRule、LandPhoneInLandRule三个类分别是座机拨打市内、省内、省外电话的计费规则类,用于实现这三种情况的费用计算。
(提示:可以从UserRecords类中获取各种类型的callRecords)。
注意:以上图中所定义的类不是限定要求,根据实际需要自行补充或修改。
输入样例:
在这里给出一组输入。例如:
u-18907910010 3
m-18907910010 13305862264 aaaaaaaaaaaaaaaaaaaaaaa
end
输出样例:
在这里给出相应的输出。例如:
18907910010 0.3 99.7
在这里给出一组输入。例如:
u-18907910010 3
m-18907910010 13305862264 aaaaaaaaaaaa
m-18907910010 13305862264 aaaaaaa.
m-18907910010 13305862264 bb,bbbb
end
输出样例1:
在这里给出相应的输出。例如:
18907910010 0.5 99.5
全码如下:

import java.text.DecimalFormat; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.Scanner; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); String str = sc.nextLine(); ArrayList<User> Array = new ArrayList<>(); boolean flag = false; while (!str.equals("end")) { // flag = false; if (str.matches("u\\-1\\d{10}\\s3")) { String str1[] = str.split("-"); String str2[] = str1[1].split(" "); for (int i = 0; i < Array.size(); i++) { if (Array.get(i).getNumber().equals(str2[0])) { flag = true; break; } } if (flag == false) { User user = new User(str2[0]); MessagePhoneCharging messagePhoneCharging = new MessagePhoneCharging(); user.setChargeMode(messagePhoneCharging); Array.add(user); } } if(str.matches("m\\-1\\d{10}\\s1\\d{10}\\s[0-9a-zA-Z\\s\\.\\,]+")){ String str3[] = str.split("-"); String str4[] = str3[1].split(" "); String str5 = str3[1].substring(24); for (int i = 0; i < Array.size(); i++) { if(Array.get(i).getNumber().equals(str4[0])){ Array.get(i).getUserRecords().addSendMessageRecords(new MessageRecord(str5)); } } } str = sc.nextLine(); } //按照号码顺序排列 Collections.sort(Array, new Comparator<User>() { @Override public int compare(User o1, User o2) { return o1.getNumber().compareTo(o2.getNumber()); } }); for (int i = 0; i < Array.size(); i++) { System.out.println(Array.get(i).getNumber() + " " + new DecimalFormat("0.0#").format(Array.get(i).getChargeMode().calCost(Array.get(i).getUserRecords())) + " " + new DecimalFormat("0.0#").format(Array.get(i).getBalance() - Array.get(i).getChargeMode().calCost(Array.get(i).getUserRecords()) - Array.get(i).getChargeMode().getMonthlyRent())); } } } abstract class CallChargeRule extends ChargeRule { public abstract double calCost(ArrayList<CallRecord> callRecords); } class CallRecord extends CommunicationRecord { private Date startTime; private Date endTime; private String callingAddressAreaCode; private String answerAddressAreaCode; public CallRecord() { } public CallRecord(String startTime, String endTime) { SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy.MM.dd HH:mm:ss"); try { this.startTime = simpleDateFormat.parse(startTime); } catch (ParseException e) { } try { this.endTime = simpleDateFormat.parse(endTime); } catch (ParseException e) { } } public Date getStartTime() { return startTime; } public void setStartTime(Date startTime) { this.startTime = startTime; } public String getCallingAddressAreaCode() { return callingAddressAreaCode; } public void setCallingAddressAreaCode(String callingAddressAreaCode) { this.callingAddressAreaCode = callingAddressAreaCode; } public Date getEndTime() { return endTime; } public void setEndTime(Date endTime) { this.endTime = endTime; } public String getAnswerAddressAreaCode() { return answerAddressAreaCode; } public void setAnswerAddressAreaCode(String answerAddressAreaCode) { this.answerAddressAreaCode = answerAddressAreaCode; } } abstract class ChargeMode { ArrayList<ChargeRule> chargeRules = new ArrayList<>(); public ArrayList<ChargeRule> getChargeRules() { return this.chargeRules; } public void setChargeRules(ArrayList<ChargeRule> chargeRules) { this.chargeRules = chargeRules; } public abstract double calCost(UserRecords userRecords); public abstract double getMonthlyRent(); } abstract class ChargeRule { } abstract class CommunicationRecord { protected String callingNumber; protected String answerNumber; public CommunicationRecord(){ } public CommunicationRecord(String callingNumber, String answerNumber) { this.callingNumber = callingNumber; this.answerNumber = answerNumber; } public String getCallingNumber() { return callingNumber; } public void setCallingNumber(String callingNumber) { this.callingNumber = callingNumber; } public String getAnswerNumber() { return answerNumber; } public void setAnswerNumber(String answerNumber) { this.answerNumber = answerNumber; } } abstract class MessageChargeRule extends ChargeRule{ public abstract double calCost(ArrayList<MessageRecord> messageRecord); } class MessagePhoneCharging extends ChargeMode { @Override public double calCost(UserRecords userRecords) { double cost = new SendMessageRule().calCost(userRecords.getSendMessageRecords()); return cost; } @Override public double getMonthlyRent() { return 0; } } class MessageRecord extends CommunicationRecord { private String message; public MessageRecord(String message) { this.message = message; } public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } } class SendMessageRule extends MessageChargeRule { @Override public double calCost(ArrayList<MessageRecord> messageRecord) { double cost = 0; int cnt = 0; for (MessageRecord m : messageRecord) { if (m.getMessage().length() % 10 * 10 / 10 == 0) { cnt += m.getMessage().length() * 10 / 100; } else { cnt += m.getMessage().length() * 10 / 100 + 1; } } if (cnt > 5) { cost = 0.7 + (cnt - 5) * 0.3; } else if (cnt <= 5 && cnt > 3) { cost = 0.3 + 0.2 * (cnt - 3); } else { cost = 0.1 * cnt; } return cost; } } class User { private UserRecords userRecords = new UserRecords(); private double balance = 100; private String number; private ChargeMode chargeMode; public User(String number) { this.number = number; } public double CalBalance() { //return 0; return balance - calCost() - chargeMode.getMonthlyRent(); } public double calCost() { return 0; } public UserRecords getUserRecords() { return userRecords; } public void setUserRecords(UserRecords userRecords) { this.userRecords = userRecords; } public double getBalance() { return balance; } public String getNumber() { return number; } public void setNumber(String number) { this.number = number; } public ChargeMode getChargeMode() { return chargeMode; } public void setChargeMode(ChargeMode chargeMode) { this.chargeMode = chargeMode; } } class UserRecords { private ArrayList<CallRecord> CallingInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CallingInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CallingInLandRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInCityRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInProvinceRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> CityCallingInLandRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> ProvinceCallingRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> LandCallingRecords = new ArrayList<CallRecord>(); private ArrayList<CallRecord> answerInLandRecords = new ArrayList<CallRecord>(); private ArrayList<MessageRecord> sendMessageRecords = new ArrayList<MessageRecord>(); private ArrayList<MessageRecord> receiveMessageRecords = new ArrayList<MessageRecord>(); public void addCityCallingInCityRecords(CallRecord callRecord) { CityCallingInCityRecords.add(callRecord); } public void addCityCallingInProvinceRecords(CallRecord callRecord) { CityCallingInProvinceRecords.add(callRecord); } public void addCityCallingInLandRecords(CallRecord callRecord) { CityCallingInLandRecords.add(callRecord); } public ArrayList<CallRecord> getCityCallingInCityRecords() { return CityCallingInCityRecords; } public ArrayList<CallRecord> getCityCallingInProvinceRecords() { return CityCallingInProvinceRecords; } public ArrayList<CallRecord> getCityCallingInLandRecords() { return CityCallingInLandRecords; } public void addCallingInCityRecords(CallRecord callRecord) { CallingInCityRecords.add(callRecord); } public void addCallingInProvinceRecords(CallRecord callRecord) { CallingInProvinceRecords.add(callRecord); } public void addCallingInLandRecords(CallRecord callRecord) { CallingInLandRecords.add(callRecord); } public void addProvinceCallingRecords(CallRecord callRecord) { ProvinceCallingRecords.add(callRecord); } public void addLandCallingRecords(CallRecord callRecord) { LandCallingRecords.add(callRecord); } public void addanswerInLandRecords(CallRecord callRecord) { answerInLandRecords.add(callRecord); } public void addSendMessageRecords(MessageRecord sendMessageRecord) { sendMessageRecords.add(sendMessageRecord); } public void addReceiveMessageRecords(MessageRecord receiveMessageRecord) { receiveMessageRecords.add(receiveMessageRecord); } public ArrayList<CallRecord> getCallingInCityRecords() { return CallingInCityRecords; } public ArrayList<CallRecord> getCallingInProvinceRecords() { return CallingInProvinceRecords; } public ArrayList<CallRecord> getCallingInLandRecords() { return CallingInLandRecords; } public ArrayList<CallRecord> getProvinceCallingRecords() { return ProvinceCallingRecords; } public ArrayList<CallRecord> getLandCallingRecords() { return LandCallingRecords; } public ArrayList<CallRecord> getAnswerInLandRecords() { return answerInLandRecords; } public ArrayList<MessageRecord> getSendMessageRecords() { return sendMessageRecords; } public ArrayList<MessageRecord> getReceiveMessageRecords() { return receiveMessageRecords; } }
SourceMonitor分析图:
IDEA生成程序类图:
总结:
本题是电信计费系列的第三题,本题的测试点仅有短信计费,和手机计费不同,我们可以像考虑座机计费的构成那样去思考,而无需进一步完善先前的体系。
踩坑心得
在过去一个月的学习中,我再次总结了如下几个自己容易犯的错误:
1、设计类的时候忘记添加构造函数。一般题目中其实很少用到构造函数,但作为学习的过程,添加构造函数也是创建一个完整类的一部分。以及设计父类的构造方法后忘记也要在子类的构造方法中添加super。
2、对正则表达式的理解匮乏。
3、对新知识的掌握尚未完全。在这个阶段的学习中,我们在类与对象、方法基本构建、继承、多态的使用的基础上,新学习并着重强化了对抽象类、抽象方法的构造,接口的加入(这个在实验里比较多)。
改进意见
在看到题目时,首先要分析题目中蕴含的结构。比如电信计费都给了类图,相当于已经告知了基本结构,那我们就要分析都有哪些类,创建什么对象,继承关系是怎样的,子类的方法哪些可以直接继承抽象父类使用,哪些是自己独有的方法。其次,要扎实自己的基本功。思路清晰的同时,要学会根据不同的题干环境进行变通。比如第一题,建立了用户方面,还需要建立通讯记录,既然是电信计费,那就要建立对应的计费准则。第一题仅考虑电信计费中的座机,第二题就要学会在其基础上加入对手机的操作,第三题由于出题的原因可以简略很多。
总结
通过SourceMontior检测发现,自己写的部分代码各项数值的测量值和规定的范围相差甚远,说明还有很多的提升空间。Java是一门以面向对象为主的编程语言,所以对类和对象及其方法的要求十分看重。在对类的使用上还有所欠缺,相信在以后的学习中会越来越好。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义