|
@@ -0,0 +1,184 @@
|
|
|
+package com.qmrb.parking.fee;
|
|
|
+
|
|
|
+import java.time.Duration;
|
|
|
+import java.time.LocalDateTime;
|
|
|
+import java.time.LocalTime;
|
|
|
+import java.util.ArrayList;
|
|
|
+import java.util.List;
|
|
|
+
|
|
|
+public class ParkingFeeCalculator {
|
|
|
+
|
|
|
+
|
|
|
+ public static class FeeRule {
|
|
|
+ public int freeDurationMinutes;
|
|
|
+ public double firstCycleFee;
|
|
|
+ public int firstCycleDurationMinutes;
|
|
|
+ public int cycleDurationMinutes;
|
|
|
+ public double cycleFee;
|
|
|
+ public double dailyCapFee;
|
|
|
+ public int gracePeriodMinutes;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ public static class ParkingRecord {
|
|
|
+ public LocalDateTime entryTime;
|
|
|
+ public LocalDateTime exitTime;
|
|
|
+ public LocalDateTime monthlyCardExpiryTime;
|
|
|
+ public double paidAmount;
|
|
|
+ public List<PaymentRecord> paymentRecords = new ArrayList<>();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ public static class PaymentRecord {
|
|
|
+ public LocalDateTime paymentTime;
|
|
|
+ public double amount;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ public static double calculateParkingFee(ParkingRecord parkingRecord, FeeRule feeRule, boolean isMonthlyCardMode) {
|
|
|
+
|
|
|
+ LocalDateTime billingStartTime = parkingRecord.entryTime;
|
|
|
+ if (parkingRecord.monthlyCardExpiryTime != null && parkingRecord.monthlyCardExpiryTime.isAfter(parkingRecord.entryTime)) {
|
|
|
+ billingStartTime = parkingRecord.monthlyCardExpiryTime;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ LocalDateTime billingEndTime = isMonthlyCardMode
|
|
|
+ ? LocalDateTime.of(parkingRecord.exitTime.toLocalDate(), LocalTime.MIN)
|
|
|
+ : parkingRecord.exitTime;
|
|
|
+
|
|
|
+
|
|
|
+ Duration billingDuration = Duration.between(billingStartTime, billingEndTime);
|
|
|
+
|
|
|
+
|
|
|
+ if (feeRule.gracePeriodMinutes > 0) {
|
|
|
+ for (PaymentRecord payment : parkingRecord.paymentRecords) {
|
|
|
+ if (payment.paymentTime.isAfter(billingEndTime.minusMinutes(feeRule.gracePeriodMinutes))) {
|
|
|
+ billingDuration = billingDuration.minusMinutes(feeRule.gracePeriodMinutes);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ if (billingDuration.toMinutes() <= feeRule.freeDurationMinutes) {
|
|
|
+ return 0.0;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ double totalFee = 0.0;
|
|
|
+
|
|
|
+
|
|
|
+ if (billingDuration.toHours() >= 24) {
|
|
|
+ int fullDays = (int) (billingDuration.toHours() / 24);
|
|
|
+ totalFee += fullDays * feeRule.dailyCapFee;
|
|
|
+
|
|
|
+
|
|
|
+ Duration remainingDuration = billingDuration.minusHours(fullDays * 24);
|
|
|
+ double remainingFee = calculateRemainingFeeDue(remainingDuration, feeRule);
|
|
|
+ totalFee += Math.min(remainingFee, feeRule.dailyCapFee);
|
|
|
+ }
|
|
|
+
|
|
|
+ else {
|
|
|
+ totalFee = calculateRemainingFee(billingDuration, feeRule);
|
|
|
+ totalFee = Math.min(totalFee, feeRule.dailyCapFee);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ totalFee = Math.max(0, totalFee - parkingRecord.paidAmount);
|
|
|
+
|
|
|
+ return totalFee;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static double calculateRemainingFee(Duration duration, FeeRule feeRule) {
|
|
|
+ double totalFee = 0.0;
|
|
|
+
|
|
|
+
|
|
|
+ Duration remainingDuration = duration.minusMinutes(feeRule.firstCycleDurationMinutes);
|
|
|
+ totalFee += feeRule.firstCycleFee;
|
|
|
+
|
|
|
+
|
|
|
+ if (remainingDuration.toMinutes() > 0) {
|
|
|
+ int cycles = (int) Math.ceil((double) remainingDuration.toMinutes() / feeRule.cycleDurationMinutes);
|
|
|
+ totalFee += cycles * feeRule.cycleFee;
|
|
|
+ }
|
|
|
+
|
|
|
+ return totalFee;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * 超过24小时的剩余不足24小时部分
|
|
|
+ * @param duration
|
|
|
+ * @param feeRule
|
|
|
+ * @return
|
|
|
+ */
|
|
|
+ private static double calculateRemainingFeeDue(Duration duration, FeeRule feeRule) {
|
|
|
+ double totalFee = 0.0;
|
|
|
+
|
|
|
+ if (duration.toMinutes() > 0) {
|
|
|
+ int cycles = (int) Math.ceil((double) duration.toMinutes() / feeRule.cycleDurationMinutes);
|
|
|
+ totalFee += cycles * feeRule.cycleFee;
|
|
|
+ }
|
|
|
+
|
|
|
+ return totalFee;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ */
|
|
|
+ public static void e2(){
|
|
|
+
|
|
|
+ FeeRule feeRule = new FeeRule();
|
|
|
+ feeRule.freeDurationMinutes = 10;
|
|
|
+ feeRule.firstCycleFee = 20.0;
|
|
|
+ feeRule.firstCycleDurationMinutes = 120;
|
|
|
+ feeRule.cycleDurationMinutes = 30;
|
|
|
+ feeRule.cycleFee = 5.0;
|
|
|
+ feeRule.dailyCapFee = 50.0;
|
|
|
+ feeRule.gracePeriodMinutes = 15;
|
|
|
+
|
|
|
+
|
|
|
+ ParkingRecord parkingRecord = new ParkingRecord();
|
|
|
+ parkingRecord.entryTime = LocalDateTime.of(2025, 3, 4, 0, 0);
|
|
|
+ parkingRecord.exitTime = LocalDateTime.of(2025, 3, 5, 0, 16);
|
|
|
+ parkingRecord.monthlyCardExpiryTime = null;
|
|
|
+ parkingRecord.paidAmount = 0.0;
|
|
|
+
|
|
|
+
|
|
|
+ double fee = calculateParkingFee(parkingRecord, feeRule, false);
|
|
|
+ System.out.println("停车费用: " + fee + " 元");
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * 设【免费时长】=0;【24小时收费金额】=30元;【首周期计费时长】=60分钟;【首周期计费金额】=20元;【计费周期】=30分钟;【每周期计费金额】=2元
|
|
|
+ */
|
|
|
+ public static void e1(){
|
|
|
+
|
|
|
+ FeeRule feeRule = new FeeRule();
|
|
|
+ feeRule.freeDurationMinutes = 0;
|
|
|
+ feeRule.firstCycleFee = 20.0;
|
|
|
+ feeRule.firstCycleDurationMinutes = 60;
|
|
|
+ feeRule.cycleDurationMinutes = 30;
|
|
|
+ feeRule.cycleFee = 2.0;
|
|
|
+ feeRule.dailyCapFee = 30.0;
|
|
|
+ feeRule.gracePeriodMinutes = 15;
|
|
|
+
|
|
|
+
|
|
|
+ ParkingRecord parkingRecord = new ParkingRecord();
|
|
|
+ parkingRecord.entryTime = LocalDateTime.of(2025, 3, 3, 0, 0);
|
|
|
+ parkingRecord.exitTime = LocalDateTime.of(2025, 3, 8, 10, 0);
|
|
|
+ parkingRecord.monthlyCardExpiryTime = null;
|
|
|
+ parkingRecord.paidAmount = 0.0;
|
|
|
+
|
|
|
+
|
|
|
+ double fee = calculateParkingFee(parkingRecord, feeRule, false);
|
|
|
+ System.out.println("停车费用: " + fee + " 元");
|
|
|
+ }
|
|
|
+
|
|
|
+ * @param args
|
|
|
+ */
|
|
|
+ public static void main(String[] args) {
|
|
|
+ e1();
|
|
|
+
|
|
|
+ }
|
|
|
+}
|