mirror of
https://github.com/wassname/pytorch-soft-actor-critic.git
synced 2026-06-27 16:46:28 +08:00
128 lines
5.3 KiB
Python
128 lines
5.3 KiB
Python
import sys
|
|
import os
|
|
import numpy as np
|
|
import torch
|
|
import torch.nn as nn
|
|
from torch.optim import Adam
|
|
from torch.distributions import MultivariateNormal
|
|
from utils import soft_update, hard_update
|
|
from model import GaussianMixturePolicy, GaussianPolicy, QNetwork, ValueNetwork
|
|
|
|
|
|
class SAC(object):
|
|
def __init__(self, num_inputs, action_space, args):
|
|
|
|
self.num_inputs = num_inputs
|
|
self.action_space = action_space.shape[0]
|
|
self.gamma = args.gamma
|
|
self.tau = args.tau
|
|
self.k = args.k
|
|
self.scale_R = args.scale_R
|
|
self.algo = args.algo
|
|
self.reparam = args.reparam
|
|
|
|
if args.algo == "SAC":
|
|
self.policy = GaussianPolicy(self.num_inputs, self.action_space, args.hidden_size)
|
|
self.policy_optim = Adam(self.policy.parameters(), lr=3e-4)
|
|
else:
|
|
self.policy = GaussianMixturePolicy(self.num_inputs, self.action_space, args.hidden_size, self.k)
|
|
self.policy_optim = Adam(self.policy.parameters(), lr=3e-4)
|
|
|
|
self.critic = QNetwork(self.num_inputs, self.action_space, args.hidden_size)
|
|
self.critic_optim = Adam(self.critic.parameters(), lr=3e-4)
|
|
|
|
self.value = ValueNetwork(self.num_inputs, args.hidden_size)
|
|
self.value_target = ValueNetwork(self.num_inputs, args.hidden_size)
|
|
self.value_optim = Adam(self.value.parameters(), lr=3e-4)
|
|
|
|
self.value_criterion = nn.MSELoss()
|
|
self.soft_q_criterion = nn.MSELoss()
|
|
self.action_prior = "uniform"
|
|
# Make sure target is with the same weight
|
|
hard_update(self.value_target, self.value)
|
|
|
|
def select_action(self, state):
|
|
action = self.policy.get_action(state)
|
|
return action
|
|
|
|
|
|
def update_parameters(self, state_batch, action_batch, reward_batch, next_state_batch, mask_batch, step):
|
|
state_batch = torch.FloatTensor(state_batch)
|
|
next_state_batch = torch.FloatTensor(next_state_batch)
|
|
action_batch = torch.FloatTensor(action_batch)
|
|
reward_batch = torch.FloatTensor(reward_batch)
|
|
mask_batch = torch.FloatTensor(np.float32(mask_batch))
|
|
|
|
expected_q_value = self.critic(state_batch, action_batch)
|
|
expected_value = self.value(state_batch)
|
|
|
|
new_action, log_prob, x_t, mean, log_std = self.policy.evaluate(state_batch, reparam=self.reparam)
|
|
if self.action_prior == "normal":
|
|
act = new_action
|
|
act = act.size()
|
|
policy_prior = MultivariateNormal(torch.zeros(act[-1]), torch.eye(act[-1]))
|
|
policy_prior_log_probs = policy_prior.log_prob(new_action)
|
|
policy_prior_log_probs = policy_prior_log_probs.unsqueeze(1)
|
|
else:
|
|
policy_prior_log_probs = 0.0
|
|
|
|
target_value = self.value_target(next_state_batch)
|
|
reward_batch = reward_batch.unsqueeze(1)
|
|
mask_batch = mask_batch.unsqueeze(1)
|
|
next_q_value = self.scale_R * reward_batch + mask_batch * self.gamma * target_value
|
|
q_value_loss = self.soft_q_criterion(expected_q_value, next_q_value.detach())
|
|
|
|
expected_new_q_value = self.critic(state_batch, new_action)
|
|
next_value = expected_new_q_value - log_prob + policy_prior_log_probs
|
|
value_loss = self.value_criterion(expected_value, next_value.detach())
|
|
|
|
log_prob_target = expected_new_q_value - expected_value
|
|
if self.reparam == True and self.algo == "SAC":
|
|
policy_loss = (log_prob - expected_new_q_value).mean()
|
|
else:
|
|
policy_loss = (log_prob * (log_prob - policy_prior_log_probs - log_prob_target).detach()).mean()
|
|
|
|
mean_loss = 0.001 * mean.pow(2).mean()
|
|
std_loss = 0.001 * log_std.pow(2).mean()
|
|
x_t_loss = 0.0 * x_t.pow(2).sum(1).mean()
|
|
|
|
policy_loss += mean_loss + std_loss + x_t_loss
|
|
|
|
self.critic_optim.zero_grad()
|
|
q_value_loss.backward()
|
|
self.critic_optim.step()
|
|
|
|
self.value_optim.zero_grad()
|
|
value_loss.backward()
|
|
self.value_optim.step()
|
|
|
|
self.policy_optim.zero_grad()
|
|
policy_loss.backward()
|
|
self.policy_optim.step()
|
|
|
|
soft_update(self.value_target, self.value, self.tau)
|
|
|
|
|
|
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/')
|
|
|
|
if actor_path is None:
|
|
actor_path = "models/sac_actor_{}_{}".format(env_name, suffix)
|
|
if critic_path is None:
|
|
critic_path = "models/sac_critic_{}_{}".format(env_name, suffix)
|
|
if value_path is None:
|
|
value_path = "models/sac_value_{}_{}".format(env_name, suffix)
|
|
print('Saving models to {}, {} and {}'.format(actor_path, critic_path, value_path))
|
|
torch.save(self.value.state_dict(), value_path)
|
|
torch.save(self.policy.state_dict(), actor_path)
|
|
torch.save(self.critic.state_dict(), critic_path)
|
|
|
|
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:
|
|
self.policy.load_state_dict(torch.load(actor_path))
|
|
if critic_path is not None:
|
|
self.critic.load_state_dict(torch.load(critic_path))
|
|
if value_path is not None:
|
|
self.value.load_state_dict(torch.load(value_path)) |