配列内の要素を選択し、それを別のインデックスに移動してから、「中間」要素を1ずつ「シフト/回転」したいと思います。
ドラッグアンドドロップ機能を想像してみてください。人差し指from
が人差し指to
より小さい場合は、左にシフトし、そうでない場合は右にシフトします。
入力:
_let data = [ 0, 1, 2, 3, 4, 5, 6 ]
_
タスク1:insertAndShift(data, 0, 3): Take element at index 0, shift indexes 1, 2, 3 to the left, and then insert index 0 at 3
期待される出力:
_[ 1, 2, 3, 0, 4, 5, 6 ]
_
タスク2:insertAndShift(data, 3, 0) Take element at index 3, shift indexes 0, 1, 2 to the right and then insert index 3 at 0
期待される出力:
_[ 0, 1, 2, 3, 4, 5, 6 ]
_
私はこれを試しました:
次の 配列を所定の位置に回転させる 、醜くて機能しないコード(一時的に常に未定義+結果が完全ではない+右でのみ機能する可能性がある):
_insertAndShift(data, from, to)
{
if(from < to)
{
// rotate in between - LEFT
}
else
{
// swap elements
let temp = data[from];
data[from] = data[to];
data[to] = temp;
// rotate in between - RIGHT
let newData = this.rotate(data.slice(to, from - 1), 1)
let result = [];
result.Push(data.slice(0,1));
newData.map((element) => result.Push(element))
for(let i = from+1; i < data.length; i++)
{
result.Push(data[i]);
}
this.setState({data: result})
}
}
rotate(arr, k) {
var l = arr.length;
arr = this.reverseArr(arr, 0, l - 1);
arr = this.reverseArr(arr, 0, k - 1);
arr = this.reverseArr(arr, k, l - 1);
return arr;
}
reverseArr(arr, left, right) {
while (left < right) {
var temp = arr[left];
arr[left] = arr[right];
arr[right] = temp;
left++;
right--;
}
return arr;
}
_
また、次の Javascript(動的)配列に挿入し、すべての要素を+1の下にシフトします これは1つの項目のみを返します:
_else
{
let result = data.splice(to, 1, data[from]);
this.setState({allTasksArray: result})
}
_
左と右の両方でこれをどのように達成できますか?
Array.prototype.splice
を使用して、必要な要素を切り取り、目的のインデックスに挿入できます(シフトすると自動的に発生します):
function insertAndShift(arr, from, to) {
let cutOut = arr.splice(from, 1) [0]; // cut the element at index 'from'
arr.splice(to, 0, cutOut); // insert it at index 'to'
}
let data = [ 0, 1, 2, 3, 4, 5, 6 ];
insertAndShift(data, 0, 3);
console.log("[" + data + "]");
insertAndShift(data, 3, 0);
console.log("[" + data + "]");
上記の答えはうまく機能しますが、少し波打っています...それが何をしているのかを詳しく説明するソリューションに興味があるなら、私はこれを書きました:
const insertAndShift = (arr: number[], to: number, from: number) => {
let newArray: number[] = [];
const fromItem = arr[from];
if (from > to) {
const startToTo = (to > 0) ? arr.slice(0, to) : [];
const toToFrom = arr.slice(to, from);
const fromToEnd = arr.slice(from + 1, arr.length);
newArray = newArray.concat(startToTo, [fromItem], toToFrom, fromToEnd);
}
if (to > from) {
const startToFrom = (from > 0) ? arr.slice(0, from) : [];
const fromToTo = arr.slice(from + 1, to + 1);
const toToEnd = arr.slice(to + 1, arr.length);
newArray = newArray.concat(startToFrom, fromToTo, fromItem, toToEnd);
}
return newArray;
};
const array = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
console.log(insertAndShift(array, 9, 0));
console.log(insertAndShift(array, 5, 1));
このメソッドを見てください: Array.prototype.copyWithin() 。
それを行うためのより効率的なインプレース方法があります:
function insertAndShift(data, src, dest) {
if (src === dest)
return;
let tmp = data[src];
if (src < dest) {
data.copyWithin(src, src + 1, dest + 1);
} else /* if (src > dest) */ {
data.copyWithin(dest + 1, dest, src);
}
data[dest] = tmp;
}
let data = [0, 1, 2, 3, 4, 5, 6];
insertAndShift(data, 0, 3);
console.log("[" + data + "]");
insertAndShift(data, 3, 0);
console.log("[" + data + "]");
以下はイラストです:
Case 1 (src < dest):
[..Sabcd..]
//// // data.copyWithin(src, src+1, dest+1);
[..abcdd..]
[..abcdS..] // data[dest] = tmp;
Case 2 (src > dest):
[..abcdS..]
\\\\ // data.copyWithin(dest+1, dest, src);
[..aabcd..]
[..Sabcd..] // data[dest] = tmp;