All Downloads are FREE. Search and download functionalities are using the official Maven repository.

hex.gam.MatrixFrameUtils.GenISplineGamOneColumn Maven / Gradle / Ivy

There is a newer version: 3.46.0.6
Show newest version
package hex.gam.MatrixFrameUtils;

import hex.gam.GAMModel;
import hex.genmodel.algos.gam.ISplines;
import water.MRTask;
import water.fvec.Chunk;
import water.fvec.Frame;
import water.fvec.NewChunk;
import water.util.ArrayUtils;

import static hex.gam.GamSplines.NBSplinesTypeIDerivative.genISPenaltyMatrix;

/**
 * Gamified one gam column at a time using I-spline.  See doc in JIRA: https://h2oai.atlassian.net/browse/PUBDEV-8398.
 * This one is different from AddISGamColumns because it generates the penalty matrix, zTranform if applicable and
 * perform scaling of the penalty matrix apart from performing gamification.
 */
public class GenISplineGamOneColumn extends MRTask {
    private final double[] _knots;      // knots without duplication
    private final int _order;
    double[] _maxAbsRowSum;             // store maximum row sum
    public double _s_scale;
    private final int _gamColNChunks;
    public double[][] _penaltyMat;      // store penalty matrix
    public final int _numBasis;
    public final int _totKnots;
    
    public GenISplineGamOneColumn(GAMModel.GAMParameters parm, double[] knots, int gamColIndex, Frame gamCol, 
                                  int nBasis, int totKnots) {
        _knots = knots;
        _order = parm._spline_orders_sorted[gamColIndex];
        _numBasis = nBasis > 0 ? nBasis : knots.length+_order-2;
        _totKnots = totKnots > 0 ? totKnots : knots.length+2*_order-2;
        _gamColNChunks = gamCol.vec(0).nChunks();
        _penaltyMat = genISPenaltyMatrix(knots, parm._spline_orders_sorted[gamColIndex]);
    }

    @Override
    public void map(Chunk[] chk, NewChunk[] newGamCols) {
        ISplines basisFuncs = new ISplines(_order, _knots);
        _maxAbsRowSum = new double[_gamColNChunks];
        double[] basisVals = new double[_numBasis]; // array to hold each gamified row
        int cIndex = chk[0].cidx();
        _maxAbsRowSum[cIndex] = Double.NEGATIVE_INFINITY;
        int chkRows = chk[0].len();
        for (int rowIndex=0; rowIndex < chkRows; rowIndex++) {
            double gamRowSum = 0.0;
            if (chk[1].atd(rowIndex) != 0) {
                double xval = chk[0].atd(rowIndex);
                if (Double.isNaN(xval)) {
                    for (int colIndex = 0; colIndex < _numBasis; colIndex++)
                        newGamCols[colIndex].addNum(Double.NaN);
                } else {
                    basisFuncs.gamifyVal(basisVals, xval);
                    // copy updates to the newChunk row
                    for (int colIndex = 0; colIndex < _numBasis; colIndex++) {
                        newGamCols[colIndex].addNum(basisVals[colIndex]);
                        gamRowSum += Math.abs(basisVals[colIndex]);
                    }
                    if (gamRowSum > _maxAbsRowSum[cIndex])
                        _maxAbsRowSum[cIndex] = gamRowSum;
                }
            } else {
                for (int colIndex = 0; colIndex < _numBasis; colIndex++)
                    newGamCols[colIndex].addNum(0.0);
            }
        }
    }


    public void reduce(GenISplineGamOneColumn other) {
        ArrayUtils.add(_maxAbsRowSum, other._maxAbsRowSum);
    }

    @Override
    public void postGlobal() {  // scale the _penalty function according to R
        double tempMaxValue = ArrayUtils.maxValue(_maxAbsRowSum);
        _s_scale = tempMaxValue*tempMaxValue/ArrayUtils.rNorm(_penaltyMat, 'i');
        if (Double.isFinite(_s_scale))
            ArrayUtils.mult(_penaltyMat, _s_scale);
        _s_scale = 1.0/ _s_scale;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy