package main

/*
#include <stdlib.h>
*/
import "C"
import (
"unsafe"
"fmt"
) type Slice struct {
Data unsafe.Pointer //万能指针类型 对应C语言中的void*
len int //有效的长度
cap int //有效的容量
} const TAG = 8 /*
func main() {
//定义一个切片
//1、数据内存地址 2、len 有效数据长度 3、cap 可扩容的有效容量 24字节
var s []int //unsafe.Sizeof 计算数据类型在内存中占的字节大小
fmt.Println(unsafe.Sizeof(s))
}
*/
/*
func main(){
var i interface{}
i=10//只支持== != //类型断言 是基于接口类型数据的转换
//value,ok:=i.(int)
//if ok{
// fmt.Println("整型数据:",value)
// fmt.Printf("%T\n",value)
//}
//反射获取接口的数据类型
//t:=reflect.TypeOf(i)
//fmt.Println(t) //反射获取接口类型数据的值
v:=reflect.ValueOf(i)
fmt.Println(v) i1:=10
i2:=20 if reflect.Typeof(i1)==reflect.Typeof(i2){
v1:=reflect.Valueof(i1)
v2:=reflect.Valueof(i2)
结果=v1+v2
}
}
*/
//Create(长度 容量 数据)
func (s *Slice) Create(l int, c int, Data ...int) {
//如果数据为空返回
if len(Data) == 0 {
return
}
//长度小于0 容量小于0 长度大于容量 数据大于长度
if l < 0 || c < 0 || l > c || len(Data) > l {
return
}
//ulonglong unsigned long long 无符号的长长整型
//通过C语言代码开辟空间 存储数据
//如果堆空间开辟失败 返回值为NULL 相当于nil 内存地址编号为0的空间
s.Data = C.malloc(C.ulonglong(c) * 8)
s.len = l
s.cap = c //转成可以计算的指针类型
p := uintptr(s.Data)
for _, v := range Data {
//数据存储
*(*int)(unsafe.Pointer(p)) = v
//指针偏移
p += TAG
//p+=unsafe.Sizeof(1)
}
} //Print 打印切片
func (s *Slice) Print() {
if s == nil {
return
} //将万能指针转成可以计算的指针
p := uintptr(s.Data)
for i := 0; i < s.len; i++ {
//获取内存中的数据
fmt.Print(*(*int)(unsafe.Pointer(p)), " ")
p += TAG
} } //切片追加
func (s *Slice) Append(Data ...int) {
if s == nil {
return
}
if len(Data) == 0 {
return
} //如果添加的数据超出了容量
if s.len+len(Data) > s.cap {
//扩充容量
//C.realloc(指针,字节大小), go 语言 2 倍扩容。
s.Data = C.realloc(s.Data, C.ulonglong(s.cap)*2*8)
//改变容量的值
s.cap = s.cap * 2
} p := uintptr(s.Data)
for i := 0; i < s.len; i++ {
//指针偏移
p += TAG
} //添加数据
for _, v := range Data {
*(*int)(unsafe.Pointer(p)) = v
p += TAG
}
//更新有效数据(长度)
s.len = s.len + len(Data)
} //获取元素 GetData(下标) 返回值为int 元素
func (s *Slice) GetdData(index int) int {
if s == nil || s.Data == nil {
return 0
}
if index < 0 || index > s.len-1 {
return 0
} p := uintptr(s.Data)
for i := 0; i < index; i++ {
p += TAG
}
return *(*int)(unsafe.Pointer(p))
} //查找元素 Search(元素)返回值为int 下标
func (s *Slice) Search(Data int) int {
if s == nil || s.Data == nil {
return -1
} p := uintptr(s.Data)
for i := 0; i < s.len; i++ {
//查找数据 返回第一次元素出现的位置
if *(*int)(unsafe.Pointer(p)) == Data {
return i
}
//指针偏移
p += TAG
}
return -1
} //删除元素 Delete(下标)
func (s *Slice) Delete(index int) {
if s == nil || s.Data == nil {
return
}
if index < 0 || index > s.len-1 {
return
} //将指针指向需要删除的下标位置
p := uintptr(s.Data)
for i := 0; i < index; i++ {
p += TAG
} //用下一个指针对应的值为当前指针对应的值进行赋值
temp := p
for i := index; i < s.len; i++ {
temp += TAG
*(*int)(unsafe.Pointer(p)) = *(*int)(unsafe.Pointer(temp))
p += TAG
} s.len--
} //插入元素 Insert(下标 元素)
func (s *Slice) Insert(index int, Data int) {
if s == nil || s.Data == nil {
return
}
if index < 0 || index > s.len-1 {
return
} //如果插入数据是最后一个
//if index == s.len-1 {
// p := uintptr(s.Data)
//
// //for i := 0; i < s.len; i++ {
// // p += TAG
// //}
// p += TAG * uintptr(s.len-1)
// *(*int)(unsafe.Pointer(p)) = Data
// s.len++
// return
//}
//调用追加方法
if index == s.len-1 {
s.Append(Data)
return
} //获取插入数据的位置
p := uintptr(s.Data)
for i := 0; i < index; i++ {
p += TAG
}
//获取末尾的指针位置 temp := uintptr(s.Data)
temp += TAG * uintptr(s.len) //将后面数据依次向后移动
for i := s.len; i > index; i-- {
//用前一个数据为当前数据赋值
*(*int)(unsafe.Pointer(temp)) = *(*int)(unsafe.Pointer(temp - TAG))
temp -= TAG
} //修改插入下标的数据
*(*int)(unsafe.Pointer(p)) = Data
s.len++
} //销毁切片
func (s *Slice) Destroy() {
//调用C语言 适释放堆空间
C.free(s.Data)
s.Data = nil
s.len = 0
s.cap = 0
s = nil
}

slice 实现原理的更多相关文章

  1. golang slice 切片原理

    golang 中的 slice 非常强大,让数组操作非常方便高效.在开发中不定长度表示的数组全部都是 slice .但是很多同学对 slice 的模糊认识,造成认为golang中的数组是引用类型,结果 ...

  2. [js] Array.slice和类数组转数组

    a.call(b) 相当于把a方法放到b的原型上(实例私有方法)执行 Array.slice的用途 https://juejin.im/post/5b20b8596fb9a01e8d6a47c0 用法 ...

  3. slice 切片实现 Slice object interface

    1.Python切片对象可以为任意类型 https://github.com/python/cpython/blob/master/Include/sliceobject.h /* Slice obj ...

  4. 03. Go 语言容器

    Go语言容器(container) 变量在一定程度上能满足函数及代码要求.如果编写一些复杂算法.结构和逻辑,就需要更复杂的类型来实现.这类复杂类型一般情况下具有各种形式的存储和处理数据的功能,将它们称 ...

  5. GO学习笔记 - 数据校验

    本文主题:基于asaskevich/govalidator实现Golang数据校验 小慢哥的原创文章,欢迎转载 目录 ▪ 一. asaskevich/govalidator介绍 ▪ 二. 字符串匹配 ...

  6. 面试官问:Go 中的参数传递是值传递还是引用传递?

    一个程序中,变量分为变量名和变量内容,变量内容的存储一般会被分配到堆和栈上.而在 Go 语言中有两种传递变量的方式值传递和引用传递.其中值传递会直接将变量内容附在变量名上传递,而引用传递会将变量内容的 ...

  7. 深度学习实践-物体检测-faster-RCNN(原理和部分代码说明) 1.tf.image.resize_and_crop(根据比例取出特征层,进行维度变化) 2.tf.slice(数据切片) 3.x.argsort()(对数据进行排列,返回索引值) 4.np.empty(生成空矩阵) 5.np.meshgrid(生成二维数据) 6.np.where(符合条件的索引) 7.tf.gather取值

    1. tf.image.resize_and_crop(net, bbox, 256, [14, 14], name)  # 根据bbox的y1,x1,y2,x2获得net中的位置,将其转换为14*1 ...

  8. 【前端基础系列】slice方法将类数组转换数组实现原理

    问题描述 在日常编码中会遇到将类数组对象转换为数组的问题,其中常用到的一种方式使用Array.prototype.slice()方法. 类数组对象 所谓的类数组对象,JavaScript对它们定义为: ...

  9. [转载]Array.prototype.slice.call(arguments,1)原理

    Array.prototype.slice.call(arguments,1)该语句涉及两个知识点. arguments是一个关键字,代表当前参数,在javascript中虽然arguments表面上 ...

随机推荐

  1. 论文阅读笔记(二十二)【CVPR2017】:See the Forest for the Trees: Joint Spatial and Temporal Recurrent Neural Networks for Video-based Person Re-identification

    Introduction 在视频序列中,有些帧由于被严重遮挡,需要被尽可能的“忽略”掉,因此本文提出了时间注意力模型(temporal attention model,TAM),注重于更有相关性的帧. ...

  2. MySQL 8 mysql system schema

    在大的分类上:mysql schema包括存储数据库对象元数据的数据字典表和用于其他操作目的的系统表 数据字典表和系统表一般使用InnoDB存储引擎 与之前的版本不同,数据字典表和系统表存储在数据目录 ...

  3. 安全 - 内容安全策略(CSP)(未完)

    威胁 跨站脚本攻击(Cross-site scripting) 跨站脚本攻击Cross-site scripting (XSS)是一种安全漏洞,攻击者可以利用这种漏洞在网站上注入恶意的客户端代码. 攻 ...

  4. Supermarket POJ - 1456 贪心+并查集

    #include<iostream> #include<algorithm> using namespace std; const int N=1e5; struct edge ...

  5. Docker造化钟神秀

    之前闲暇之余有玩过一哈Dokcer容器,但是日子挺久了,然后挺多东西就忘记了,趁着今天有时间重拾一下docker的相关知识. 搜索下载镜像 docker search ubuntu docker pu ...

  6. python基础(1):第一个python程序的编写

    1.第一个python编程 1.1 python的安装 1> https://www.python.org/  进入python官网,选择目标版本进行download 2> 点击setup ...

  7. C# Dynamic与Newtonsoft.Json的应用

    C#中Dynamic关键字 dynamic关键字和动态语言运行时(DLR)是.Net 4.0中新增的功能. 什么是"动态"? 编程语言有时可以划分为静态类型化语言和动态类型化语言. ...

  8. 【转】IEnumerable接口的一个简单示例

    转:https://blog.csdn.net/wang371756299/article/details/8197075 摘要: IEnumerable接口是实现foreach循环的一个重要的接口, ...

  9. CodeForces 1144C

    链接 https://vjudge.net/problem/CodeForces-1144C #include<bits/stdc++.h> using namespace std; in ...

  10. 2019ICPC南昌站C.And and Pair

    题意: 给一个二进制数n,求满足0<=j<=i<=n且i&&n==i&&i&j==0的数对(i,j)有多少对,n可能有前导0. 解析: 对一一 ...