From 301577a4e687a1b6eb9a2b5f433e674457128b52 Mon Sep 17 00:00:00 2001 From: Pranjal Tandon Date: Fri, 14 Sep 2018 23:47:01 +0530 Subject: [PATCH] Update sac.py --- sac.py | 54 ++++++++++++++++++++++++++++++++++++++++++++---------- 1 file changed, 44 insertions(+), 10 deletions(-) diff --git a/sac.py b/sac.py index 114880f..60ce7f9 100644 --- a/sac.py +++ b/sac.py @@ -58,28 +58,48 @@ class SAC(object): reward_batch = torch.FloatTensor(reward_batch) mask_batch = torch.FloatTensor(np.float32(mask_batch)) + reward_batch = reward_batch.unsqueeze(1) # reward_batch = [batch_size, 1] + mask_batch = mask_batch.unsqueeze(1) # mask_batch = [batch_size, 1] + expected_q1_value, expected_q2_value = self.critic(state_batch, action_batch) - new_action, log_prob, x_t, mean, log_std = self.policy.evaluate(state_batch, reparam=self.reparam) - reward_batch = reward_batch.unsqueeze(1) - mask_batch = mask_batch.unsqueeze(1) if self.deterministic == False: + """ + Including a separate function approximator for the soft value can stabilize training. + """ expected_value = self.value(state_batch) target_value = self.value_target(next_state_batch) - next_q_value = self.scale_R * reward_batch + mask_batch * self.gamma * target_value + next_q_value = self.scale_R * reward_batch + mask_batch * self.gamma * target_value # Reward Scale * r(st,at) - γV(target)(st+1)) else: + """ + There is no need in principle to include a separate function approximator for the state value. + We use a target critic network for deterministic policy and eradicate the value value network completely. + """ target_critic_1, target_critic_2 = self.critic_target(next_state_batch, new_action) target_critic = torch.min(target_critic_1, target_critic_2) - next_q_value = self.scale_R * reward_batch + mask_batch * self.gamma * target_critic - + next_q_value = self.scale_R * reward_batch + mask_batch * self.gamma * target_critic # Reward Scale * r(st,at) - γQ(target)(st+1) + + + """ + Soft Q-function parameters can be trained to minimize the soft Bellman residual + JQ = 𝔼(st,at)~D[0.5(Q1(st,at) - r(st,at) - γ(𝔼st+1~p[V(st+1)]))^2] + ∇JQ = ∇Q(st,at)(Q(st,at) - r(st,at) - γV(target)(st+1)) + """ q1_value_loss = self.soft_q_criterion(expected_q1_value, next_q_value.detach()) q2_value_loss = self.soft_q_criterion(expected_q2_value, next_q_value.detach()) q1_new, q2_new = self.critic(state_batch, new_action) expected_new_q_value = torch.min(q1_new, q2_new) if self.deterministic == False: + """ + Including a separate function approximator for the soft value can stabilize training and is convenient to + train simultaneously with the other networks + Update the V towards the min of two Q-functions in order to reduce overestimation bias from function approximation error. + JV = 𝔼st~D[0.5(V(st) - (𝔼at~π[Qmin(st,at) - log π(at|st)]))^2] + ∇JV = ∇V(st)(V(st) - Q(st,at) + logπ(at|st)) + """ next_value = expected_new_q_value - log_prob value_loss = self.value_criterion(expected_value, next_value.detach()) log_prob_target = expected_new_q_value - expected_value @@ -87,10 +107,18 @@ class SAC(object): log_prob_target = expected_new_q_value if self.reparam == True: + """ + Reparameterization trick is used to get a low variance estimator + f(εt;st) = action sampled from the policy + εt is an input noise vector, sampled from some fixed distribution + Jπ = 𝔼st∼D,εt∼N[logπ(f(εt;st)|st)−Q(st,f(εt;st))] + ∇Jπ =∇log π + ([∇at log π(at|st) − ∇at Q(st,at)])∇f(εt;st) + """ policy_loss = (log_prob - expected_new_q_value).mean() else: - policy_loss = (log_prob * (log_prob - log_prob_target).detach()).mean() + policy_loss = (log_prob * (log_prob - log_prob_target).detach()).mean() # likelihood ratio gradient estimator + # Regularization Loss mean_loss = 0.001 * mean.pow(2).mean() std_loss = 0.001 * log_std.pow(2).mean() @@ -112,13 +140,18 @@ class SAC(object): self.policy_optim.zero_grad() policy_loss.backward() self.policy_optim.step() - + + + """ + We update the target weights to match the current value function weights periodically + Update target parameter after every n(args.value_update) updates + """ if updates % self.value_update == 0 and self.deterministic == True: soft_update(self.critic_target, self.critic, self.tau) elif updates % self.value_update == 0 and self.deterministic == False: soft_update(self.value_target, self.value, self.tau) - + # Save model parameters def save_model(self, env_name, suffix="", actor_path=None, critic_path=None, value_path=None): if not os.path.exists('models/'): os.makedirs('models/') @@ -133,7 +166,8 @@ class SAC(object): torch.save(self.value.state_dict(), value_path) torch.save(self.policy.state_dict(), actor_path) torch.save(self.critic.state_dict(), critic_path) - + + # Load model parameters def load_model(self, actor_path, critic_path, value_path): print('Loading models from {}, {} and {}'.format(actor_path, critic_path, value_path)) if actor_path is not None: