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

hex.tree.CompressedTree Maven / Gradle / Ivy

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

import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.*;

import hex.genmodel.algos.tree.SharedTreeMojoModel;
import hex.genmodel.algos.tree.SharedTreeSubgraph;
import water.*;
import water.util.IcedBitSet;
import water.util.SB;

import static hex.genmodel.algos.tree.SharedTreeMojoModel.AuxInfo;

//---------------------------------------------------------------------------
// Note: this description seems to be out-of-date
//
// Highly compressed tree encoding:
//    tree: 1B nodeType, 2B colId, 4B splitVal, left-tree-size, left, right
//    nodeType: (from lsb):
//        2 bits (1,2) skip-tree-size-size,
//        2 bits (4,8) operator flag (0 --> <, 1 --> ==, 2 --> small (4B) group, 3 --> big (var size) group),
//        1 bit  ( 16) left leaf flag,
//        1 bit  ( 32) left leaf type flag (0: subtree, 1: small cat, 2: big cat, 3: float)
//        1 bit  ( 64) right leaf flag,
//        1 bit  (128) right leaf type flag (0: subtree, 1: small cat, 2: big cat, 3: float)
//    left, right: tree | prediction
//    prediction: 4 bytes of float (or 1 or 2 bytes of class prediction)
//

public class CompressedTree extends Keyed {

  private static final String KEY_PREFIX = "tree_";

  final byte [] _bits;
  final long _seed;

  public CompressedTree(byte[] bits, long seed, int tid, int cls) {
    super(makeTreeKey(tid, cls));
    _bits = bits;
    _seed = seed;
  }

  private CompressedTree(Key key, byte[] bits, long seed) {
    super(key);
    _bits = bits;
    _seed = seed;
  }

  public double score(final double row[], final String[][] domains) {
    return SharedTreeMojoModel.scoreTree(_bits, row, false, domains);
  }

  @Deprecated
  public String getDecisionPath(final double row[], final String[][] domains) {
    double d = SharedTreeMojoModel.scoreTree(_bits, row, true, domains);
    return SharedTreeMojoModel.getDecisionPath(d);
  }

  public  T getDecisionPath(final double row[], final String[][] domains, final SharedTreeMojoModel.DecisionPathTracker tr) {
    double d = SharedTreeMojoModel.scoreTree(_bits, row, true, domains);
    return SharedTreeMojoModel.getDecisionPath(d, tr);
  }

  public Map toAuxInfos() {
    return SharedTreeMojoModel.readAuxInfos(_bits);
  }

  public int findMaxNodeId() {
    return SharedTreeMojoModel.findMaxNodeId(_bits);
  }

  public CompressedTree updateLeafNodeWeights(double[] leafNodeWeights) {
    Map nodeIdToAuxInfo = SharedTreeMojoModel.readAuxInfos(_bits);
    List auxInfos = new ArrayList<>(nodeIdToAuxInfo.values());
    auxInfos.sort(Comparator.comparingInt(o -> -o.pid));
    for (AuxInfo auxInfo : auxInfos) {
      auxInfo.weightL = 0;
      auxInfo.weightR = 0;
    }
    for (AuxInfo auxInfo : auxInfos) {
      auxInfo.weightL += (float) leafNodeWeights[auxInfo.nidL];
      auxInfo.weightR += (float) leafNodeWeights[auxInfo.nidR];
      if (auxInfo.pid >= 0) {
        AuxInfo parentInfo = nodeIdToAuxInfo.get(auxInfo.pid);
        float nodeWeight = auxInfo.weightL + auxInfo.weightR;
        if (parentInfo.nidL == auxInfo.nid)
          parentInfo.weightL += nodeWeight;
        else
          parentInfo.weightR += nodeWeight;
      }
    }
    ByteBuffer bb = ByteBuffer.allocate(_bits.length).order(ByteOrder.nativeOrder());
    SharedTreeMojoModel.writeUpdatedAuxInfos(_bits, nodeIdToAuxInfo, bb);
    byte[] bits = bb.array();
    return new CompressedTree(_key, bits, _seed);
  }
  
  public boolean hasZeroWeight() {
    return SharedTreeMojoModel.readAuxInfos(_bits)
            .values()
            .stream()
            .anyMatch(auxInfo -> auxInfo.weightL == 0 || auxInfo.weightR == 0);
  }
  
  public SharedTreeSubgraph toSharedTreeSubgraph(final CompressedTree auxTreeInfo,
                                                 final String[] colNames, final String[][] domains) {
    TreeCoords tc = getTreeCoords();
    String treeName = SharedTreeMojoModel.treeName(tc._treeId, tc._clazz, domains[domains.length - 1]);
    return SharedTreeMojoModel.computeTreeGraph(tc._treeId, treeName, _bits, auxTreeInfo._bits, colNames, domains);
  }

  public Random rngForChunk(int cidx) {
    Random rand = new Random(_seed);
    for (int i = 0; i < cidx; i++) rand.nextLong();
    long seed = rand.nextLong();
    return new Random(seed);
  }

  public String toString(SharedTreeModel.SharedTreeOutput tm) {
    final String[] names = tm._names;
    final SB sb = new SB();
    new TreeVisitor(this) {
      @Override protected void pre(int col, float fcmp, IcedBitSet gcmp, int equal, int naSplitDirInt) {
        if (naSplitDirInt == DhnasdNaVsRest)
          sb.p("!Double.isNaN(" + sb.i().p(names[col]).p(")"));
        else if (naSplitDirInt == DhnasdNaLeft)
          sb.p("Double.isNaN(" + sb.i().p(names[col]).p(") || "));
        else if (equal==1)
          sb.p("!Double.isNaN(" + sb.i().p(names[col]).p(") && "));
        if (naSplitDirInt != DhnasdNaVsRest) {
          sb.i().p(names[col]).p(' ');
          if (equal == 0) sb.p("< ").pj(fcmp);
          else if (equal == 1) sb.p("!=").pj(fcmp);
          else sb.p("in ").p(gcmp);
        }
        sb.ii(1).nl();
      }
      @Override protected void post(int col, float fcmp, int equal) {
        sb.di(1);
      }
      @Override protected void leaf(float pred) {
        sb.i().p("return ").pj(pred).nl();
      }
    }.visit();
    return sb.toString();
  }

  public static Key makeTreeKey(int treeId, int clazz) {
    return Key.makeSystem("tree_" + treeId + "_" + clazz + "_" + Key.rand());
  }

  /**
   * Retrieves tree coordinates in the tree ensemble
   * @return instance of TreeCoord
   */
  TreeCoords getTreeCoords() {
    return TreeCoords.parseTreeCoords(_key);
  }

  @Override protected long checksum_impl() {
    throw new UnsupportedOperationException();
  }

  static class TreeCoords {
    int _treeId;
    int _clazz;

    private static TreeCoords parseTreeCoords(Key ctKey) {
      String key = ctKey.toString();
      int prefixIdx = key.indexOf(KEY_PREFIX);
      if (prefixIdx < 0)
        throw new IllegalStateException("Unexpected structure of a CompressedTree key=" + key);
      String[] keyParts = key.substring(prefixIdx + KEY_PREFIX.length()).split("_", 3);
      TreeCoords tc = new TreeCoords();
      tc._treeId = Integer.valueOf(keyParts[0]);
      tc._clazz = Integer.valueOf(keyParts[1]);
      return tc;
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy