swift语言为了简化,把指针隐形化了。没有像OC中那样的“ * ”。把底层的问题交给C语言去处理,我们可以在swift中调用C 语言来解决。当然,OC也是可以调用的。

        但是在某些场景下,这种调用可能不是很方便。比如,基于字节流的解析中,这时我们所接受的数据是要进行解析,可能用到指针。如果在C中去调用,当然也是可以的。但如果想写的简洁一些,用swift直接去处理这些,是否可以呢。那就要看在swift中是否很好的去使用指针呢。

       事实上,swift是支持使用指针的。苹果已经公开了swift的源码,这个大家都可以去看。源码是用C++写的。不过既然是一种新型的语言,又要避免OC走过的老路。所以肯定不能直接像在OC中那样直接使用指针了。在Swift 中指针被映射为了一个泛型类型,并且还比较抽象。这在一定程度上造成了在 Swift 中指针使用的困难。虽然如此,还是可以使用的,那么这就是使得字节流解析,在swift中变成了可能。

     一、指针的定义

        在swift中,我们如何定义指针呢。比如:

   let  tmpInt   =  20;
   let  ptr      =  &tmpInt;
   这样的话编译不通过。所以不能这么使用。
   
   比如在C语言里,分配内存空间,然后赋值,可以写成这样。
   
   先申请内存空间
   int * ptr = (int*)malloc(sizeof(int)*1);
   
   根据地址将数据写入申请到的内存空间
   *ptr = 10;
   
   这个我们用swift来写,写成下面的形式。
   var intPtr = UnsafeMutablePointer.alloc(1);
   print("intPtr = \(intPtr)");
   
   打印可以得到分配到空间的地址,一般来讲是4个字节的首地址。
   由此,我们可以看到,在swift中,主要使用UnsafeMutablePointer的几种形式,来
   进行指针方面的操作
   
   通过指针来给相应的内存空间赋值,在swift中如何去做呢?
   
   如同C的写法,直接访问内存空间
   intPtr.memory = 20;
   print("用C的方式赋值以后的值是 \(intPtr.memory)");
   
   通过打印,可以看到内存空间的值确实被改变了。
   
   除了用这种方式以外,还可以用另外的一种方式,先分配空间,然后再进行初始化
   
   var intPtr = UnsafeMutablePointer.alloc(1);
   intPtr.initialize(10);
   print("intPtr = \(intPtr)");
   print("intPtr value =\(intPtr.memory)");
   
   通过如上的代码,可以看到如同C一样可以使用。
   
二、内存的释放
   在C语言和C++中,我们申请堆空间的原则是“谁申请,谁释放”。在OC中,早期也是采用MRC的机制,后来才使用ARC的机制。那么在swift中,我们申请的内存空间,需不需释放呢?答案是显然需要释放。如同C的写法一样,我们先来看一下在C语言中的完整的写法。
   
   //申请内存空间,4个字节
   int * ptr = (int*)malloc(sizeof(int)*1);
   //通过指针,赋值
   *ptr = 30;
   使用完成,释放空间
    //释放ptr所指向的内存空间
    free(ptr);
    //将指针变量的值设置为空
    ptr = NULL;
    
    那么在swift中如何释放内存呢?我们来看完整的过程。
    
    //1.申请4字节的内存空间
    var intPtr = UnsafeMutablePointer.alloc(1);
    //2.初始化内存空间的值,或者直接给内存空间赋值
    intPtr.initialize(10);
    
    //使用完成以后,释放内存空间
    
    intPtr.destroy();  //1.用来销毁对象
    intPtr.dealloc(1); //2.释放指针所指向的内存空间
    intPtr  = nil;     //3.指针设为空指针

           由此可以看出,swift的内存释放过程比C语言的要复杂一些。


 三、在函数中的传参中使用

           1.C语言在函数传参中使用指针。这个我们先看C语言的一个小例子。如:           

    //定义一个int类型的变量a,并给a赋值
    int a = 10;
    //定义一个函数,并通过该函数来修改变量a的值。
    void changeData(int * tmpA,int value)
    {
        *tmpA = value;
    }
    
    //我们来调用该函数来修改变量a的值
    changeData(&a,100);
    //打印a的值
    printf("a = %d\n",a);
    
    2.swift中使用
    
    var tmp = 20;
    //在swfit中定义该函数
    func incrementor(ptr:UnsafeMutablePointer)
    {
       ptr.memory += 10;

    }
    //调用该函数
    incrementor(&tmp);
    print("tmp = \(tmp)");

    在swift中,还可以通过inout这关键字,在函数中使用
    //定义函数,使用关键字inout
    func testPointUse(inout num:Int){
       num += 1
    }
    //调用该函数,注意,这个地方的参数要传时加上取地址符号。
    testPointUse(&tmp);
    print("tmp = \(tmp)");  
    
四、指向数组的指针
    在swift中,如何用指针指向数组呢?
    //定义一个swift的数组
    var array = [1,2,3,4,5];
    //定义一个指向该数组的指针,参数是数组的地址和数组的count
    var arrayPtr = UnsafeMutableBufferPointer(start: &array, count: array.count)

    var basePtr = arrayPtr.baseAddress as UnsafeMutablePointer;
    print("basePtr.memory = \(basePtr.memory)");
    print("basePtr = \(basePtr)");

    basePtr.memory = 10;
    print("basePtr.memory = \(basePtr.memory)");
    print("basePtr = \(basePtr)");

    var nextPtr = basePtr.successor();
    print("nextPtr.memory = \(nextPtr.memory)");

         swift的指针使用较少,但在目前智能家电app的开发中,如果使用swift而不是OC的话,或者你的代码要从oc迁移到swift的话,那么这种指针方法的函数一定不少。当然在OC中,你是可以直接使用C的。这个没有问题。但你的项目迁移到swift以后,这些函数你就要在C中实现,然后用swift去调用,这样做当然没有错。就是有时这种调用感觉不爽,那你也可以用swift去试试。看能否把那些用C写的函数用swift改写了。这样,我们的项目中,对于新人来说,特别是那些C语言功底很薄弱的开发人员来说,不用每次去看C函数。修改也变得直接了。

        当然这只是我一家之言,大家对文中有不对的地方,如有时间,可以多交流。