namespace std::ranges {
template <permutable I,
sentinel_for<I> S>
constexpr subrange<I>
shift_right(I first,
S last,
iter_difference_t<I> n); // (1) C++23
template <forward_range R>
requires permutable<iterator_t<R>>
constexpr borrowed_subrange_t<R>
shift_right(R&& r,
range_difference_t<R> n); // (2) C++23
}
概要
範囲の要素をn個だけ右にシフトさせる。
- (1): イテレータ範囲を指定する
- (2): Rangeを直接指定する
この関数に符号付き整数型のシフト数として、0および負数を指定した場合はなにもしない。
この関数によって要素をn個だけ右にシフトすると、[first, first + n)
の範囲は、ムーブされたあとの「使用してはいけないオブジェクト」となる。その範囲には、循環バッファ (circular buffer) のように新たな要素を代入するか、コンテナのerase()
メンバ関数を使用して使わなくなった範囲を削除するなどの対応が必要になる。
事前条件
n >= 0
効果
n == 0
である場合、なにもしないn >= last - first
である場合、なにもしないi < (last - first) - n
である非負の各i
について、first + i
位置の要素をfirst + n + i
位置にムーブする- (1)では、
I
型がbidirectional_iterator
のモデルとなる場合は、i = (last - first) - n - 1
からi = 0
の順に処理する
- (1)では、
戻り値
new_first
を次のように定義する。
n < last - first
である場合、first + n
- そうでなければ、
last
このとき、 {new_first, last}
計算量
最大で(last - first) - n
回の代入もしくはswap操作を行う
備考
- シフト数として負数を指定することはできないが、この関数には符号付き整数型を指定することとなっている。これは、Bidirectional Iterator向けの最適化した実装をする場合に
std::prev()
関数を使用するため、そちらのパラメータ型と合わせたことによる shift_left()
とshift_right()
で関数が分かれているのは、コンパイルしたコードサイズを小さくするためと、左シフトと右シフトでは最大パフォーマンスのための実装が異なるためである
例
#include <iostream>
#include <vector>
#include <algorithm>
int main()
{
std::vector<int> v = {1, 2, 3, 4, 5};
std::ranges::shift_right(v, 2);
for (int x : v) {
std::cout << x << ',';
}
std::cout << std::endl;
}
出力
1,2,1,2,3,
バージョン
言語
- C++23
処理系
- Clang: ??
- GCC: ??
- ICC: ??
- Visual C++: ??