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
a963f5d5
Commit
a963f5d5
authored
Sep 17, 2011
by
Davis King
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Added unit tests for the new object detection tools
parent
4d3cef36
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
412 additions
and
0 deletions
+412
-0
CMakeLists.txt
dlib/test/CMakeLists.txt
+1
-0
makefile
dlib/test/makefile
+1
-0
object_detector.cpp
dlib/test/object_detector.cpp
+410
-0
No files found.
dlib/test/CMakeLists.txt
View file @
a963f5d5
...
...
@@ -69,6 +69,7 @@ set (tests
member_function_pointer.cpp
metaprogramming.cpp
multithreaded_object.cpp
object_detector.cpp
one_vs_all_trainer.cpp
one_vs_one_trainer.cpp
optimization.cpp
...
...
dlib/test/makefile
View file @
a963f5d5
...
...
@@ -84,6 +84,7 @@ SRC += md5.cpp
SRC
+=
member_function_pointer.cpp
SRC
+=
metaprogramming.cpp
SRC
+=
multithreaded_object.cpp
SRC
+=
object_detector.cpp
SRC
+=
one_vs_all_trainer.cpp
SRC
+=
one_vs_one_trainer.cpp
SRC
+=
optimization.cpp
...
...
dlib/test/object_detector.cpp
0 → 100644
View file @
a963f5d5
// Copyright (C) 2011 Davis E. King (davis@dlib.net)
// License: Boost Software License See LICENSE.txt for the full license.
#include <sstream>
#include <string>
#include <cstdlib>
#include <ctime>
#include "tester.h"
#include <dlib/pixel.h>
#include <dlib/svm_threaded.h>
#include <dlib/gui_widgets.h>
#include <dlib/array.h>
#include <dlib/array2d.h>
#include <dlib/image_keypoint.h>
#include <dlib/image_processing.h>
namespace
{
using
namespace
test
;
using
namespace
dlib
;
using
namespace
std
;
logger
dlog
(
"test.object_detector"
);
// ----------------------------------------------------------------------------------------
struct
funny_image
{
array2d
<
unsigned
char
>
img
;
long
nr
()
const
{
return
img
.
nr
();
}
long
nc
()
const
{
return
img
.
nc
();
}
};
void
swap
(
funny_image
&
a
,
funny_image
&
b
)
{
a
.
img
.
swap
(
b
.
img
);
}
// ----------------------------------------------------------------------------------------
class
very_simple_feature_extractor
:
noncopyable
{
/*!
WHAT THIS OBJECT REPRESENTS
This object is a feature extractor which goes to every pixel in an image and
produces a 32 dimensional feature vector. This vector is an indicator vector
which records the pattern of pixel values in a 4-connected region. So it should
be able to distinguish basic things like whether or not a location falls on the
corner of a white box, on an edge, in the middle, etc.
Note that this object also implements the interface defined in dlib/image_keypoint/hashed_feature_image_abstract.h.
This means all the member functions in this object are supposed to behave as
described in the hashed_feature_image specification. So when you define your own
feature extractor objects you should probably refer yourself to that documentation
in addition to reading this example program.
!*/
public
:
inline
void
load
(
const
funny_image
&
img_
)
{
const
array2d
<
unsigned
char
>&
img
=
img_
.
img
;
feat_image
.
set_size
(
img
.
nr
(),
img
.
nc
());
assign_all_pixels
(
feat_image
,
0
);
for
(
long
r
=
1
;
r
+
1
<
img
.
nr
();
++
r
)
{
for
(
long
c
=
1
;
c
+
1
<
img
.
nc
();
++
c
)
{
unsigned
char
f
=
0
;
if
(
img
[
r
][
c
])
f
|=
0x1
;
if
(
img
[
r
][
c
+
1
])
f
|=
0x2
;
if
(
img
[
r
][
c
-
1
])
f
|=
0x4
;
if
(
img
[
r
+
1
][
c
])
f
|=
0x8
;
if
(
img
[
r
-
1
][
c
])
f
|=
0x10
;
// Store the code value for the pattern of pixel values in the 4-connected
// neighborhood around this row and column.
feat_image
[
r
][
c
]
=
f
;
}
}
}
inline
void
load
(
const
array2d
<
unsigned
char
>&
img
)
{
feat_image
.
set_size
(
img
.
nr
(),
img
.
nc
());
assign_all_pixels
(
feat_image
,
0
);
for
(
long
r
=
1
;
r
+
1
<
img
.
nr
();
++
r
)
{
for
(
long
c
=
1
;
c
+
1
<
img
.
nc
();
++
c
)
{
unsigned
char
f
=
0
;
if
(
img
[
r
][
c
])
f
|=
0x1
;
if
(
img
[
r
][
c
+
1
])
f
|=
0x2
;
if
(
img
[
r
][
c
-
1
])
f
|=
0x4
;
if
(
img
[
r
+
1
][
c
])
f
|=
0x8
;
if
(
img
[
r
-
1
][
c
])
f
|=
0x10
;
// Store the code value for the pattern of pixel values in the 4-connected
// neighborhood around this row and column.
feat_image
[
r
][
c
]
=
f
;
}
}
}
inline
unsigned
long
size
()
const
{
return
feat_image
.
size
();
}
inline
long
nr
()
const
{
return
feat_image
.
nr
();
}
inline
long
nc
()
const
{
return
feat_image
.
nc
();
}
inline
long
get_num_dimensions
(
)
const
{
// Return the dimensionality of the vectors produced by operator()
return
32
;
}
typedef
std
::
vector
<
std
::
pair
<
unsigned
int
,
double
>
>
descriptor_type
;
inline
const
descriptor_type
&
operator
()
(
long
row
,
long
col
)
const
/*!
requires
- 0 <= row < nr()
- 0 <= col < nc()
ensures
- returns a sparse vector which describes the image at the given row and column.
In particular, this is a vector that is 0 everywhere except for one element.
!*/
{
feat
.
clear
();
const
unsigned
long
only_nonzero_element_index
=
feat_image
[
row
][
col
];
feat
.
push_back
(
make_pair
(
only_nonzero_element_index
,
1.0
));
return
feat
;
}
// This block of functions is meant to provide a way to map between the row/col space taken by
// this object's operator() function and the images supplied to load(). In this example it's trivial.
// However, in general, you might create feature extractors which don't perform extraction at every
// possible image location (e.g. the hog_image) and thus result in some more complex mapping.
inline
const
rectangle
get_block_rect
(
long
row
,
long
col
)
const
{
return
centered_rect
(
col
,
row
,
3
,
3
);
}
inline
const
point
image_to_feat_space
(
const
point
&
p
)
const
{
return
p
;
}
inline
const
rectangle
image_to_feat_space
(
const
rectangle
&
rect
)
const
{
return
rect
;
}
inline
const
point
feat_to_image_space
(
const
point
&
p
)
const
{
return
p
;
}
inline
const
rectangle
feat_to_image_space
(
const
rectangle
&
rect
)
const
{
return
rect
;
}
inline
friend
void
serialize
(
const
very_simple_feature_extractor
&
item
,
std
::
ostream
&
out
)
{
serialize
(
item
.
feat_image
,
out
);
}
inline
friend
void
deserialize
(
very_simple_feature_extractor
&
item
,
std
::
istream
&
in
)
{
deserialize
(
item
.
feat_image
,
in
);
}
void
copy_configuration
(
const
very_simple_feature_extractor
&
item
){}
private
:
array2d
<
unsigned
char
>
feat_image
;
// This variable doesn't logically contribute to the state of this object. It is here
// only to avoid returning a descriptor_type object by value inside the operator() method.
mutable
descriptor_type
feat
;
};
// ----------------------------------------------------------------------------------------
template
<
typename
image_array_type
>
void
make_simple_test_data
(
image_array_type
&
images
,
std
::
vector
<
std
::
vector
<
rectangle
>
>&
object_locations
)
{
images
.
clear
();
object_locations
.
clear
();
images
.
resize
(
3
);
images
[
0
].
set_size
(
400
,
400
);
images
[
1
].
set_size
(
400
,
400
);
images
[
2
].
set_size
(
400
,
400
);
// set all the pixel values to black
assign_all_pixels
(
images
[
0
],
0
);
assign_all_pixels
(
images
[
1
],
0
);
assign_all_pixels
(
images
[
2
],
0
);
// Now make some squares and draw them onto our black images. All the
// squares will be 70 pixels wide and tall.
std
::
vector
<
rectangle
>
temp
;
temp
.
push_back
(
centered_rect
(
point
(
100
,
100
),
70
,
70
));
fill_rect
(
images
[
0
],
temp
.
back
(),
255
);
// Paint the square white
temp
.
push_back
(
centered_rect
(
point
(
200
,
300
),
70
,
70
));
fill_rect
(
images
[
0
],
temp
.
back
(),
255
);
// Paint the square white
object_locations
.
push_back
(
temp
);
temp
.
clear
();
temp
.
push_back
(
centered_rect
(
point
(
140
,
200
),
70
,
70
));
fill_rect
(
images
[
1
],
temp
.
back
(),
255
);
// Paint the square white
temp
.
push_back
(
centered_rect
(
point
(
303
,
200
),
70
,
70
));
fill_rect
(
images
[
1
],
temp
.
back
(),
255
);
// Paint the square white
object_locations
.
push_back
(
temp
);
temp
.
clear
();
temp
.
push_back
(
centered_rect
(
point
(
123
,
121
),
70
,
70
));
fill_rect
(
images
[
2
],
temp
.
back
(),
255
);
// Paint the square white
object_locations
.
push_back
(
temp
);
}
// ----------------------------------------------------------------------------------------
void
test_1
(
)
{
print_spinner
();
dlog
<<
LINFO
<<
"test_1()"
;
typedef
array
<
array2d
<
unsigned
char
>
>::
expand_1b
grayscale_image_array_type
;
grayscale_image_array_type
images
;
std
::
vector
<
std
::
vector
<
rectangle
>
>
object_locations
;
make_simple_test_data
(
images
,
object_locations
);
typedef
hashed_feature_image
<
hog_image
<
3
,
3
,
1
,
4
,
hog_signed_gradient
,
hog_full_interpolation
>
>
feature_extractor_type
;
typedef
scan_image_pyramid
<
pyramid_down
,
feature_extractor_type
>
image_scanner_type
;
image_scanner_type
scanner
;
const
rectangle
object_box
=
compute_box_dimensions
(
1
,
35
*
35
);
scanner
.
add_detection_template
(
object_box
,
create_grid_detection_template
(
object_box
,
2
,
2
));
structural_object_detection_trainer
<
image_scanner_type
>
trainer
(
scanner
);
trainer
.
set_num_threads
(
4
);
trainer
.
set_overlap_tester
(
test_box_overlap
(
0
));
const
object_detector
<
image_scanner_type
>
detector
=
trainer
.
train
(
images
,
object_locations
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
{
ostringstream
sout
;
serialize
(
detector
,
sout
);
istringstream
sin
(
sout
.
str
());
object_detector
<
image_scanner_type
>
d2
;
deserialize
(
d2
,
sin
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
}
}
// ----------------------------------------------------------------------------------------
void
test_2
(
)
{
print_spinner
();
dlog
<<
LINFO
<<
"test_2()"
;
typedef
array
<
array2d
<
unsigned
char
>
>::
expand_1b
grayscale_image_array_type
;
grayscale_image_array_type
images
;
std
::
vector
<
std
::
vector
<
rectangle
>
>
object_locations
;
make_simple_test_data
(
images
,
object_locations
);
typedef
scan_image_pyramid
<
pyramid_down_5_4
,
very_simple_feature_extractor
>
image_scanner_type
;
image_scanner_type
scanner
;
const
rectangle
object_box
=
compute_box_dimensions
(
1
,
70
*
70
);
scanner
.
add_detection_template
(
object_box
,
create_grid_detection_template
(
object_box
,
2
,
2
));
scanner
.
set_max_pyramid_levels
(
1
);
structural_object_detection_trainer
<
image_scanner_type
>
trainer
(
scanner
);
trainer
.
set_num_threads
(
0
);
trainer
.
set_overlap_tester
(
test_box_overlap
(
0
));
const
object_detector
<
image_scanner_type
>
detector
=
trainer
.
train
(
images
,
object_locations
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
res
=
cross_validate_object_detection_trainer
(
trainer
,
images
,
object_locations
,
3
);
dlog
<<
LINFO
<<
"3-fold cross validation (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
{
ostringstream
sout
;
serialize
(
detector
,
sout
);
istringstream
sin
(
sout
.
str
());
object_detector
<
image_scanner_type
>
d2
;
deserialize
(
d2
,
sin
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
}
}
// ----------------------------------------------------------------------------------------
class
pyramid_down_funny
:
noncopyable
{
pyramid_down
pyr
;
public
:
template
<
typename
T
>
dlib
::
vector
<
double
,
2
>
point_down
(
const
dlib
::
vector
<
T
,
2
>&
p
)
const
{
return
pyr
.
point_down
(
p
);
}
template
<
typename
T
>
dlib
::
vector
<
double
,
2
>
point_up
(
const
dlib
::
vector
<
T
,
2
>&
p
)
const
{
return
pyr
.
point_up
(
p
);
}
template
<
typename
T
>
dlib
::
vector
<
double
,
2
>
point_down
(
const
dlib
::
vector
<
T
,
2
>&
p
,
unsigned
int
levels
)
const
{
return
pyr
.
point_down
(
p
,
levels
);
}
template
<
typename
T
>
dlib
::
vector
<
double
,
2
>
point_up
(
const
dlib
::
vector
<
T
,
2
>&
p
,
unsigned
int
levels
)
const
{
return
pyr
.
point_up
(
p
,
levels
);
}
rectangle
rect_up
(
const
rectangle
&
rect
)
const
{
return
pyr
.
rect_up
(
rect
);
}
rectangle
rect_up
(
const
rectangle
&
rect
,
unsigned
int
levels
)
const
{
return
pyr
.
rect_up
(
rect
,
levels
);
}
rectangle
rect_down
(
const
rectangle
&
rect
)
const
{
return
pyr
.
rect_down
(
rect
);
}
rectangle
rect_down
(
const
rectangle
&
rect
,
unsigned
int
levels
)
const
{
return
pyr
.
rect_down
(
rect
,
levels
);
}
template
<
typename
in_image_type
,
typename
out_image_type
>
void
operator
()
(
const
in_image_type
&
original
,
out_image_type
&
down
)
const
{
pyr
(
original
.
img
,
down
.
img
);
}
};
// make sure everything works even when the image isn't a dlib::array2d.
// So test with funny_image.
void
test_3
(
)
{
print_spinner
();
dlog
<<
LINFO
<<
"test_3()"
;
typedef
array
<
array2d
<
unsigned
char
>
>::
expand_1b
grayscale_image_array_type
;
typedef
array
<
funny_image
>::
expand_1b
funny_image_array_type
;
grayscale_image_array_type
images_temp
;
funny_image_array_type
images
;
std
::
vector
<
std
::
vector
<
rectangle
>
>
object_locations
;
make_simple_test_data
(
images_temp
,
object_locations
);
images
.
resize
(
images_temp
.
size
());
for
(
unsigned
long
i
=
0
;
i
<
images_temp
.
size
();
++
i
)
{
images
[
i
].
img
.
swap
(
images_temp
[
i
]);
}
typedef
scan_image_pyramid
<
pyramid_down_funny
,
very_simple_feature_extractor
>
image_scanner_type
;
image_scanner_type
scanner
;
const
rectangle
object_box
=
compute_box_dimensions
(
1
,
70
*
70
);
scanner
.
add_detection_template
(
object_box
,
create_grid_detection_template
(
object_box
,
2
,
2
));
scanner
.
set_max_pyramid_levels
(
1
);
structural_object_detection_trainer
<
image_scanner_type
>
trainer
(
scanner
);
trainer
.
set_num_threads
(
4
);
trainer
.
set_overlap_tester
(
test_box_overlap
(
0
));
const
object_detector
<
image_scanner_type
>
detector
=
trainer
.
train
(
images
,
object_locations
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
res
=
cross_validate_object_detection_trainer
(
trainer
,
images
,
object_locations
,
3
);
dlog
<<
LINFO
<<
"3-fold cross validation (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
{
ostringstream
sout
;
serialize
(
detector
,
sout
);
istringstream
sin
(
sout
.
str
());
object_detector
<
image_scanner_type
>
d2
;
deserialize
(
d2
,
sin
);
matrix
<
double
>
res
=
test_object_detection_function
(
detector
,
images
,
object_locations
);
dlog
<<
LINFO
<<
"Test detector (precision,recall): "
<<
res
;
DLIB_TEST
(
sum
(
res
)
==
2
);
}
}
// ----------------------------------------------------------------------------------------
class
object_detector_tester
:
public
tester
{
public
:
object_detector_tester
(
)
:
tester
(
"test_object_detector"
,
"Runs tests on the structural object detection stuff."
)
{}
void
perform_test
(
)
{
test_1
();
test_2
();
test_3
();
}
}
a
;
}
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