Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in
Toggle navigation
D
dlib
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
钟尚武
dlib
Commits
46864890
Commit
46864890
authored
Aug 28, 2011
by
Davis King
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Added the find_map_nmplp() function for performing approximate
MAP inference.
parent
f7c9763a
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
338 additions
and
0 deletions
+338
-0
optimization.h
dlib/optimization.h
+1
-0
find_map_nmplp.h
dlib/optimization/find_map_nmplp.h
+194
-0
find_map_nmplp_abstract.h
dlib/optimization/find_map_nmplp_abstract.h
+143
-0
No files found.
dlib/optimization.h
View file @
46864890
...
...
@@ -13,6 +13,7 @@
#include "optimization/optimization_least_squares.h"
#include "optimization/max_cost_assignment.h"
#include "optimization/max_sum_submatrix.h"
#include "optimization/find_map_nmplp.h"
#endif // DLIB_OPTIMIZATIOn_HEADER
...
...
dlib/optimization/find_map_nmplp.h
0 → 100644
View file @
46864890
// Copyright (C) 2011 Davis E. King (davis@dlib.net)
// License: Boost Software License See LICENSE.txt for the full license.
#ifndef DLIB_FIND_MAP_nMPLP_H__
#define DLIB_FIND_MAP_nMPLP_H__
#include "find_map_nmplp_abstract.h"
#include <vector>
#include <map>
#include "../matrix.h"
namespace
dlib
{
// ----------------------------------------------------------------------------------------
template
<
typename
map_problem
>
void
find_map_nmplp
(
const
map_problem
&
prob
,
std
::
vector
<
unsigned
long
>&
map_assignment
,
unsigned
long
max_iter
,
double
eps
)
{
/*
This function is an implementation of the NMPLP algorithm introduced in the
following paper:
Fixing Max-Product: Convergent Message Passing Algorithms for MAP LP-Relaxations
by Amir Globerson Tommi Jaakkola
In particular, see the pseudocode in Figure 1. The code in this function
follows what is described there.
*/
typedef
typename
map_problem
::
node_iterator
node_iterator
;
typedef
typename
map_problem
::
neighbor_iterator
neighbor_iterator
;
map_assignment
.
resize
(
prob
.
number_of_nodes
());
if
(
prob
.
number_of_nodes
()
==
0
)
return
;
std
::
vector
<
double
>
gamma_elements
;
gamma_elements
.
reserve
(
prob
.
number_of_nodes
()
*
prob
.
num_states
(
prob
.
begin
())
*
3
);
std
::
map
<
std
::
pair
<
unsigned
long
,
unsigned
long
>
,
unsigned
long
>
gamma_idx
;
// initialize gamma according to the initialization instructions at top of Figure 1
for
(
node_iterator
i
=
prob
.
begin
();
i
!=
prob
.
end
();
++
i
)
{
const
unsigned
long
id_i
=
prob
.
node_id
(
i
);
for
(
neighbor_iterator
j
=
prob
.
begin
(
i
);
j
!=
prob
.
end
(
i
);
++
j
)
{
const
unsigned
long
id_j
=
prob
.
node_id
(
j
);
gamma_idx
[
std
::
make_pair
(
id_i
,
id_j
)]
=
gamma_elements
.
size
();
const
unsigned
long
num_states_xj
=
prob
.
num_states
(
j
);
for
(
unsigned
long
xj
=
0
;
xj
<
num_states_xj
;
++
xj
)
{
const
unsigned
long
num_states_xi
=
prob
.
num_states
(
i
);
double
best_val
=
-
std
::
numeric_limits
<
double
>::
infinity
();
for
(
unsigned
long
xi
=
0
;
xi
<
num_states_xi
;
++
xi
)
{
double
val
=
prob
.
factor_value
(
i
,
j
,
xi
,
xj
);
double
sum_temp
=
0
;
for
(
neighbor_iterator
k
=
prob
.
begin
(
i
);
k
!=
prob
.
end
(
i
);
++
k
)
{
if
(
j
==
k
)
continue
;
double
max_val
=
-
std
::
numeric_limits
<
double
>::
infinity
();
for
(
unsigned
long
xk
=
0
;
xk
<
prob
.
num_states
(
k
);
++
xk
)
{
double
temp
=
prob
.
factor_value
(
k
,
i
,
xk
,
xi
);
if
(
temp
>
max_val
)
max_val
=
temp
;
}
sum_temp
+=
max_val
;
}
val
+=
0
.
5
*
sum_temp
;
if
(
val
>
best_val
)
best_val
=
val
;
}
gamma_elements
.
push_back
(
best_val
);
}
}
}
double
max_change
=
eps
+
1
;
// Now do the main body of the optimization.
for
(
unsigned
long
iter
=
0
;
iter
<
max_iter
&&
max_change
>
eps
;
++
iter
)
{
max_change
=
-
std
::
numeric_limits
<
double
>::
infinity
();
for
(
node_iterator
i
=
prob
.
begin
();
i
!=
prob
.
end
();
++
i
)
{
const
unsigned
long
id_i
=
prob
.
node_id
(
i
);
for
(
neighbor_iterator
j
=
prob
.
begin
(
i
);
j
!=
prob
.
end
(
i
);
++
j
)
{
const
unsigned
long
id_j
=
prob
.
node_id
(
j
);
double
*
const
gamma_ji
=
&
gamma_elements
[
gamma_idx
[
std
::
make_pair
(
id_j
,
id_i
)]];
double
*
const
gamma_ij
=
&
gamma_elements
[
gamma_idx
[
std
::
make_pair
(
id_i
,
id_j
)]];
const
unsigned
long
num_states_xj
=
prob
.
num_states
(
j
);
for
(
unsigned
long
xj
=
0
;
xj
<
num_states_xj
;
++
xj
)
{
const
unsigned
long
num_states_xi
=
prob
.
num_states
(
i
);
double
best_val
=
-
std
::
numeric_limits
<
double
>::
infinity
();
for
(
unsigned
long
xi
=
0
;
xi
<
num_states_xi
;
++
xi
)
{
double
val
=
prob
.
factor_value
(
i
,
j
,
xi
,
xj
)
-
gamma_ji
[
xi
];
double
sum_temp
=
0
;
int
num_neighbors
=
0
;
for
(
neighbor_iterator
k
=
prob
.
begin
(
i
);
k
!=
prob
.
end
(
i
);
++
k
)
{
const
unsigned
long
id_k
=
prob
.
node_id
(
k
);
++
num_neighbors
;
const
double
*
const
gamma_ki
=
&
gamma_elements
[
gamma_idx
[
std
::
make_pair
(
id_k
,
id_i
)]];
sum_temp
+=
gamma_ki
[
xi
];
}
val
+=
2
.
0
/
(
num_neighbors
+
1
.
0
)
*
sum_temp
;
if
(
val
>
best_val
)
best_val
=
val
;
}
if
(
std
::
abs
(
gamma_ij
[
xj
]
-
best_val
)
>
max_change
)
max_change
=
std
::
abs
(
gamma_ij
[
xj
]
-
best_val
);
gamma_ij
[
xj
]
=
best_val
;
}
}
}
}
// now decode the "beliefs"
std
::
vector
<
double
>
b
;
for
(
node_iterator
i
=
prob
.
begin
();
i
!=
prob
.
end
();
++
i
)
{
const
unsigned
long
id_i
=
prob
.
node_id
(
i
);
b
.
assign
(
prob
.
num_states
(
i
),
0
);
for
(
neighbor_iterator
k
=
prob
.
begin
(
i
);
k
!=
prob
.
end
(
i
);
++
k
)
{
const
unsigned
long
id_k
=
prob
.
node_id
(
k
);
for
(
unsigned
long
xi
=
0
;
xi
<
b
.
size
();
++
xi
)
{
const
double
*
const
gamma_ki
=
&
gamma_elements
[
gamma_idx
[
std
::
make_pair
(
id_k
,
id_i
)]];
b
[
xi
]
+=
gamma_ki
[
xi
];
}
}
map_assignment
[
id_i
]
=
index_of_max
(
vector_to_matrix
(
b
));
}
}
// ----------------------------------------------------------------------------------------
}
#endif // DLIB_FIND_MAP_nMPLP_H__
dlib/optimization/find_map_nmplp_abstract.h
0 → 100644
View file @
46864890
// Copyright (C) 2011 Davis E. King (davis@dlib.net)
// License: Boost Software License See LICENSE.txt for the full license.
#undef DLIB_FIND_MAP_nMPLP_ABSTRACT_H__
#ifdef DLIB_FIND_MAP_nMPLP_ABSTRACT_H__
#include <vector>
namespace
dlib
{
// ----------------------------------------------------------------------------------------
class
map_problem
{
/*!
WHAT THIS OBJECT REPRESENTS
!*/
public
:
class
node_iterator
{
public
:
node_iterator
();
bool
operator
==
(
const
node_iterator
&
item
)
const
;
bool
operator
!=
(
const
node_iterator
&
item
)
const
;
node_iterator
&
operator
++
();
};
class
neighbor_iterator
{
neighbor_iterator
();
bool
operator
==
(
const
neighbor_iterator
&
item
)
const
;
bool
operator
!=
(
const
neighbor_iterator
&
item
)
const
;
neighbor_iterator
&
operator
++
();
};
unsigned
long
number_of_nodes
(
)
const
;
/*!
ensures
- returns the number of nodes in the factor graph. Or in other words,
returns the number of variables in the MAP problem.
!*/
node_iterator
begin
(
)
const
;
node_iterator
end
(
)
const
;
neighbor_iterator
begin
(
const
node_iterator
&
it
)
const
;
neighbor_iterator
begin
(
const
neighbor_iterator
&
it
)
const
;
neighbor_iterator
end
(
const
node_iterator
&
it
)
const
;
neighbor_iterator
end
(
const
neighbor_iterator
&
it
)
const
;
unsigned
long
node_id
(
const
node_iterator
&
it
)
const
;
/*!
requires
- it == a valid iterator
ensures
- returns a number ID such that:
- 0 <= ID < number_of_nodes()
- ID == a number which uniquely identifies the node pointed to by it.
!*/
unsigned
long
node_id
(
const
neighbor_iterator
&
it
)
const
;
/*!
requires
- it == a valid iterator
ensures
- returns a number ID such that:
- 0 <= ID < number_of_nodes()
- ID == a number which uniquely identifies the node pointed to by it.
!*/
unsigned
long
num_states
(
const
node_iterator
&
it
)
const
;
unsigned
long
num_states
(
const
neighbor_iterator
&
it
)
const
;
double
factor_value
(
const
node_iterator
&
it1
,
const
node_iterator
&
it2
,
unsigned
long
s1
,
unsigned
long
s2
)
const
;
double
factor_value
(
const
neighbor_iterator
&
it1
,
const
node_iterator
&
it2
,
unsigned
long
s1
,
unsigned
long
s2
)
const
;
double
factor_value
(
const
node_iterator
&
it1
,
const
neighbor_iterator
&
it2
,
unsigned
long
s1
,
unsigned
long
s2
)
const
;
double
factor_value
(
const
neighbor_iterator
&
it1
,
const
neighbor_iterator
&
it2
,
unsigned
long
s1
,
unsigned
long
s2
)
const
;
};
// ----------------------------------------------------------------------------------------
template
<
typename
map_problem
>
void
find_map_nmplp
(
const
map_problem
&
prob
,
std
::
vector
<
unsigned
long
>&
map_assignment
,
unsigned
long
max_iter
,
double
eps
);
/*!
requires
- for all valid i: prob.num_states(i) >= 2
- map_problem == an object with an interface compatible with the map_problem
object defined at the top of this file.
ensures
- #map_assignment.size() == prob.number_of_nodes()
- for all valid i:
- #map_assignment[prob.node_id(i)] < prob.num_states(i)
- #map_assignment[prob.node_id(i)] == approximate MAP assignment for node i.
- This function is an implementation of the NMPLP algorithm introduced in the
following paper:
Fixing Max-Product: Convergent Message Passing Algorithms for MAP LP-Relaxations
by Amir Globerson Tommi Jaakkola
!*/
// ----------------------------------------------------------------------------------------
}
#endif // DLIB_FIND_MAP_nMPLP_H__
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment