How to write a custom Deterministic or Stochastic in pymc3 with theano.op?

前端 未结 1 326
暗喜
暗喜 2021-01-15 09:52

I\'m doing some pymc3 and I would like to create custom Stochastics, however there doesn\'t seem to be a lot documentation about how it\'s done. I know how to use the as_op

相关标签:
1条回答
  • 2021-01-15 10:28

    I realize this is a bit late now, but I thought I'd answer the question (rather vaguely) anyways.

    If you want to define a stochastic function (e.g. a probability distribution), then you need to do a couple of things:

    First, define a subclass of either Discrete (pymc3.distributions.Discrete) or Continuous, which has at least the method logp, which returns the log-likelihood of your stochastic. If you define this as a simple symbolic equation (x+1), I believe you do not need to take care of any gradients (but don't quote me on this; see the documentation about this). I'll get on to more complicated cases below. In the unfortunate case that you need to do anything more complex, as in your second example (pymc3 now has a skew normal distribution implemented, by the way), you need to define the operations required for it (used in the logp method) as a Theano Op. If you need no derivatives, then the as_op does the job, but as you said, gradients are kind of the idea of pymc3.

    This is where it gets complicated. If you want to use NUTS (or need gradients for whatever reason), then you need to implement your operation used in logp as a subclass of theano.gof.Op. Your new op class (let's call it just Op from now on) will need two or three methods at least. The first one defines inputs/outputs to the Op (check the Op documentation). The perform() method (or variants you might choose) is the one that does the operation you want (your R_forward function, for example). This can be done in pure python, if you so wish. The third method, grad(), is where you define the gradient of your perform()'s output wrt the inputs. The actual output to grad() is a bit different, but not a big deal.

    And it is in grad() that using Theano pays off. If you define your entire perform() in Theano, then it might be that you can easily use automatic differentiation (theano.tensor.grad or theano.tensor.jacobian) to do the work for you (see the example below). However, this is not necessarily going to be easy.

    In your second example, it would mean implementing your R_forward function in Theano, which could be complicated.

    Here I include a somewhat minimal example of an Op that I created while learning to do these things.

    def my_th_fun():
        """ Some needed auxiliary functions.
        """
        X = th.tensor.vector('X')
        SCALE = th.tensor.scalar('SCALE')
    
        X.tag.test_value = np.array([1,2,3,4])
        SCALE.tag.test_value = 5.
    
        Scale, upd_sm_X = th.scan(lambda x, scale: scale*(scale+ x),
                                   sequences=[X],
                                   outputs_info=[SCALE])
        fun_Scale = th.function(inputs=[X, SCALE], outputs=Scale)
        D_out_d_scale = th.tensor.grad(Scale[-1], SCALE)
        fun_d_out_d_scale = th.function([X, SCALE], D_out_d_scale)
        return Scale, fun_Scale, D_out_d_scale, fun_d_out_d_scale
    
    class myOp(th.gof.Op):
        """ Op subclass with a somewhat silly computation. It uses
        th.scan and th.tensor.grad is used to calculate the gradient
        automagically in the grad() method.
        """
        __props__ = ()
        itypes = [th.tensor.dscalar]
        otypes = [th.tensor.dvector]
        def __init__(self, *args, **kwargs):
            super(myOp, self).__init__(*args, **kwargs)
            self.base_dist = np.arange(1,5)
            (self.UPD_scale, self.fun_scale, 
             self.D_out_d_scale, self.fun_d_out_d_scale)= my_th_fun()
    
        def perform(self, node, inputs, outputs):
            scale = inputs[0]
            updated_scale = self.fun_scale(self.base_dist, scale)
            out1 = self.base_dist[0:2].sum()
            out2 = self.base_dist[2:4].sum()
            maxout = np.max([out1, out2])
            exp_out1 = np.exp(updated_scale[-1]*(out1-maxout))
            exp_out2 = np.exp(updated_scale[-1]*(out2-maxout))
            norm_const = exp_out1 + exp_out2
            outputs[0][0] = np.array([exp_out1/norm_const, exp_out2/norm_const])
    
        def grad(self, inputs, output_gradients): #working!
            """ Calculates the gradient of the output of the Op wrt
            to the input. As a simple example, the input is scalar.
    
            Notice how the output is actually the gradient multiplied
            by the output_gradients, which is an input provided by
            theano when calculating gradients.
            """
            scale = inputs[0]
            X = th.tensor.as_tensor(self.base_dist)
            # Do I need to recalculate all this or can I assume that perform() has
            # always been called before grad() and thus can take it from there?
            # In any case, this is a small enough example to recalculate quickly:
            all_scale, _ = th.scan(lambda x, scale_1: scale_1*(scale_1+ x),
                                   sequences=[X],
                                   outputs_info=[scale])
            updated_scale = all_scale[-1]
    
            out1 = self.base_dist[0:1].sum()
            out2 = self.base_dist[2:3].sum()
            maxout = np.max([out1, out2])
    
            exp_out1 = th.tensor.exp(updated_scale*(out1 - maxout))
            exp_out2 = th.tensor.exp(updated_scale*(out2 - maxout))
            norm_const = exp_out1 + exp_out2
    
            d_S_d_scale = th.theano.grad(all_scale[-1], scale)
            Jac1 = (-(out1-out2)*d_S_d_scale*
                    th.tensor.exp(updated_scale*(out1+out2 - 2*maxout))/(norm_const**2))
            Jac2 = -Jac1
            return Jac1*output_gradients[0][0]+ Jac2*output_gradients[0][1],
    

    This Op can then be used inside the logp() method of a stochastic in pymc3:

    import pymc3 as pm
    
    class myDist(pm.distributions.Discrete):
        def __init__(self, invT, *args, **kwargs):
            super(myDist, self).__init__(*args, **kwargs)
            self.invT = invT
            self.myOp = myOp()
        def logp(self, value):
            return self.myOp(self.invT)[value]
    

    I hope it helps any (hopeless) pymc3/theano newbie out there.

    0 讨论(0)
提交回复
热议问题