2020-09-30 16:53:55 +02:00

210 lines
5.3 KiB
C++

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1997 - 1998
//
// File: utility.cpp
//
//--------------------------------------------------------------------------
//
// utility.cpp: utility computation
//
#include <basetsd.h>
#include <math.h>
#include "utility.h"
#include "infer.h"
MBNET_ENTROPIC_UTILITY :: MBNET_ENTROPIC_UTILITY ( GOBJMBN_INFER_ENGINE & inferEng )
: MBNET_NODE_RANKER( inferEng.Model() ),
_inferEng( inferEng ),
_propMgr( inferEng.Model() ),
_cHypo(0),
_cInfo(0)
{
_iLblHypo = _propMgr.ILblToUser( ESTDLBL_hypo );
_iLblInfo = _propMgr.ILblToUser( ESTDLBL_info );
_iLblProblem = _propMgr.ILblToUser( ESTDLBL_problem );
BuildWorkItems();
}
//
// Collect all informational, problem defining and hypothesis nodes
// into a structure with additional working data.
//
void MBNET_ENTROPIC_UTILITY :: BuildWorkItems ()
{
ZSREF zsrPropTypeLabel = _propMgr.ZsrPropType( ESTDP_label );
MODEL::MODELENUM mdlenum( Model() );
_dquwrk.clear();
_cHypo = 0;
_cInfo = 0;
UTILWORK uwDummy;
GELEMLNK * pgelm;
// Collect all the nodes into a pointer array. Three node labels
// are collected: info and probdef nodes (considered as info)
// and hypo nodes (considered as hypo).
while ( pgelm = mdlenum.PlnkelNext() )
{
if ( pgelm->EType() != GOBJMBN::EBNO_NODE )
continue;
// We only support discrete nodes for now.
DynCastThrow( pgelm, uwDummy._pgndd );
// See if this is an expansion (created) node
if ( uwDummy._pgndd->BFlag( EIBF_Expansion ) )
continue; // not a user-identifiable artifact; skip it
// See if it has a label
PROPMBN * propLbl = uwDummy._pgndd->LtProp().PFind( zsrPropTypeLabel );
if ( ! propLbl )
continue; // no label; skip it
uwDummy._iLbl = propLbl->Real();
if ( uwDummy._iLbl == _iLblHypo )
_cHypo++;
else
if ( uwDummy._iLbl == _iLblInfo || uwDummy._iLbl == _iLblProblem )
_cInfo++;
else
continue; // not a label of interest
// Initialize the other member variables
uwDummy._rUtil = 0.0;
uwDummy._iClamp = -1;
// Put the item on the work queue
_dquwrk.push_back( uwDummy );
}
}
REAL MBNET_ENTROPIC_UTILITY :: RComputeHypoGivenInfo (
UTILWORK & uwHypo,
UTILWORK & uwInfo )
{
assert( uwHypo._iLbl == _iLblHypo );
assert( uwInfo._iLbl != _iLblHypo );
// Clamped nodes are irrelevant
if ( uwHypo._iClamp >= 0 || uwInfo._iClamp >= 0 )
return 0.0;
REAL rUtilOfInfoForHypo = 0.0;
int cState = uwInfo._pgndd->CState();
int cStateHypo = uwHypo._pgndd->CState();
MDVCPD mdvhi;
REAL rp_h0 = uwHypo._dd[0]; // Probability of hypo node being normal
for ( int istInfo = 0; istInfo < cState; istInfo++ )
{
// Get belief of hypo node given info state
_inferEng.EnterEvidence( uwInfo._pgndd, CLAMP( true, istInfo, true ) );
_inferEng.GetBelief( uwHypo._pgndd, mdvhi );
REAL rp_h0xj = mdvhi[0]; // p(h0|xj)
REAL rLogSum = 0.0;
for ( int istHypo = 1; istHypo < cStateHypo; istHypo++ )
{
REAL rp_hi = uwHypo._dd[istHypo];
REAL rp_hixj = mdvhi[istHypo];
rLogSum += fabs( log(rp_hixj) - log(rp_h0xj) - log(rp_hi) + log(rp_h0) );
}
rUtilOfInfoForHypo += rLogSum * uwInfo._dd[istInfo];
}
// Clear evidence against info node
_inferEng.EnterEvidence( uwInfo._pgndd, CLAMP() );
return rUtilOfInfoForHypo;
}
DEFINEVP(UTILWORK);
void MBNET_ENTROPIC_UTILITY :: ComputeWorkItems()
{
CLAMP clamp;
VPUTILWORK vpuw; // Remember pointers to hypo items
// Get unconditional beliefs of all relevant (unclamped) nodes
for ( DQUTILWORK::iterator itdq = _dquwrk.begin();
itdq != _dquwrk.end();
itdq++ )
{
UTILWORK & ut = *itdq;
ut._rUtil = 0.0;
ut._iClamp = -1;
// Remember the indicies of the hypo nodes
if ( ut._iLbl == _iLblHypo )
vpuw.push_back( & (*itdq) );
// Get the current evidence for the node
_inferEng.GetEvidence( ut._pgndd, clamp );
// If node is unclamped,
if ( ! clamp.BActive() )
{
// get unconditional probs, else
_inferEng.GetBelief( ut._pgndd, ut._dd );
}
else
{
// remember clamped state (serves as marker)
ut._iClamp = clamp.Ist();
}
}
for ( itdq = _dquwrk.begin();
itdq != _dquwrk.end();
itdq++ )
{
UTILWORK & utInfo = *itdq;
if ( utInfo._iLbl == _iLblHypo )
continue;
utInfo._rUtil = 0.0;
for ( int ih = 0; ih < vpuw.size(); ih++ )
{
utInfo._rUtil += RComputeHypoGivenInfo( *vpuw[ih], utInfo );
}
}
}
void MBNET_ENTROPIC_UTILITY :: operator () ()
{
// Clear any old results
Clear();
if ( _cHypo == 0 || _cInfo == 0 )
return; // Nothing to do
// Compute all utilities
ComputeWorkItems();
// Sort the work queue by utility
sort( _dquwrk.begin(), _dquwrk.end() );
// Pour the information into the output work areas
_vzsrNodes.resize(_cInfo);
_vlrValues.resize(_cInfo);
int iInfo = 0;
for ( DQUTILWORK::reverse_iterator ritdq = _dquwrk.rbegin();
ritdq != _dquwrk.rend();
ritdq++ )
{
UTILWORK & ut = *ritdq;
if ( ut._iLbl == _iLblHypo )
continue;
_vzsrNodes[iInfo] = ut._pgndd->ZsrefName();
_vlrValues[iInfo++] = ut._rUtil;
}
assert( iInfo == _cInfo );
}