Compare commits
	
		
			7 Commits
		
	
	
		
			674de604e8
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6495f56e25 | |||
| d0d9d6d18b | |||
| 2949a48e63 | |||
| b5a118994a | |||
| 043c1b71a1 | |||
| cc687b9536 | |||
| 5d9a5dda10 | 
							
								
								
									
										1
									
								
								.idea/encodings.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								.idea/encodings.xml
									
									
									
										generated
									
									
									
								
							| @@ -2,5 +2,6 @@ | |||||||
| <project version="4"> | <project version="4"> | ||||||
|   <component name="Encoding"> |   <component name="Encoding"> | ||||||
|     <file url="file://$PROJECT_DIR$/src/main/java" charset="UTF-8" /> |     <file url="file://$PROJECT_DIR$/src/main/java" charset="UTF-8" /> | ||||||
|  |     <file url="file://$PROJECT_DIR$/src/main/resources" charset="UTF-8" /> | ||||||
|   </component> |   </component> | ||||||
| </project> | </project> | ||||||
							
								
								
									
										2
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							| @@ -8,5 +8,5 @@ | |||||||
|       </list> |       </list> | ||||||
|     </option> |     </option> | ||||||
|   </component> |   </component> | ||||||
|   <component name="ProjectRootManager" version="2" languageLevel="JDK_22" project-jdk-name="21" project-jdk-type="JavaSDK" /> |   <component name="ProjectRootManager" version="2" languageLevel="JDK_19" project-jdk-name="19" project-jdk-type="JavaSDK" /> | ||||||
| </project> | </project> | ||||||
							
								
								
									
										47
									
								
								pom.xml
									
									
									
									
									
								
							
							
						
						
									
										47
									
								
								pom.xml
									
									
									
									
									
								
							| @@ -9,9 +9,52 @@ | |||||||
|     <version>1.0-SNAPSHOT</version> |     <version>1.0-SNAPSHOT</version> | ||||||
|  |  | ||||||
|     <properties> |     <properties> | ||||||
|         <maven.compiler.source>21</maven.compiler.source> |         <maven.compiler.source>17</maven.compiler.source> | ||||||
|         <maven.compiler.target>21</maven.compiler.target> |         <maven.compiler.target>17</maven.compiler.target> | ||||||
|         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> |         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> | ||||||
|     </properties> |     </properties> | ||||||
|  |  | ||||||
|  |     <dependencies> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.apache.poi</groupId> | ||||||
|  |             <artifactId>poi</artifactId> | ||||||
|  |             <version>5.2.3</version> </dependency> | ||||||
|  |         <dependency> | ||||||
|  |             <groupId>org.apache.poi</groupId> | ||||||
|  |             <artifactId>poi-ooxml</artifactId> | ||||||
|  |             <version>5.2.3</version> | ||||||
|  |         </dependency> | ||||||
|  |     </dependencies> | ||||||
|  |  | ||||||
|  |     <build> | ||||||
|  |  | ||||||
|  |         <finalName>schedule</finalName> | ||||||
|  |  | ||||||
|  |         <plugins> | ||||||
|  |             <plugin> | ||||||
|  |                 <groupId>org.apache.maven.plugins</groupId> | ||||||
|  |                 <artifactId>maven-assembly-plugin</artifactId> | ||||||
|  |                 <version>3.6.0</version> | ||||||
|  |                 <configuration> | ||||||
|  |                     <archive> | ||||||
|  |                         <manifest> | ||||||
|  |                             <mainClass>com.dota.App</mainClass> | ||||||
|  |                         </manifest> | ||||||
|  |                     </archive> | ||||||
|  |                     <descriptorRefs> | ||||||
|  |                         <descriptorRef>jar-with-dependencies</descriptorRef> | ||||||
|  |                     </descriptorRefs> | ||||||
|  |                 </configuration> | ||||||
|  |                 <executions> | ||||||
|  |                     <execution> | ||||||
|  |                         <id>make-assembly</id> | ||||||
|  |                         <phase>package</phase> | ||||||
|  |                         <goals> | ||||||
|  |                             <goal>single</goal> | ||||||
|  |                         </goals> | ||||||
|  |                     </execution> | ||||||
|  |                 </executions> | ||||||
|  |             </plugin> | ||||||
|  |         </plugins> | ||||||
|  |     </build> | ||||||
| </project> | </project> | ||||||
							
								
								
									
										59
									
								
								src/main/java/com/dota/App.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								src/main/java/com/dota/App.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | package com.dota; | ||||||
|  |  | ||||||
|  | import javax.swing.*; | ||||||
|  | import java.awt.*; | ||||||
|  |  | ||||||
|  | public class App { | ||||||
|  |     public static void main(String[] args) { | ||||||
|  |         SwingUtilities.invokeLater(App::lanuch); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static String[] doctors = new String[]{"顾磊", "章亮", "唐婷婷", "杨祖怡", "李宁", "周晖"}; | ||||||
|  |  | ||||||
|  |     static void lanuch() { | ||||||
|  |         JFrame jframe = new JFrame("排班系统"); | ||||||
|  |         jframe.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); | ||||||
|  |         jframe.setSize(600,400); | ||||||
|  |         jframe.setLayout(new BorderLayout(10, 10)); | ||||||
|  |  | ||||||
|  |         var dateField = new JTextField("2025-10-13"); | ||||||
|  |         var weekNumField = new JTextField("2"); | ||||||
|  |         var filePathField = new JTextField("C:\\Users\\yangz\\Documents\\医生排班.xlsx"); | ||||||
|  |         var doctorField = new JComboBox<>(doctors); | ||||||
|  |         doctorField.setSelectedIndex(0); | ||||||
|  |         var resText = new JTextArea("点击确定按钮,输出执行结果"); | ||||||
|  |         var btn = new JButton("确定"); | ||||||
|  |  | ||||||
|  |         // 输入区域 | ||||||
|  |         JPanel inputPanel = new JPanel(new GridLayout(4,2,10,5)); | ||||||
|  |         jframe.add(inputPanel, BorderLayout.NORTH); | ||||||
|  |         inputPanel.add(new JLabel("排班日期")); | ||||||
|  |         inputPanel.add(dateField); | ||||||
|  |         inputPanel.add(new JLabel("排班周数")); | ||||||
|  |         inputPanel.add(weekNumField); | ||||||
|  |         inputPanel.add(new JLabel("第一个值班的医生")); | ||||||
|  |         inputPanel.add(doctorField); | ||||||
|  |         inputPanel.add(new JLabel("输出文件位置")); | ||||||
|  |         inputPanel.add(filePathField); | ||||||
|  |  | ||||||
|  |         //结果区域 | ||||||
|  |         JPanel resPanel = new JPanel(); | ||||||
|  |         jframe.add(resPanel, BorderLayout.CENTER); | ||||||
|  |         resPanel.add(resText); | ||||||
|  |  | ||||||
|  |         // 按钮区域 | ||||||
|  |         var btnPanel = new JPanel(); | ||||||
|  |         jframe.add(btnPanel, BorderLayout.SOUTH); | ||||||
|  |         btnPanel.add(btn); | ||||||
|  |  | ||||||
|  |         btn.addActionListener(e->{ | ||||||
|  |             resText.setText("运行中,请稍等"); | ||||||
|  |             Main.run(dateField.getText(), Integer.parseInt(weekNumField.getText()), (String) doctorField.getSelectedItem(), filePathField.getText()); | ||||||
|  |             resText.setText("运行结束,请查看排班文件"); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         jframe.setLocationRelativeTo(null); | ||||||
|  |         jframe.setVisible(true); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -3,10 +3,16 @@ package com.dota; | |||||||
| import com.dota.domain.Doctor; | import com.dota.domain.Doctor; | ||||||
| import com.dota.domain.Week; | import com.dota.domain.Week; | ||||||
| import com.dota.domain.WorkEnum; | import com.dota.domain.WorkEnum; | ||||||
|  | import org.apache.poi.ss.usermodel.Sheet; | ||||||
|  | import org.apache.poi.xssf.usermodel.XSSFWorkbook; | ||||||
|  |  | ||||||
| import java.util.Arrays; | import java.io.FileOutputStream; | ||||||
|  | import java.time.LocalDate; | ||||||
|  | import java.time.format.DateTimeFormatter; | ||||||
|  | import java.util.*; | ||||||
|  |  | ||||||
| public class Main { | public class Main { | ||||||
|  |     static String filePath = "C:\\Users\\yangz\\Documents\\医生排班.xlsx"; | ||||||
|     static Doctor gl; |     static Doctor gl; | ||||||
|     static Doctor zl; |     static Doctor zl; | ||||||
|     static Doctor ttt; |     static Doctor ttt; | ||||||
| @@ -15,7 +21,19 @@ public class Main { | |||||||
|     static Doctor zh; |     static Doctor zh; | ||||||
|     static Doctor tgy; |     static Doctor tgy; | ||||||
|  |  | ||||||
|  |     // 日期 | ||||||
|  |     static int days = 0; | ||||||
|  |     // 医生下标 | ||||||
|  |     static int idx = 0; | ||||||
|  |  | ||||||
|  |     //开始日期 | ||||||
|  |     static LocalDate day; | ||||||
|  |  | ||||||
|     public static void main(String[] args) { |     public static void main(String[] args) { | ||||||
|  |         run("2025-10-13", 2, "周晖", "C:\\Users\\yangz\\Documents\\医生排班.xlsx"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static void run(String date, int weekNum, String name, String file){ | ||||||
|         init(); |         init(); | ||||||
|         var doctorOrder = new Doctor[]{ |         var doctorOrder = new Doctor[]{ | ||||||
|                 gl, zl, ttt, zy, ln, zh |                 gl, zl, ttt, zy, ln, zh | ||||||
| @@ -24,8 +42,87 @@ public class Main { | |||||||
|                 ln, zy, ttt, zl, zh, gl, tgy |                 ln, zy, ttt, zl, zh, gl, tgy | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         for (int i = 0; i < 2; i++) { |         filePath = file; | ||||||
|             schedule(doctorOrder, outputOrder); |         write(date, weekNum, name, doctorOrder, outputOrder); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @param startDay 开始第一天 | ||||||
|  |      * @param weekNum 输出周数 | ||||||
|  |      * @param name 第一周第一天值班的医生名字 | ||||||
|  |      */ | ||||||
|  |     static void write(String startDay, int weekNum, String name, Doctor[] doctors, Doctor[] outputDoctors) { | ||||||
|  |         day = LocalDate.parse(startDay); | ||||||
|  |         for (int i = 0; i < doctors.length; i++) { | ||||||
|  |             if (doctors[i].name.equals(name)) { | ||||||
|  |                 idx = i; | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var list = new ArrayList<Doctor>(); | ||||||
|  |         for (int i = 0; i < weekNum; i++) { | ||||||
|  |             list.addAll(schedule(doctors, outputDoctors)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         try(var workbook = new XSSFWorkbook()) { | ||||||
|  |             var sheet = workbook.createSheet(); | ||||||
|  |             int rowNum = 0; | ||||||
|  |  | ||||||
|  |             for (int i = 0; i < weekNum; i++) { | ||||||
|  |                 writeHead(sheet, rowNum, day); | ||||||
|  |                 day = day.plusDays(7); | ||||||
|  |                 rowNum += 2; | ||||||
|  |                 writeSchedult(sheet, rowNum, list.subList(i * 7, (i+1) * 7)); | ||||||
|  |                 rowNum += 8; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             for (int i = 0; i < 8; i++) { | ||||||
|  |                 sheet.autoSizeColumn(i); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             try(var output = new FileOutputStream(filePath)) { | ||||||
|  |              workbook.write(output); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         }catch (Exception e) { | ||||||
|  |             e.printStackTrace(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 写入表头,周几和日期 | ||||||
|  |      */ | ||||||
|  |     static void writeHead(Sheet sheet, int rowNum, LocalDate day){ | ||||||
|  |         DateTimeFormatter mmdd = DateTimeFormatter.ofPattern("MM月dd日"); | ||||||
|  |         int i = 1; | ||||||
|  |         var row = sheet.createRow(rowNum++); | ||||||
|  |         var r2 = sheet.createRow(rowNum); | ||||||
|  |         for (Week w:Week.values()) { | ||||||
|  |             var cell = row.createCell(i); | ||||||
|  |             cell.setCellValue(w.getValue()); | ||||||
|  |  | ||||||
|  |             cell = r2.createCell(i++); | ||||||
|  |             cell.setCellValue(day.format(mmdd)); | ||||||
|  |             day = day.plusDays(1); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * 写入医生排班 | ||||||
|  |      */ | ||||||
|  |     static void writeSchedult(Sheet sheet, int rowNum, List<Doctor> doctors){ | ||||||
|  |         for (Doctor doctor : doctors) { | ||||||
|  |             var row = sheet.createRow(rowNum++); | ||||||
|  |             int colNum = 0; | ||||||
|  |             var cell = row.createCell(colNum++); | ||||||
|  |             cell.setCellValue(doctor.name); | ||||||
|  |             for (WorkEnum work : doctor.works) { | ||||||
|  |                 cell = row.createCell(colNum++); | ||||||
|  |                 cell.setCellValue(work.getName()); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -55,16 +152,11 @@ public class Main { | |||||||
|         ttt.swap = zl; |         ttt.swap = zl; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     static void schedule(Doctor[] doctors, Doctor[] outputOrder) { |     //生成一周的排班 | ||||||
|         // 日期 |     static List<Doctor> schedule(Doctor[] doctors, Doctor[] outputOrder) { | ||||||
|         int days = 0; |         var res = new ArrayList<Doctor>(); | ||||||
|         // 医生下标 |  | ||||||
|         int idx = 0; |  | ||||||
|  |  | ||||||
|         for (int k = 0; k < 2; k++) { |  | ||||||
|             for (int i = 0; i < 7; i++) { |             for (int i = 0; i < 7; i++) { | ||||||
|                 var d = doctors[idx]; |                 var d = doctors[idx]; | ||||||
|  |  | ||||||
|                 if (d.firstRest) { |                 if (d.firstRest) { | ||||||
|                     d.firstRest = false; |                     d.firstRest = false; | ||||||
|                     d.works[0] = WorkEnum.HOLIDAY; |                     d.works[0] = WorkEnum.HOLIDAY; | ||||||
| @@ -92,40 +184,63 @@ public class Main { | |||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 days = (days + 1) % 7; |                 days = (days + 1) % 7; | ||||||
|                 idx = (idx + 1) % doctors.length; |                 idx = (idx + 1) % 6; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             fix(doctors); |             fix(doctors); | ||||||
|             rest(doctors); |             rest(doctors); | ||||||
|             print(outputOrder); |             for (Doctor d : outputOrder) { | ||||||
|             for (Doctor d : doctors) { |                 res.add(d.clone()); | ||||||
|                 d.reset(); |                 d.reset(); | ||||||
|             } |             } | ||||||
|  |         return res; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static class Rest { | ||||||
|  |         int day; | ||||||
|  |         int cnt; | ||||||
|  |         public Rest(int day, int cnt){ | ||||||
|  |             this.day = day; | ||||||
|  |             this.cnt = cnt; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // 给有调休的安排休息 |     // 给有调休的安排休息 | ||||||
|     // 保证同一组必须有一个在 |     // 保证同一组必须有一个在 | ||||||
|     static void rest(Doctor[] doctors) { |     static void rest(Doctor[] doctors) { | ||||||
|  |         var list = new Rest[5]; | ||||||
|  |         for (int i = 0; i < 5; i++) { | ||||||
|  |             list[i] = new Rest(i, 0); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         for (Doctor doctor : doctors) { |         for (Doctor doctor : doctors) { | ||||||
|             if (doctor.lastRestCount == 0) { |             for (int i = 0; i < 5; i++) { | ||||||
|  |                 if (doctor.works[i] == WorkEnum.DUTY || doctor.works[i] == WorkEnum.WORK) { | ||||||
|  |                     list[i].cnt++; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var queue = new PriorityQueue<Rest>((a,b)->b.cnt-a.cnt); | ||||||
|  |         for (Rest rest : list) { | ||||||
|  |             queue.offer(rest); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         for (Doctor doctor : doctors) { | ||||||
|  |             if (doctor.restCount == 0) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             int count = 0; |             int i = 0; | ||||||
|             for (int j = 0; j < 5; j++) { |             var temp = new ArrayList<Rest>(); | ||||||
|                 if (checkRest(j, doctor, doctors)) { |             while (i < doctor.restCount) { | ||||||
|                     doctor.works[j] = WorkEnum.REST; |                 Rest r = queue.poll(); | ||||||
|                     count++; |                 if (checkRest(r.day, doctor, doctors)) { | ||||||
|  |                     doctor.works[r.day] = WorkEnum.REST; | ||||||
|  |                     i++; | ||||||
|                 } |                 } | ||||||
|  |                 temp.add(r); | ||||||
|                 if (count==doctor.lastRestCount) { |  | ||||||
|                     break; |  | ||||||
|             } |             } | ||||||
|             } |             queue.addAll(temp); | ||||||
|         } |  | ||||||
|         for (Doctor doctor : doctors) { |  | ||||||
|             doctor.lastRestCount = doctor.restCount; |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -261,14 +376,18 @@ public class Main { | |||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     static void print(Doctor[] outputOrder) { |     static void print(List<Doctor> outputOrder) { | ||||||
|  |         var size = outputOrder.size() / 7; | ||||||
|  |         int idx = 0; | ||||||
|  |         for (int i = 0; i < size; i++) { | ||||||
|             System.out.printf("%10s", " "); |             System.out.printf("%10s", " "); | ||||||
|             for (Week value : Week.values()) { |             for (Week value : Week.values()) { | ||||||
|                 System.out.printf("%10s", value.getValue()); |                 System.out.printf("%10s", value.getValue()); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             System.out.println(" "); |             System.out.println(" "); | ||||||
|         for (Doctor doctor : outputOrder) { |             for (int j = 0;j<7;j++) { | ||||||
|  |                 var doctor = outputOrder.get(idx++); | ||||||
|                 System.out.printf("%10s", doctor.name); |                 System.out.printf("%10s", doctor.name); | ||||||
|                 for (WorkEnum work : doctor.works) { |                 for (WorkEnum work : doctor.works) { | ||||||
|                     System.out.printf("%10s", work.getName()); |                     System.out.printf("%10s", work.getName()); | ||||||
| @@ -278,3 +397,4 @@ public class Main { | |||||||
|             System.out.println(" "); |             System.out.println(" "); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | } | ||||||
| @@ -2,9 +2,7 @@ package com.dota.domain; | |||||||
|  |  | ||||||
| public class Doctor { | public class Doctor { | ||||||
|     public String name; |     public String name; | ||||||
|     // 上周的调休 |     // 周五六日值班的调休 | ||||||
|     public int lastRestCount; |  | ||||||
|     // 下周的调休 |  | ||||||
|     public int restCount = 0; |     public int restCount = 0; | ||||||
|  |  | ||||||
|     // 门诊时间 |     // 门诊时间 | ||||||
| @@ -33,7 +31,15 @@ public class Doctor { | |||||||
|         this.name = name; |         this.name = name; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Doctor clone() { | ||||||
|  |         var d = new Doctor(this.name); | ||||||
|  |         System.arraycopy(this.works, 0, d.works, 0, this.works.length); | ||||||
|  |         return d; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public void reset() { |     public void reset() { | ||||||
|  |         this.restCount = 0; | ||||||
|         for (int i = 0; i < 5; i++) { |         for (int i = 0; i < 5; i++) { | ||||||
|             works[i] = WorkEnum.WORK; |             works[i] = WorkEnum.WORK; | ||||||
|         } |         } | ||||||
| @@ -42,10 +48,11 @@ public class Doctor { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         switch (name) { |         switch (name) { | ||||||
|  |             case "田国燕" -> works[0] = WorkEnum.OPD; | ||||||
|             case "周晖" -> works[1] = WorkEnum.OPD; |             case "周晖" -> works[1] = WorkEnum.OPD; | ||||||
|             case "唐婷婷" -> works[2] = WorkEnum.OPD; |             case "唐婷婷" -> works[2] = WorkEnum.OPD; | ||||||
|             case "李宁" ->  works[3] = WorkEnum.OPD; |             case "李宁" ->  works[4] = WorkEnum.OPD; | ||||||
|             case "章亮" -> works[4] = WorkEnum.OPD; |             case "章亮" -> works[3] = WorkEnum.OPD; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user