2010-11-24 02:34:14 +08:00
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
// File: sorthelper.h
|
|
|
|
// Description: Generic sort and maxfinding class.
|
|
|
|
// Author: Ray Smith
|
|
|
|
// Created: Thu May 20 17:48:21 PDT 2010
|
|
|
|
//
|
|
|
|
// (C) Copyright 2010, Google Inc.
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#ifndef TESSERACT_CCUTIL_SORTHELPER_H_
|
|
|
|
#define TESSERACT_CCUTIL_SORTHELPER_H_
|
|
|
|
|
2018-05-20 05:52:04 +08:00
|
|
|
#include <cstdlib>
|
2010-11-24 02:34:14 +08:00
|
|
|
#include "genericvector.h"
|
|
|
|
|
|
|
|
// Generic class to provide functions based on a <value,count> pair.
|
|
|
|
// T is the value type.
|
|
|
|
// The class keeps a count of each value and can return the most frequent
|
|
|
|
// value or a sorted array of the values with counts.
|
|
|
|
// Note that this class uses linear search for adding. It is better
|
|
|
|
// to use the STATS class to get the mode of a large number of values
|
|
|
|
// in a small space. SortHelper is better to get the mode of a small number
|
|
|
|
// of values from a large space.
|
|
|
|
// T must have a copy constructor.
|
|
|
|
template <typename T>
|
|
|
|
class SortHelper {
|
|
|
|
public:
|
|
|
|
// Simple pair class to hold the values and counts.
|
|
|
|
template<typename PairT> struct SortPair {
|
|
|
|
PairT value;
|
|
|
|
int count;
|
|
|
|
};
|
|
|
|
// qsort function to sort by decreasing count.
|
|
|
|
static int SortPairsByCount(const void* v1, const void* v2) {
|
2019-03-26 14:55:08 +08:00
|
|
|
const auto* p1 = static_cast<const SortPair<T>*>(v1);
|
|
|
|
const auto* p2 = static_cast<const SortPair<T>*>(v2);
|
2010-11-24 02:34:14 +08:00
|
|
|
return p2->count - p1->count;
|
|
|
|
}
|
|
|
|
// qsort function to sort by decreasing value.
|
|
|
|
static int SortPairsByValue(const void* v1, const void* v2) {
|
2019-03-26 14:55:08 +08:00
|
|
|
const auto* p1 = static_cast<const SortPair<T>*>(v1);
|
|
|
|
const auto* p2 = static_cast<const SortPair<T>*>(v2);
|
2010-11-24 02:34:14 +08:00
|
|
|
if (p2->value - p1->value < 0) return -1;
|
|
|
|
if (p2->value - p1->value > 0) return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Constructor takes a hint of the array size, but it need not be accurate.
|
2013-09-23 23:26:50 +08:00
|
|
|
explicit SortHelper(int sizehint) {
|
|
|
|
counts_.reserve(sizehint);
|
|
|
|
}
|
2010-11-24 02:34:14 +08:00
|
|
|
|
|
|
|
// Add a value that may be a duplicate of an existing value.
|
|
|
|
// Uses a linear search.
|
|
|
|
void Add(T value, int count) {
|
|
|
|
// Linear search for value.
|
|
|
|
for (int i = 0; i < counts_.size(); ++i) {
|
|
|
|
if (counts_[i].value == value) {
|
|
|
|
counts_[i].count += count;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SortPair<T> new_pair = {value, count};
|
|
|
|
counts_.push_back(SortPair<T>(new_pair));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the frequency of the most frequent value.
|
2016-12-13 00:20:28 +08:00
|
|
|
// If max_value is not nullptr, returns the most frequent value.
|
Use POSIX data types and macros (#878)
* api: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccmain: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccstruct: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* classify: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* cutil: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* dict: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* textord: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* training: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* wordrec: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccutil: Replace Tesseract data types by POSIX data types
Now all Tesseract data types which are no longer needed can be removed
from ccutil/host.h.
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccmain: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccstruct: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* classify: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* dict: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* lstm: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* textord: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* wordrec: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccutil: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Remove the macros which are now unused from ccutil/host.h.
Remove also the obsolete history comments.
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* Fix build error caused by ambiguous ClipToRange
Error message vom Appveyor CI:
C:\projects\tesseract\ccstruct\coutln.cpp(818): error C2672: 'ClipToRange': no matching overloaded function found [C:\projects\tesseract\build\libtesseract.vcxproj]
C:\projects\tesseract\ccstruct\coutln.cpp(818): error C2782: 'T ClipToRange(const T &,const T &,const T &)': template parameter 'T' is ambiguous [C:\projects\tesseract\build\libtesseract.vcxproj]
c:\projects\tesseract\ccutil\helpers.h(122): note: see declaration of 'ClipToRange'
C:\projects\tesseract\ccstruct\coutln.cpp(818): note: could be 'char'
C:\projects\tesseract\ccstruct\coutln.cpp(818): note: or 'int'
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* unittest: Replace Tesseract's MAX_INT8 by POSIX INT8_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* arch: Replace Tesseract's MAX_INT8 by POSIX INT8_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
2018-03-14 04:36:30 +08:00
|
|
|
// If the array is empty, returns -INT32_MAX and max_value is unchanged.
|
2010-11-24 02:34:14 +08:00
|
|
|
int MaxCount(T* max_value) const {
|
Use POSIX data types and macros (#878)
* api: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccmain: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccstruct: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* classify: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* cutil: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* dict: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* textord: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* training: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* wordrec: Replace Tesseract data types by POSIX data types
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccutil: Replace Tesseract data types by POSIX data types
Now all Tesseract data types which are no longer needed can be removed
from ccutil/host.h.
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccmain: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccstruct: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* classify: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* dict: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* lstm: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* textord: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* wordrec: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* ccutil: Replace Tesseract's MIN_*INT, MAX_*INT* by POSIX *INT*_MIN, *INT*_MAX
Remove the macros which are now unused from ccutil/host.h.
Remove also the obsolete history comments.
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* Fix build error caused by ambiguous ClipToRange
Error message vom Appveyor CI:
C:\projects\tesseract\ccstruct\coutln.cpp(818): error C2672: 'ClipToRange': no matching overloaded function found [C:\projects\tesseract\build\libtesseract.vcxproj]
C:\projects\tesseract\ccstruct\coutln.cpp(818): error C2782: 'T ClipToRange(const T &,const T &,const T &)': template parameter 'T' is ambiguous [C:\projects\tesseract\build\libtesseract.vcxproj]
c:\projects\tesseract\ccutil\helpers.h(122): note: see declaration of 'ClipToRange'
C:\projects\tesseract\ccstruct\coutln.cpp(818): note: could be 'char'
C:\projects\tesseract\ccstruct\coutln.cpp(818): note: or 'int'
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* unittest: Replace Tesseract's MAX_INT8 by POSIX INT8_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
* arch: Replace Tesseract's MAX_INT8 by POSIX INT8_MAX
Signed-off-by: Stefan Weil <sw@weilnetz.de>
2018-03-14 04:36:30 +08:00
|
|
|
int best_count = -INT32_MAX;
|
2010-11-24 02:34:14 +08:00
|
|
|
for (int i = 0; i < counts_.size(); ++i) {
|
|
|
|
if (counts_[i].count > best_count) {
|
|
|
|
best_count = counts_[i].count;
|
2016-12-13 00:20:28 +08:00
|
|
|
if (max_value != nullptr)
|
2010-11-24 02:34:14 +08:00
|
|
|
*max_value = counts_[i].value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return best_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns the data array sorted by decreasing frequency.
|
|
|
|
const GenericVector<SortPair<T> >& SortByCount() {
|
|
|
|
counts_.sort(&SortPairsByCount);
|
|
|
|
return counts_;
|
|
|
|
}
|
|
|
|
// Returns the data array sorted by decreasing value.
|
|
|
|
const GenericVector<SortPair<T> >& SortByValue() {
|
|
|
|
counts_.sort(&SortPairsByValue);
|
|
|
|
return counts_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
GenericVector<SortPair<T> > counts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif // TESSERACT_CCUTIL_SORTHELPER_H_.
|