mirror of
https://github.com/rust-lang/rust.git
synced 2025-01-01 18:35:25 +00:00
Auto merge of #21806 - edwardw:new-range-impl, r=alexcrichton
The new `::ops::Range` has separated implementations for each of the numeric types, while the old `::iter::Range` has one for type `Int`. However, we do not take output bindings into account when selecting traits. So it confuses `typeck` and makes the new range does not work as good as the old one when it comes to type inference. This patch implements `Iterator` for the new range for one type `Int`. This limitation could be lifted, however, if we ever reconsider the output types' role in type inference. Closes #21595 Closes #21649 Closes #21672
This commit is contained in:
commit
c2bda2a5bb
@ -2797,93 +2797,71 @@ impl<A: Int> Iterator for RangeStepInclusive<A> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! range_impl {
|
macro_rules! range_exact_iter_impl {
|
||||||
($($t:ty)*) => ($(
|
($($t:ty)*) => ($(
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
#[stable(feature = "rust1", since = "1.0.0")]
|
||||||
impl Iterator for ::ops::Range<$t> {
|
impl ExactSizeIterator for ::ops::Range<$t> {
|
||||||
type Item = $t;
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn next(&mut self) -> Option<$t> {
|
fn len(&self) -> usize {
|
||||||
if self.start < self.end {
|
|
||||||
let result = self.start;
|
|
||||||
self.start += 1;
|
|
||||||
return Some(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
debug_assert!(self.end >= self.start);
|
debug_assert!(self.end >= self.start);
|
||||||
let hint = (self.end - self.start) as usize;
|
(self.end - self.start) as usize
|
||||||
(hint, Some(hint))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
|
||||||
impl ExactSizeIterator for ::ops::Range<$t> {}
|
|
||||||
)*)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! range_impl_no_hint {
|
|
||||||
($($t:ty)*) => ($(
|
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
|
||||||
impl Iterator for ::ops::Range<$t> {
|
|
||||||
type Item = $t;
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn next(&mut self) -> Option<$t> {
|
|
||||||
if self.start < self.end {
|
|
||||||
let result = self.start;
|
|
||||||
self.start += 1;
|
|
||||||
return Some(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return None;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)*)
|
)*)
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! range_other_impls {
|
#[stable(feature = "rust1", since = "1.0.0")]
|
||||||
($($t:ty)*) => ($(
|
impl<A: Int> Iterator for ::ops::Range<A> {
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
type Item = A;
|
||||||
impl DoubleEndedIterator for ::ops::Range<$t> {
|
|
||||||
#[inline]
|
|
||||||
fn next_back(&mut self) -> Option<$t> {
|
|
||||||
if self.start < self.end {
|
|
||||||
self.end -= 1;
|
|
||||||
return Some(self.end);
|
|
||||||
}
|
|
||||||
|
|
||||||
return None;
|
#[inline]
|
||||||
}
|
fn next(&mut self) -> Option<A> {
|
||||||
|
if self.start < self.end {
|
||||||
|
let result = self.start;
|
||||||
|
self.start = self.start + Int::one();
|
||||||
|
Some(result)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[stable(feature = "rust1", since = "1.0.0")]
|
#[inline]
|
||||||
impl Iterator for ::ops::RangeFrom<$t> {
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
type Item = $t;
|
debug_assert!(self.end >= self.start);
|
||||||
|
let hint = (self.end - self.start).to_uint();
|
||||||
#[inline]
|
(hint.unwrap_or(0), hint)
|
||||||
fn next(&mut self) -> Option<$t> {
|
}
|
||||||
let result = self.start;
|
|
||||||
self.start += 1;
|
|
||||||
debug_assert!(result < self.start);
|
|
||||||
return Some(result);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)*)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
range_impl!(usize u8 u16 u32 isize i8 i16 i32);
|
range_exact_iter_impl!(usize u8 u16 u32 isize i8 i16 i32);
|
||||||
#[cfg(target_pointer_width = "64")]
|
#[cfg(target_pointer_width = "64")]
|
||||||
range_impl!(u64 i64);
|
range_exact_iter_impl!(u64 i64);
|
||||||
#[cfg(target_pointer_width = "32")]
|
|
||||||
range_impl_no_hint!(u64 i64);
|
|
||||||
|
|
||||||
range_other_impls!(usize u8 u16 u32 u64 isize i8 i16 i32 i64);
|
#[stable(feature = "rust1", since = "1.0.0")]
|
||||||
|
impl<A: Int> DoubleEndedIterator for ::ops::Range<A> {
|
||||||
|
#[inline]
|
||||||
|
fn next_back(&mut self) -> Option<A> {
|
||||||
|
if self.start < self.end {
|
||||||
|
self.end = self.end - Int::one();
|
||||||
|
Some(self.end)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[stable(feature = "rust1", since = "1.0.0")]
|
||||||
|
impl<A: Int> Iterator for ::ops::RangeFrom<A> {
|
||||||
|
type Item = A;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn next(&mut self) -> Option<A> {
|
||||||
|
let result = self.start;
|
||||||
|
self.start = self.start + Int::one();
|
||||||
|
debug_assert!(result < self.start);
|
||||||
|
Some(result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// An iterator that repeats an element endlessly
|
/// An iterator that repeats an element endlessly
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
|
@ -17,10 +17,7 @@ pub fn main() {
|
|||||||
|
|
||||||
// Float => does not implement iterator.
|
// Float => does not implement iterator.
|
||||||
for i in 0f32..42f32 {}
|
for i in 0f32..42f32 {}
|
||||||
//~^ ERROR `core::iter::Iterator` is not implemented for the type `core::ops::Range<f32>`
|
//~^ ERROR the trait `core::num::Int` is not implemented for the type `f32`
|
||||||
//~^^ ERROR
|
|
||||||
//~^^^ ERROR
|
|
||||||
// FIXME(#21528) not fulfilled obligation error should be reported once, not thrice
|
|
||||||
|
|
||||||
// Unsized type.
|
// Unsized type.
|
||||||
let arr: &[_] = &[1u32, 2, 3];
|
let arr: &[_] = &[1u32, 2, 3];
|
||||||
|
28
src/test/run-pass/range-type-infer.rs
Normal file
28
src/test/run-pass/range-type-infer.rs
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
||||||
|
// file at the top-level directory of this distribution and at
|
||||||
|
// http://rust-lang.org/COPYRIGHT.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||||
|
// option. This file may not be copied, modified, or distributed
|
||||||
|
// except according to those terms.
|
||||||
|
|
||||||
|
// Make sure the type inference for the new range expression work as
|
||||||
|
// good as the old one. Check out issue #21672, #21595 and #21649 for
|
||||||
|
// more details.
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let xs = (0..8).map(|i| i == 1u64).collect::<Vec<_>>();
|
||||||
|
assert_eq!(xs[1], true);
|
||||||
|
let xs = (0..8).map(|i| 1u64 == i).collect::<Vec<_>>();
|
||||||
|
assert_eq!(xs[1], true);
|
||||||
|
let xs: Vec<u8> = (0..10).collect();
|
||||||
|
assert_eq!(xs.len(), 10);
|
||||||
|
|
||||||
|
for x in 0..10 { x % 2; }
|
||||||
|
for x in 0..100 { x as f32; }
|
||||||
|
|
||||||
|
let array = [true, false];
|
||||||
|
for i in 0..1 { array[i]; }
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user