Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
J
json
Project
Overview
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
Chen Yisong
json
Commits
4f270e38
Unverified
Commit
4f270e38
authored
Dec 19, 2018
by
Niels Lohmann
Committed by
GitHub
Dec 19, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1342 from davedissian/bugfix/sfinae-iterator-traits
Add a SFINAE friendly iterator_traits and use that instead.
parents
5d390e91
f1080d7c
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
125 additions
and
20 deletions
+125
-20
input_adapters.hpp
include/nlohmann/detail/input/input_adapters.hpp
+3
-3
iterator_traits.hpp
include/nlohmann/detail/iterators/iterator_traits.hpp
+49
-0
type_traits.hpp
include/nlohmann/detail/meta/type_traits.hpp
+5
-4
json.hpp
include/nlohmann/json.hpp
+1
-1
json.hpp
single_include/nlohmann/json.hpp
+67
-12
No files found.
include/nlohmann/detail/input/input_adapters.hpp
View file @
4f270e38
...
@@ -360,7 +360,7 @@ class input_adapter
...
@@ -360,7 +360,7 @@ class input_adapter
/// input adapter for iterator range with contiguous storage
/// input adapter for iterator range with contiguous storage
template
<
class
IteratorType
,
template
<
class
IteratorType
,
typename
std
::
enable_if
<
typename
std
::
enable_if
<
std
::
is_same
<
typename
std
::
iterator_traits
<
IteratorType
>::
iterator_category
,
std
::
random_access_iterator_tag
>::
value
,
std
::
is_same
<
typename
iterator_traits
<
IteratorType
>::
iterator_category
,
std
::
random_access_iterator_tag
>::
value
,
int
>::
type
=
0
>
int
>::
type
=
0
>
input_adapter
(
IteratorType
first
,
IteratorType
last
)
input_adapter
(
IteratorType
first
,
IteratorType
last
)
{
{
...
@@ -379,7 +379,7 @@ class input_adapter
...
@@ -379,7 +379,7 @@ class input_adapter
// assertion to check that each element is 1 byte long
// assertion to check that each element is 1 byte long
static_assert
(
static_assert
(
sizeof
(
typename
std
::
iterator_traits
<
IteratorType
>::
value_type
)
==
1
,
sizeof
(
typename
iterator_traits
<
IteratorType
>::
value_type
)
==
1
,
"each element in the iterator range must have the size of 1 byte"
);
"each element in the iterator range must have the size of 1 byte"
);
const
auto
len
=
static_cast
<
size_t
>
(
std
::
distance
(
first
,
last
));
const
auto
len
=
static_cast
<
size_t
>
(
std
::
distance
(
first
,
last
));
...
@@ -403,7 +403,7 @@ class input_adapter
...
@@ -403,7 +403,7 @@ class input_adapter
/// input adapter for contiguous container
/// input adapter for contiguous container
template
<
class
ContiguousContainer
,
typename
template
<
class
ContiguousContainer
,
typename
std
::
enable_if
<
not
std
::
is_pointer
<
ContiguousContainer
>::
value
and
std
::
enable_if
<
not
std
::
is_pointer
<
ContiguousContainer
>::
value
and
std
::
is_base_of
<
std
::
random_access_iterator_tag
,
typename
std
::
iterator_traits
<
decltype
(
std
::
begin
(
std
::
declval
<
ContiguousContainer
const
>
()))
>::
iterator_category
>::
value
,
std
::
is_base_of
<
std
::
random_access_iterator_tag
,
typename
iterator_traits
<
decltype
(
std
::
begin
(
std
::
declval
<
ContiguousContainer
const
>
()))
>::
iterator_category
>::
value
,
int
>::
type
=
0
>
int
>::
type
=
0
>
input_adapter
(
const
ContiguousContainer
&
c
)
input_adapter
(
const
ContiguousContainer
&
c
)
:
input_adapter
(
std
::
begin
(
c
),
std
::
end
(
c
))
{}
:
input_adapter
(
std
::
begin
(
c
),
std
::
end
(
c
))
{}
...
...
include/nlohmann/detail/iterators/iterator_traits.hpp
0 → 100644
View file @
4f270e38
#pragma once
#include <iterator> // random_access_iterator_tag
#include <nlohmann/detail/meta/void_t.hpp>
#include <nlohmann/detail/meta/cpp_future.hpp>
namespace
nlohmann
{
namespace
detail
{
template
<
typename
It
,
typename
=
void
>
struct
iterator_types
{};
template
<
typename
It
>
struct
iterator_types
<
It
,
void_t
<
typename
It
::
difference_type
,
typename
It
::
value_type
,
typename
It
::
pointer
,
typename
It
::
reference
,
typename
It
::
iterator_category
>>
{
using
difference_type
=
typename
It
::
difference_type
;
using
value_type
=
typename
It
::
value_type
;
using
pointer
=
typename
It
::
pointer
;
using
reference
=
typename
It
::
reference
;
using
iterator_category
=
typename
It
::
iterator_category
;
};
// This is required as some compilers implement std::iterator_traits in a way that
// doesn't work with SFINAE. See https://github.com/nlohmann/json/issues/1341.
template
<
typename
T
,
typename
=
void
>
struct
iterator_traits
{
};
template
<
typename
T
>
struct
iterator_traits
<
T
,
enable_if_t
<!
std
::
is_pointer
<
T
>::
value
>>
:
iterator_types
<
T
>
{
};
template
<
typename
T
>
struct
iterator_traits
<
T
*
,
enable_if_t
<
std
::
is_object
<
T
>::
value
>>
{
using
iterator_category
=
std
::
random_access_iterator_tag
;
using
value_type
=
T
;
using
difference_type
=
ptrdiff_t
;
using
pointer
=
T
*
;
using
reference
=
T
&
;
};
}
}
include/nlohmann/detail/meta/type_traits.hpp
View file @
4f270e38
...
@@ -6,6 +6,7 @@
...
@@ -6,6 +6,7 @@
#include <utility> // declval
#include <utility> // declval
#include <nlohmann/json_fwd.hpp>
#include <nlohmann/json_fwd.hpp>
#include <nlohmann/detail/iterators/iterator_traits.hpp>
#include <nlohmann/detail/meta/cpp_future.hpp>
#include <nlohmann/detail/meta/cpp_future.hpp>
#include <nlohmann/detail/meta/detected.hpp>
#include <nlohmann/detail/meta/detected.hpp>
#include <nlohmann/detail/macro_scope.hpp>
#include <nlohmann/detail/macro_scope.hpp>
...
@@ -131,10 +132,10 @@ template <typename T, typename = void>
...
@@ -131,10 +132,10 @@ template <typename T, typename = void>
struct
is_iterator_traits
:
std
::
false_type
{};
struct
is_iterator_traits
:
std
::
false_type
{};
template
<
typename
T
>
template
<
typename
T
>
struct
is_iterator_traits
<
std
::
iterator_traits
<
T
>>
struct
is_iterator_traits
<
iterator_traits
<
T
>>
{
{
private
:
private
:
using
traits
=
std
::
iterator_traits
<
T
>
;
using
traits
=
iterator_traits
<
T
>
;
public
:
public
:
static
constexpr
auto
value
=
static
constexpr
auto
value
=
...
@@ -251,7 +252,7 @@ struct is_compatible_array_type_impl <
...
@@ -251,7 +252,7 @@ struct is_compatible_array_type_impl <
// Therefore it is detected as a CompatibleArrayType.
// Therefore it is detected as a CompatibleArrayType.
// The real fix would be to have an Iterable concept.
// The real fix would be to have an Iterable concept.
not
is_iterator_traits
<
not
is_iterator_traits
<
std
::
iterator_traits
<
CompatibleArrayType
>>::
value
>>
iterator_traits
<
CompatibleArrayType
>>::
value
>>
{
{
static
constexpr
bool
value
=
static
constexpr
bool
value
=
std
::
is_constructible
<
BasicJsonType
,
std
::
is_constructible
<
BasicJsonType
,
...
@@ -288,7 +289,7 @@ struct is_constructible_array_type_impl <
...
@@ -288,7 +289,7 @@ struct is_constructible_array_type_impl <
// Therefore it is detected as a ConstructibleArrayType.
// Therefore it is detected as a ConstructibleArrayType.
// The real fix would be to have an Iterable concept.
// The real fix would be to have an Iterable concept.
not
is_iterator_traits
<
not
is_iterator_traits
<
std
::
iterator_traits
<
ConstructibleArrayType
>>::
value
and
iterator_traits
<
ConstructibleArrayType
>>::
value
and
(
std
::
is_same
<
typename
ConstructibleArrayType
::
value_type
,
typename
BasicJsonType
::
array_t
::
value_type
>::
value
or
(
std
::
is_same
<
typename
ConstructibleArrayType
::
value_type
,
typename
BasicJsonType
::
array_t
::
value_type
>::
value
or
has_from_json
<
BasicJsonType
,
has_from_json
<
BasicJsonType
,
...
...
include/nlohmann/json.hpp
View file @
4f270e38
...
@@ -41,7 +41,7 @@ SOFTWARE.
...
@@ -41,7 +41,7 @@ SOFTWARE.
#include <functional> // hash, less
#include <functional> // hash, less
#include <initializer_list> // initializer_list
#include <initializer_list> // initializer_list
#include <iosfwd> // istream, ostream
#include <iosfwd> // istream, ostream
#include <iterator> //
iterator_traits,
random_access_iterator_tag
#include <iterator> // random_access_iterator_tag
#include <numeric> // accumulate
#include <numeric> // accumulate
#include <string> // string, stoi, to_string
#include <string> // string, stoi, to_string
#include <utility> // declval, forward, move, pair, swap
#include <utility> // declval, forward, move, pair, swap
...
...
single_include/nlohmann/json.hpp
View file @
4f270e38
...
@@ -41,7 +41,7 @@ SOFTWARE.
...
@@ -41,7 +41,7 @@ SOFTWARE.
#include <functional> // hash, less
#include <functional> // hash, less
#include <initializer_list> // initializer_list
#include <initializer_list> // initializer_list
#include <iosfwd> // istream, ostream
#include <iosfwd> // istream, ostream
#include <iterator> //
iterator_traits,
random_access_iterator_tag
#include <iterator> // random_access_iterator_tag
#include <numeric> // accumulate
#include <numeric> // accumulate
#include <string> // string, stoi, to_string
#include <string> // string, stoi, to_string
#include <utility> // declval, forward, move, pair, swap
#include <utility> // declval, forward, move, pair, swap
...
@@ -324,12 +324,10 @@ constexpr T static_const<T>::value;
...
@@ -324,12 +324,10 @@ constexpr T static_const<T>::value;
// #include <nlohmann/json_fwd.hpp>
// #include <nlohmann/json_fwd.hpp>
// #include <nlohmann/detail/
meta/cpp_future
.hpp>
// #include <nlohmann/detail/
iterators/iterator_traits
.hpp>
// #include <nlohmann/detail/meta/detected.hpp>
#include <iterator> // random_access_iterator_tag
#include <type_traits>
// #include <nlohmann/detail/meta/void_t.hpp>
// #include <nlohmann/detail/meta/void_t.hpp>
...
@@ -346,6 +344,63 @@ template <typename ...Ts> using void_t = typename make_void<Ts...>::type;
...
@@ -346,6 +344,63 @@ template <typename ...Ts> using void_t = typename make_void<Ts...>::type;
}
// namespace detail
}
// namespace detail
}
// namespace nlohmann
}
// namespace nlohmann
// #include <nlohmann/detail/meta/cpp_future.hpp>
namespace
nlohmann
{
namespace
detail
{
template
<
typename
It
,
typename
=
void
>
struct
iterator_types
{};
template
<
typename
It
>
struct
iterator_types
<
It
,
void_t
<
typename
It
::
difference_type
,
typename
It
::
value_type
,
typename
It
::
pointer
,
typename
It
::
reference
,
typename
It
::
iterator_category
>>
{
using
difference_type
=
typename
It
::
difference_type
;
using
value_type
=
typename
It
::
value_type
;
using
pointer
=
typename
It
::
pointer
;
using
reference
=
typename
It
::
reference
;
using
iterator_category
=
typename
It
::
iterator_category
;
};
// This is required as some compilers implement std::iterator_traits in a way that
// doesn't work with SFINAE. See https://github.com/nlohmann/json/issues/1341.
template
<
typename
T
,
typename
=
void
>
struct
iterator_traits
{
};
template
<
typename
T
>
struct
iterator_traits
<
T
,
enable_if_t
<
!
std
::
is_pointer
<
T
>::
value
>>
:
iterator_types
<
T
>
{
};
template
<
typename
T
>
struct
iterator_traits
<
T
*
,
enable_if_t
<
std
::
is_object
<
T
>::
value
>>
{
using
iterator_category
=
std
::
random_access_iterator_tag
;
using
value_type
=
T
;
using
difference_type
=
ptrdiff_t
;
using
pointer
=
T
*
;
using
reference
=
T
&
;
};
}
}
// #include <nlohmann/detail/meta/cpp_future.hpp>
// #include <nlohmann/detail/meta/detected.hpp>
#include <type_traits>
// #include <nlohmann/detail/meta/void_t.hpp>
// http://en.cppreference.com/w/cpp/experimental/is_detected
// http://en.cppreference.com/w/cpp/experimental/is_detected
namespace
nlohmann
namespace
nlohmann
...
@@ -522,10 +577,10 @@ template <typename T, typename = void>
...
@@ -522,10 +577,10 @@ template <typename T, typename = void>
struct
is_iterator_traits
:
std
::
false_type
{};
struct
is_iterator_traits
:
std
::
false_type
{};
template
<
typename
T
>
template
<
typename
T
>
struct
is_iterator_traits
<
std
::
iterator_traits
<
T
>>
struct
is_iterator_traits
<
iterator_traits
<
T
>>
{
{
private
:
private
:
using
traits
=
std
::
iterator_traits
<
T
>
;
using
traits
=
iterator_traits
<
T
>
;
public
:
public
:
static
constexpr
auto
value
=
static
constexpr
auto
value
=
...
@@ -642,7 +697,7 @@ struct is_compatible_array_type_impl <
...
@@ -642,7 +697,7 @@ struct is_compatible_array_type_impl <
// Therefore it is detected as a CompatibleArrayType.
// Therefore it is detected as a CompatibleArrayType.
// The real fix would be to have an Iterable concept.
// The real fix would be to have an Iterable concept.
not
is_iterator_traits
<
not
is_iterator_traits
<
std
::
iterator_traits
<
CompatibleArrayType
>>::
value
>>
iterator_traits
<
CompatibleArrayType
>>::
value
>>
{
{
static
constexpr
bool
value
=
static
constexpr
bool
value
=
std
::
is_constructible
<
BasicJsonType
,
std
::
is_constructible
<
BasicJsonType
,
...
@@ -679,7 +734,7 @@ struct is_constructible_array_type_impl <
...
@@ -679,7 +734,7 @@ struct is_constructible_array_type_impl <
// Therefore it is detected as a ConstructibleArrayType.
// Therefore it is detected as a ConstructibleArrayType.
// The real fix would be to have an Iterable concept.
// The real fix would be to have an Iterable concept.
not
is_iterator_traits
<
not
is_iterator_traits
<
std
::
iterator_traits
<
ConstructibleArrayType
>>::
value
and
iterator_traits
<
ConstructibleArrayType
>>::
value
and
(
std
::
is_same
<
typename
ConstructibleArrayType
::
value_type
,
typename
BasicJsonType
::
array_t
::
value_type
>::
value
or
(
std
::
is_same
<
typename
ConstructibleArrayType
::
value_type
,
typename
BasicJsonType
::
array_t
::
value_type
>::
value
or
has_from_json
<
BasicJsonType
,
has_from_json
<
BasicJsonType
,
...
@@ -2409,7 +2464,7 @@ class input_adapter
...
@@ -2409,7 +2464,7 @@ class input_adapter
/// input adapter for iterator range with contiguous storage
/// input adapter for iterator range with contiguous storage
template
<
class
IteratorType
,
template
<
class
IteratorType
,
typename
std
::
enable_if
<
typename
std
::
enable_if
<
std
::
is_same
<
typename
std
::
iterator_traits
<
IteratorType
>::
iterator_category
,
std
::
random_access_iterator_tag
>::
value
,
std
::
is_same
<
typename
iterator_traits
<
IteratorType
>::
iterator_category
,
std
::
random_access_iterator_tag
>::
value
,
int
>::
type
=
0
>
int
>::
type
=
0
>
input_adapter
(
IteratorType
first
,
IteratorType
last
)
input_adapter
(
IteratorType
first
,
IteratorType
last
)
{
{
...
@@ -2428,7 +2483,7 @@ class input_adapter
...
@@ -2428,7 +2483,7 @@ class input_adapter
// assertion to check that each element is 1 byte long
// assertion to check that each element is 1 byte long
static_assert
(
static_assert
(
sizeof
(
typename
std
::
iterator_traits
<
IteratorType
>::
value_type
)
==
1
,
sizeof
(
typename
iterator_traits
<
IteratorType
>::
value_type
)
==
1
,
"each element in the iterator range must have the size of 1 byte"
);
"each element in the iterator range must have the size of 1 byte"
);
const
auto
len
=
static_cast
<
size_t
>
(
std
::
distance
(
first
,
last
));
const
auto
len
=
static_cast
<
size_t
>
(
std
::
distance
(
first
,
last
));
...
@@ -2452,7 +2507,7 @@ class input_adapter
...
@@ -2452,7 +2507,7 @@ class input_adapter
/// input adapter for contiguous container
/// input adapter for contiguous container
template
<
class
ContiguousContainer
,
typename
template
<
class
ContiguousContainer
,
typename
std
::
enable_if
<
not
std
::
is_pointer
<
ContiguousContainer
>::
value
and
std
::
enable_if
<
not
std
::
is_pointer
<
ContiguousContainer
>::
value
and
std
::
is_base_of
<
std
::
random_access_iterator_tag
,
typename
std
::
iterator_traits
<
decltype
(
std
::
begin
(
std
::
declval
<
ContiguousContainer
const
>
()))
>::
iterator_category
>::
value
,
std
::
is_base_of
<
std
::
random_access_iterator_tag
,
typename
iterator_traits
<
decltype
(
std
::
begin
(
std
::
declval
<
ContiguousContainer
const
>
()))
>::
iterator_category
>::
value
,
int
>::
type
=
0
>
int
>::
type
=
0
>
input_adapter
(
const
ContiguousContainer
&
c
)
input_adapter
(
const
ContiguousContainer
&
c
)
:
input_adapter
(
std
::
begin
(
c
),
std
::
end
(
c
))
{}
:
input_adapter
(
std
::
begin
(
c
),
std
::
end
(
c
))
{}
...
...
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