run_dqn_decide.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. """
  2. run_dqn_decide.py
  3. UF 超滤 DQN 决策主入口(Inference / Online Assist)
  4. 职责:
  5. 1. 构造物理世界(physics)
  6. 2. 实例化决策器(UFDQNDecider)
  7. 3. 构造当前工厂状态(observation)
  8. 4. 调用模型给出策略建议
  9. 5. 生成 PLC 下发指令(限幅 / 限速)
  10. 6. 评估该指令在物理模型下的效果(只评估,不下发)
  11. """
  12. from pathlib import Path
  13. # ============================================================
  14. # 1. 导入模块
  15. # ============================================================
  16. CURRENT_DIR = Path(__file__).resolve().parent
  17. UF_RL_ROOT = CURRENT_DIR.parents[2] # uf_train # uf-rl
  18. # ========== 参数 / 物理 ==========
  19. from env.uf_resistance_models_load import load_resistance_models
  20. from env.uf_physics import UFPhysicsModel
  21. from env.env_params import UFState, UFActionSpec
  22. from env.env_config_loader import EnvConfigLoader, create_env_params_from_yaml
  23. # ========== 决策器 ==========
  24. from rl_model.DQN.uf_decide.dqn_decider import UFDQNDecider
  25. def build_physics(IS_TIMES, phys_params):
  26. """
  27. 构造与训练一致的物理模型(只做一次)
  28. """
  29. res_fp, res_bw = load_resistance_models(phys_params)
  30. physics = UFPhysicsModel(
  31. phys_params=phys_params,
  32. resistance_model_fp=res_fp,
  33. resistance_model_bw=res_bw,
  34. IS_TIMES = IS_TIMES
  35. )
  36. return physics
  37. def generate_plc_instructions(action_spec,current_L_s, current_t_bw_s, model_prev_L_s, model_prev_t_bw_s, model_L_s, model_t_bw_s):
  38. """
  39. 根据工厂当前值、模型上一轮决策值和模型当前轮决策值,生成PLC指令。
  40. 新增功能:
  41. 1. 处理None值情况:如果模型上一轮值为None,则使用工厂当前值;
  42. 如果工厂当前值也为None,则返回None并提示错误。
  43. """
  44. action_spec = action_spec
  45. adjustment_threshold = 1.0
  46. # 处理None值情况
  47. if model_prev_L_s is None:
  48. if current_L_s is None:
  49. print("错误: 过滤时长的工厂当前值和模型上一轮值均为None")
  50. return None, None
  51. else:
  52. # 使用工厂当前值作为基准
  53. effective_current_L = current_L_s
  54. source_L = "工厂当前值(模型上一轮值为None)"
  55. else:
  56. # 模型上一轮值不为None,继续检查工厂当前值
  57. if current_L_s is None:
  58. effective_current_L = model_prev_L_s
  59. source_L = "模型上一轮值(工厂当前值为None)"
  60. else:
  61. effective_current_L = model_prev_L_s
  62. source_L = "模型上一轮值"
  63. # 对反洗时长进行同样的处理
  64. if model_prev_t_bw_s is None:
  65. if current_t_bw_s is None:
  66. print("错误: 反洗时长的工厂当前值和模型上一轮值均为None")
  67. return None, None
  68. else:
  69. effective_current_t_bw = current_t_bw_s
  70. source_t_bw = "工厂当前值(模型上一轮值为None)"
  71. else:
  72. if current_t_bw_s is None:
  73. effective_current_t_bw = model_prev_t_bw_s
  74. source_t_bw = "模型上一轮值(工厂当前值为None)"
  75. else:
  76. effective_current_t_bw = model_prev_t_bw_s
  77. source_t_bw = "模型上一轮值"
  78. # 检测所有输入值是否在规定范围内(只对非None值进行检查)
  79. # 工厂当前值检查(警告)
  80. if current_L_s is not None and not (action_spec.L_min_s <= current_L_s <= action_spec.L_max_s):
  81. print(f"警告: 当前过滤时长 {current_L_s} 秒不在允许范围内 [{action_spec.L_min_s}, {action_spec.L_max_s}]")
  82. if current_t_bw_s is not None and not (action_spec.t_bw_min_s <= current_t_bw_s <= action_spec.t_bw_max_s):
  83. print(f"警告: 当前反洗时长 {current_t_bw_s} 秒不在允许范围内 [{action_spec.t_bw_min_s}, {action_spec.t_bw_max_s}]")
  84. # 模型上一轮决策值检查(警告)
  85. if model_prev_L_s is not None and not (action_spec.L_min_s <= model_prev_L_s <= action_spec.L_max_s):
  86. print(f"警告: 模型上一轮过滤时长 {model_prev_L_s} 秒不在允许范围内 [{action_spec.L_min_s}, {action_spec.L_max_s}]")
  87. if model_prev_t_bw_s is not None and not (action_spec.t_bw_min_s <= model_prev_t_bw_s <= action_spec.t_bw_max_s):
  88. print(f"警告: 模型上一轮反洗时长 {model_prev_t_bw_s} 秒不在允许范围内 [{action_spec.t_bw_min_s}, {action_spec.t_bw_max_s}]")
  89. # 模型当前轮决策值检查(错误)
  90. if model_L_s is None:
  91. raise ValueError("错误: 决策模型建议的过滤时长不能为None")
  92. elif not (action_spec.L_min_s <= model_L_s <= action_spec.L_max_s):
  93. raise ValueError(f"错误: 决策模型建议的过滤时长 {model_L_s} 秒不在允许范围内 [{action_spec.L_min_s}, {action_spec.L_max_s}]")
  94. if model_t_bw_s is None:
  95. raise ValueError("错误: 决策模型建议的反洗时长不能为None")
  96. elif not (action_spec.t_bw_min_s <= model_t_bw_s <= action_spec.t_bw_max_s):
  97. raise ValueError(f"错误: 决策模型建议的反洗时长 {model_t_bw_s} 秒不在允许范围内 [{action_spec.t_bw_min_s}, {action_spec.t_bw_max_s}]")
  98. print(f"过滤时长基准: {source_L}, 值: {effective_current_L}")
  99. print(f"反洗时长基准: {source_t_bw}, 值: {effective_current_t_bw}")
  100. # 使用选定的基准值进行计算调整
  101. L_diff = model_L_s - effective_current_L
  102. L_adjustment = 0
  103. if abs(L_diff) >= adjustment_threshold * action_spec.L_step_s:
  104. if L_diff >= 0:
  105. L_adjustment = action_spec.L_step_s
  106. else:
  107. L_adjustment = -action_spec.L_step_s
  108. next_L_s = effective_current_L + L_adjustment
  109. t_bw_diff = model_t_bw_s - effective_current_t_bw
  110. t_bw_adjustment = 0
  111. if abs(t_bw_diff) >= adjustment_threshold * action_spec.t_bw_step_s:
  112. if t_bw_diff >= 0:
  113. t_bw_adjustment = action_spec.t_bw_step_s
  114. else:
  115. t_bw_adjustment = -action_spec.t_bw_step_s
  116. next_t_bw_s = effective_current_t_bw + t_bw_adjustment
  117. return next_L_s, next_t_bw_s
  118. def calc_uf_cycle_metrics(current_state, max_tmp_during_filtration, min_tmp_during_filtration, L_s: float, t_bw_s: float):
  119. """
  120. 计算 UF 超滤系统的核心性能指标
  121. 参数:
  122. L_s (float): 单次过滤时间(秒)
  123. t_bw_s (float): 单次反洗时间(秒)
  124. 返回:
  125. dict: {
  126. "k_bw_per_ceb": 小周期次数,
  127. "ton_water_energy_kWh_per_m3": 吨水电耗,
  128. "recovery": 回收率,
  129. "net_delivery_rate_m3ph": 净供水率 (m³/h),
  130. "daily_prod_time_h": 日均产水时间 (小时/天)
  131. "max_permeability": 全周期最高渗透率(lmh/bar)
  132. }
  133. """
  134. # 模拟该参数下的超级周期
  135. info, next_state = physics.simulate_one_supercycle(current_state, L_s=L_s, t_bw_s=t_bw_s)
  136. # 获得模型模拟周期信息
  137. k_bw_per_ceb = info["k_bw_per_ceb"]
  138. ton_water_energy_kWh_per_m3 = info["ton_water_energy_kWh_per_m3"]
  139. recovery = info["recovery"]
  140. daily_prod_time_h = info["daily_prod_time_h"]
  141. # 获得模型模拟周期内最高跨膜压差/最低跨膜压差
  142. if max_tmp_during_filtration is None:
  143. max_tmp_during_filtration = info["max_TMP_during_filtration"]
  144. if min_tmp_during_filtration is None:
  145. min_tmp_during_filtration = info["min_TMP_during_filtration"]
  146. # 计算最高渗透率
  147. max_permeability = 100 * current_state.q_UF / (128*40) / min_tmp_during_filtration
  148. return {
  149. "k_bw_per_ceb": k_bw_per_ceb,
  150. "ton_water_energy_kWh_per_m3": ton_water_energy_kWh_per_m3,
  151. "recovery": recovery,
  152. "daily_prod_time_h": daily_prod_time_h,
  153. "max_permeability": max_permeability
  154. }
  155. def run_dqn_decide(
  156. model_path: Path,
  157. physics,
  158. action_spec,
  159. reward_params,
  160. state_bounds,
  161. # -------- 工厂当前值 --------
  162. current_state: UFState
  163. ):
  164. """
  165. 单轮 DQN 决策流程
  166. """
  167. # 构造决策器
  168. decider = UFDQNDecider(
  169. physics=physics,
  170. action_spec=action_spec,
  171. reward_params=reward_params,
  172. state_bounds=state_bounds,
  173. model_path=model_path,
  174. seed=0,
  175. )
  176. # 模型决策(不推进真实环境)
  177. decision = decider.decide(current_state)
  178. action_id = decision["action_id"]
  179. model_L_s = decision["L_s"]
  180. model_t_bw_s = decision["t_bw_s"]
  181. return action_id, model_L_s, model_t_bw_s
  182. # ==============================
  183. # 示例调用
  184. # ==============================
  185. if __name__ == "__main__":
  186. MODEL_PATH = UF_RL_ROOT / "longting" / "48h_dqn_model.zip"
  187. ENV_CONFIG_PATH = UF_RL_ROOT / "longting" / "env_config.yaml"
  188. TMP0 = 0.019 # 原始 TMP0
  189. q_UF = 300 # 进水流量
  190. temp = 20.0 #进水温度
  191. IS_TIMES = False # 新增指定变量,表示CEB间隔为时间控制/次数控制,T表示48次bw一次CEB,F表示48h一次CEB
  192. current_state = UFState(TMP=TMP0, q_UF=q_UF, temp=temp)
  193. config_loader = EnvConfigLoader(ENV_CONFIG_PATH)
  194. config_loader.validate_config()
  195. config_loader.print_config_summary()
  196. (
  197. uf_state_default, # UFState默认值
  198. phys_params, # UFPhysicsParams
  199. action_spec, # UFActionSpec
  200. reward_params, # UFRewardParams
  201. state_bounds # UFStateBounds
  202. ) = create_env_params_from_yaml(ENV_CONFIG_PATH)
  203. physics = build_physics(IS_TIMES, phys_params)
  204. action_id, model_L_s, model_t_bw_s = run_dqn_decide(
  205. model_path=MODEL_PATH,
  206. physics=physics,
  207. action_spec=action_spec,
  208. reward_params=reward_params,
  209. state_bounds=state_bounds,
  210. current_state=current_state,
  211. ) # 环境实例化,模型加载等功能放在UFDQNDecider类中
  212. current_L_s = 3800
  213. current_t_bw_s = 40
  214. model_prev_L_s = 4040
  215. model_prev_t_bw_s = 60
  216. L_s, t_bw_s = generate_plc_instructions(action_spec, current_L_s, current_t_bw_s, model_prev_L_s, model_prev_t_bw_s, model_L_s,
  217. model_t_bw_s) # 获取模型下发指令
  218. L_s = 4100
  219. t_bw_s = 96
  220. max_tmp_during_filtration = 0.050176 # 新增工厂数据接口:周期最高/最低跨膜压差,无工厂数据接入时传入None,calc_uf_cycle_metrics()自动获取模拟周期中的跨膜压差最值
  221. min_tmp_during_filtration = 0.012496
  222. execution_result = calc_uf_cycle_metrics(current_state, max_tmp_during_filtration, min_tmp_during_filtration, L_s, t_bw_s)
  223. print("\n===== 单步决策结果 =====")
  224. print(f"模型选择的动作: {action_id}")
  225. print(f"模型选择的L_s: {model_L_s} 秒, 模型选择的t_bw_s: {model_t_bw_s} 秒")
  226. print(f"指令下发的L_s: {L_s} 秒, 指令下发的t_bw_s: {t_bw_s} 秒")
  227. print(f"指令对应的反洗次数: {execution_result['k_bw_per_ceb']}")
  228. print(f"指令对应的吨水电耗: {execution_result['ton_water_energy_kWh_per_m3']}")
  229. print(f"指令对应的回收率: {execution_result['recovery']}")
  230. print(f"指令对应的日均产水时间: {execution_result['daily_prod_time_h']}")
  231. print(f"指令对应的最高渗透率: {execution_result['max_permeability']}")