web-dev-qa-db-ja.com

配列内の要素の位置を変更し、その間で要素をシフトします

配列内の要素を選択し、それを別のインデックスに移動してから、「中間」要素を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})
}
_

左と右の両方でこれをどのように達成できますか?

6
Khalil Khalaf

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 + "]");
19
ibrahim mahrir

上記の答えはうまく機能しますが、少し波打っています...それが何をしているのかを詳しく説明するソリューションに興味があるなら、私はこれを書きました:

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));
1
Devin Clark

このメソッドを見てください: 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;
1
Raku Zeta