You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
225 lines
7.9 KiB
225 lines
7.9 KiB
/*
|
|
Copyright 2005-2014 Intel Corporation. All Rights Reserved.
|
|
|
|
This file is part of Threading Building Blocks.
|
|
|
|
Threading Building Blocks is free software; you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License
|
|
version 2 as published by the Free Software Foundation.
|
|
|
|
Threading Building Blocks is distributed in the hope that it will be
|
|
useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
|
of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Threading Building Blocks; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
As a special exception, you may use this file as part of a free software
|
|
library without restriction. Specifically, if other files instantiate
|
|
templates or use macros or inline functions from this file, or you compile
|
|
this file and link it with other files to produce an executable, this
|
|
file does not by itself cause the resulting executable to be covered by
|
|
the GNU General Public License. This exception does not however
|
|
invalidate any other reasons why the executable file might be covered by
|
|
the GNU General Public License.
|
|
*/
|
|
|
|
//
|
|
// Self-organizing map in TBB flow::graph
|
|
//
|
|
// we will do a color map (the simple example.)
|
|
//
|
|
// serial algorithm
|
|
//
|
|
// initialize map with vectors (could be random, gradient, or something else)
|
|
// for some number of iterations
|
|
// update radius r, weight of change L
|
|
// for each example V
|
|
// find the best matching unit
|
|
// for each part of map within radius of BMU W
|
|
// update vector: W(t+1) = W(t) + w(dist)*L*(V - W(t))
|
|
|
|
#include "som.h"
|
|
#include "tbb/task.h"
|
|
|
|
std::ostream& operator<<( std::ostream &out, const SOM_element &s) {
|
|
out << "(";
|
|
for(int i=0;i<(int)s.w.size();++i) {
|
|
out << s.w[i];
|
|
if(i < (int)s.w.size()-1) {
|
|
out << ",";
|
|
}
|
|
}
|
|
out << ")";
|
|
return out;
|
|
}
|
|
|
|
void remark_SOM_element(const SOM_element &s) {
|
|
printf("(");
|
|
for(int i=0;i<(int)s.w.size();++i) {
|
|
printf("%g",s.w[i]);
|
|
if(i < (int)s.w.size()-1) {
|
|
printf(",");
|
|
}
|
|
}
|
|
printf(")");
|
|
}
|
|
|
|
std::ostream& operator<<( std::ostream &out, const search_result_type &s) {
|
|
out << "<";
|
|
out << get<RADIUS>(s);
|
|
out << ", " << get<XV>(s);
|
|
out << ", ";
|
|
out << get<YV>(s);
|
|
out << ">";
|
|
return out;
|
|
}
|
|
|
|
void remark_search_result_type(const search_result_type &s) {
|
|
printf("<%g,%d,%d>", get<RADIUS>(s), get<XV>(s), get<YV>(s));
|
|
}
|
|
|
|
double
|
|
randval( double lowlimit, double highlimit) {
|
|
return double(rand()) / double(RAND_MAX) * (highlimit - lowlimit) + lowlimit;
|
|
}
|
|
|
|
void
|
|
find_data_ranges(teaching_vector_type &teaching, SOM_element &max_range, SOM_element &min_range ) {
|
|
if(teaching.size() == 0) return;
|
|
max_range = min_range = teaching[0];
|
|
for(int i = 1; i < (int)teaching.size(); ++i) {
|
|
max_range.elementwise_max(teaching[i]);
|
|
min_range.elementwise_min(teaching[i]);
|
|
}
|
|
}
|
|
|
|
void add_fraction_of_difference( SOM_element &to, SOM_element const &from, double frac) {
|
|
for(int i = 0; i < (int)from.size(); ++i) {
|
|
to[i] += frac*(from[i] - to[i]);
|
|
}
|
|
}
|
|
|
|
double
|
|
distance_squared(SOM_element x, SOM_element y) {
|
|
double rval = 0.0; for(int i=0;i<(int)x.size();++i) {
|
|
double diff = x[i] - y[i];
|
|
rval += diff*diff;
|
|
}
|
|
return rval;
|
|
}
|
|
|
|
void SOMap::initialize(InitializeType it, SOM_element &max_range, SOM_element &min_range) {
|
|
for(int x = 0; x < xMax; ++x) {
|
|
for(int y = 0; y < yMax; ++y) {
|
|
for( int i = 0; i < (int)max_range.size(); ++i) {
|
|
if(it == InitializeRandom) {
|
|
my_map[x][y][i] = (randval(min_range[i], max_range[i]));
|
|
}
|
|
else if(it == InitializeGradient) {
|
|
my_map[x][y][i] = ((double)(x+y)/(xMax+yMax)*(max_range[i]-min_range[i]) + min_range[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// subsquare [low,high)
|
|
double
|
|
SOMap::BMU_range( const SOM_element &s, int &xval, int &yval, subsquare_type &r) {
|
|
double min_distance_squared = DBL_MAX;
|
|
task &my_task = task::self();
|
|
int min_x = -1;
|
|
int min_y = -1;
|
|
for(int x = r.rows().begin(); x != r.rows().end(); ++x) {
|
|
for( int y = r.cols().begin(); y != r.cols().end(); ++y) {
|
|
double dist = distance_squared(s,my_map[x][y]);
|
|
if(dist < min_distance_squared) {
|
|
min_distance_squared = dist;
|
|
min_x = x;
|
|
min_y = y;
|
|
}
|
|
if(cancel_test && my_task.is_cancelled()) {
|
|
xval = r.rows().begin();
|
|
yval = r.cols().begin();
|
|
return DBL_MAX;
|
|
}
|
|
}
|
|
}
|
|
xval = min_x;
|
|
yval = min_y;
|
|
return sqrt(min_distance_squared);
|
|
}
|
|
|
|
void
|
|
SOMap::epoch_update_range( SOM_element const &s, int epoch, int min_x, int min_y, double radius, double learning_rate, blocked_range<int> &r) {
|
|
int min_xiter = (int)((double)min_x - radius);
|
|
if(min_xiter < 0) min_xiter = 0;
|
|
int max_xiter = (int)((double)min_x + radius);
|
|
if(max_xiter > (int)my_map.size()-1) max_xiter = (int)my_map.size()-1;
|
|
for(int xx = r.begin(); xx <= r.end(); ++xx) {
|
|
double xrsq = (xx-min_x)*(xx-min_x);
|
|
double ysq = radius*radius - xrsq; // max extent of y influence
|
|
double yd;
|
|
if(ysq > 0) {
|
|
yd = sqrt(ysq);
|
|
int lb = (int)(min_y - yd);
|
|
int ub = (int)(min_y + yd);
|
|
for(int yy = lb; yy < ub; ++yy) {
|
|
if(yy >= 0 && yy < (int)my_map[xx].size()) {
|
|
// [xx, yy] is in the range of the update.
|
|
double my_rsq = xrsq + (yy-min_y)*(yy-min_y); // distance from BMU squared
|
|
double theta = exp(-(radius*radius) /(2.0* my_rsq));
|
|
add_fraction_of_difference(my_map[xx][yy], s, theta * learning_rate);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SOMap::teach(teaching_vector_type &in) {
|
|
for(int i = 0; i < nPasses; ++i ) {
|
|
int j = (int)(randval(0, (double)in.size())); // this won't be reproducible.
|
|
if(j == in.size()) --j;
|
|
|
|
int min_x = -1;
|
|
int min_y = -1;
|
|
subsquare_type br2(0, (int)my_map.size(), 1, 0, (int)my_map[0].size(), 1);
|
|
(void) BMU_range(in[j],min_x, min_y, br2); // just need min_x, min_y
|
|
// radius of interest
|
|
double radius = max_radius * exp(-(double)i*radius_decay_rate);
|
|
// update circle is min_xiter to max_xiter inclusive.
|
|
double learning_rate = max_learning_rate * exp( -(double)i * learning_decay_rate);
|
|
epoch_update(in[j], i, min_x, min_y, radius, learning_rate);
|
|
}
|
|
}
|
|
|
|
void SOMap::debug_output() {
|
|
printf("SOMap:\n");
|
|
for(int i = 0; i < (int)(this->my_map.size()); ++i) {
|
|
for(int j = 0; j < (int)(this->my_map[i].size()); ++j) {
|
|
printf( "map[%d, %d] == ", i, j );
|
|
remark_SOM_element( this->my_map[i][j] );
|
|
printf("\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
#define RED 0
|
|
#define GREEN 1
|
|
#define BLUE 2
|
|
|
|
void readInputData() {
|
|
my_teaching.push_back(SOM_element());
|
|
my_teaching.push_back(SOM_element());
|
|
my_teaching.push_back(SOM_element());
|
|
my_teaching.push_back(SOM_element());
|
|
my_teaching.push_back(SOM_element());
|
|
my_teaching[0][RED] = 1.0; my_teaching[0][GREEN] = 0.0; my_teaching[0][BLUE] = 0.0;
|
|
my_teaching[1][RED] = 0.0; my_teaching[1][GREEN] = 1.0; my_teaching[1][BLUE] = 0.0;
|
|
my_teaching[2][RED] = 0.0; my_teaching[2][GREEN] = 0.0; my_teaching[2][BLUE] = 1.0;
|
|
my_teaching[3][RED] = 0.3; my_teaching[3][GREEN] = 0.3; my_teaching[3][BLUE] = 0.0;
|
|
my_teaching[4][RED] = 0.5; my_teaching[4][GREEN] = 0.5; my_teaching[4][BLUE] = 0.9;
|
|
}
|