How to delete an element from a Slice in Golang

前端 未结 14 1448
时光取名叫无心
时光取名叫无心 2021-01-30 03:03
fmt.Println(\"Enter position to delete::\")
fmt.Scanln(&pos)

new_arr := make([]int, (len(arr) - 1))
k := 0
for i := 0; i < (len(arr) - 1); {
    if i != pos {
           


        
14条回答
  •  太阳男子
    2021-01-30 03:37

    This is a little strange to see but most answers here are dangerous and gloss over what they are actually doing. Looking at the original question that was asked about removing an item from the slice a copy of the slice is being made and then it's being filled. This ensures that as the slices are passed around your program you don't introduce subtle bugs.

    Here is some code comparing users answers in this thread and the original post. Here is a go playground to mess around with this code in.

    Append based removal

    package main
    
    import (
        "fmt"
    )
    
    func RemoveIndex(s []int, index int) []int {
        return append(s[:index], s[index+1:]...)
    }
    
    func main() {
        all := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        removeIndex := RemoveIndex(all, 5)
    
        fmt.Println("all: ", all) //[0 1 2 3 4 6 7 8 9 9]
        fmt.Println("removeIndex: ", removeIndex) //[0 1 2 3 4 6 7 8 9]
    
        removeIndex[0] = 999
        fmt.Println("all: ", all) //[999 1 2 3 4 6 7 9 9]
        fmt.Println("removeIndex: ", removeIndex) //[999 1 2 3 4 6 7 8 9]
    }
    

    In the above example you can see me create a slice and fill it manually with numbers 0 to 9. We then remove index 5 from all and assign it to remove index. However when we go to print out all now we see that it has been modified as well. This is because slices are pointers to an underlying array. Writing it out to removeIndex causes all to be modified as well with the difference being all is longer by one element that is no longer reachable from removeIndex. Next we change a value in removeIndex and we can see all gets modified as well. Effective go goes into some more detail on this.

    The following example I won't go into but it does the same thing for our purposes. And just illustrates that using copy is no different.

    package main
    
    import (
        "fmt"
    )
    
    func RemoveCopy(slice []int, i int) []int {
        copy(slice[i:], slice[i+1:])
        return slice[:len(slice)-1]
    }
    
    func main() {
        all := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        removeCopy := RemoveCopy(all, 5)
    
        fmt.Println("all: ", all) //[0 1 2 3 4 6 7 8 9 9]
        fmt.Println("removeCopy: ", removeCopy) //[0 1 2 3 4 6 7 8 9]
    
        removeCopy[0] = 999
        fmt.Println("all: ", all) //[99 1 2 3 4 6 7 9 9]
        fmt.Println("removeCopy: ", removeCopy) //[999 1 2 3 4 6 7 8 9]
    }
    

    The questions original answer

    Looking at the original question it does not modify the slice that it's removing an item from. Making the original answer in this thread the best so far for most people coming to this page.

    package main
    
    import (
        "fmt"
    )
    
    func OriginalRemoveIndex(arr []int, pos int) []int {
        new_arr := make([]int, (len(arr) - 1))
        k := 0
        for i := 0; i < (len(arr) - 1); {
            if i != pos {
                new_arr[i] = arr[k]
                k++
            } else {
                k++
            }
            i++
        }
    
        return new_arr
    }
    
    func main() {
        all := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        originalRemove := OriginalRemoveIndex(all, 5)
    
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        fmt.Println("originalRemove: ", originalRemove) //[0 1 2 3 4 6 7 8 9]
    
        originalRemove[0] = 999
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        fmt.Println("originalRemove: ", originalRemove) //[999 1 2 3 4 6 7 8 9]
    }
    

    As you can see this output acts as most people would expect and likely what most people want. Modification of originalRemove doesn't cause changes in all and the operation of removing the index and assigning it doesn't cause changes as well! Fantastic!

    This code is a little lengthy though so the above can be changed to this.

    A correct answer

    package main
    
    import (
        "fmt"
    )
    
    func RemoveIndex(s []int, index int) []int {
        ret := make([]int, 0)
        ret = append(ret, s[:index]...)
        return append(ret, s[index+1:]...)
    }
    
    func main() {
        all := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        removeIndex := RemoveIndex(all, 5)
    
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 8 9]
        fmt.Println("removeIndex: ", removeIndex) //[0 1 2 3 4 6 7 8 9]
    
        removeIndex[0] = 999
        fmt.Println("all: ", all) //[0 1 2 3 4 5 6 7 9 9]
        fmt.Println("removeIndex: ", removeIndex) //[999 1 2 3 4 6 7 8 9]
    }
    

    Almost identical to the original remove index solution however we make a new slice to append to before returning.

提交回复
热议问题