从零开始写android 的智能指针

Android中定义了两种智能指针类型,一种是强指针sp(strong pointer),源码中的位置在system/core/include/utils/StrongPointer.h。另外一种是弱指针(weak pointer)。其实称之为强引用和弱引用更合适一些。强指针与一般意义的智能指针概念相同,通过引用计数来记录有多少使用者在使用一个对象,如果所有使用者都放弃了对该对象的引用,则该对象将被自动销毁。

弱指针也指向一个对象,但是弱指针仅仅记录该对象的地址,不能通过弱指针来访问该对象,也就是说不能通过弱指针来调用对象的成员函数或访问对象的成员变量。要想访问弱指针所指向的对象,需首先通过wp类所提供的promote()方法将弱指针升级为强指针。弱指针所指向的对象是有可能在其它地方被销毁的,如果对象已经被销毁,wp的promote()方法将返回空指针,这样就能避免出现地址访问错的情况。

强弱智能是用来解决什么问题的呢?

1.问题是如何产生的,当我们用new 的时候,如果忘了delet。

复制代码
#include <iostream>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>

using namespace std;

class Person{
public:
    string  name;
    Person(string n){
        name =n;
        cout<<"build person:"<<name<<endl;
    }
    void printInfo(){
        cout<<"i am "<<name<<endl;
    }
    ~Person(){
        cout<<"release person:"<<name<<endl;
    }
};
void test_func(void)
{
    Person *p = new Person("zhangsan");
    p->printInfo();
//    Person p = Person("zhangsan");
//    p.printInfo();
}

int main(int argc, char **argv)
{
    int i;

    for (i = 0; i < 2; i++)
        test_func();
    return 0;
}

那么问题产生了。

有内存泄露。

2.开始改进,使用局部变量

Person *p = new Person("zhangsan" );
p->printInfo();
// Person p = Person("zhangsan");
// p.printInfo();

析构函数执行,内存泄露没有了。

结合以上两点,继续改进。

3. 在一个类中包含另一个类的指针,在执行这个类的析构函数时, 删除另一个类的指针。

复制代码
#include <iostream>
#include <string.h>

using namespace std;

class Person{
public:

    Person(){
        cout<<"build person"<<endl;
    }
    ~Person(){
        cout<<"release person"<<endl;
    }
    void printInfo(){
        cout<<"print person"<<endl;
    }
};

class sp {
private:
    Person *p;

public:
    sp() : p(0) {}

    sp(Person *other)
    {
        cout<<"sp( Person *other)"<<endl;
        p = other;
    }

    sp(const sp &other)
    {
        cout<<"sp(const Person *other)"<<endl;
        p = other.p;
    }

    ~sp()
    {
        cout<<"~sp()"<<endl;
        if (p)
            delete p;
    }

    Person *operator->()
    {
        return p;
    }
};

void test_func(sp &other)
{
    sp s = other;
    s->printInfo();
}

int main(int argc, char **argv)
{

    sp other = new Person();
    test_func(other);

    cout<<"main() end"<<endl;
    return 0;
}

sp出现了。但是还有问题, 在上面的程序中 person被创建了一次,但是被release了两次。会产生错误。

4. 解决思路:引入计数,当是最后一个使用者,销毁p。

复制代码
#include <iostream>
#include <string.h>

using namespace std;

class Person{

private:
    int count=0;
public:
    
    Person(){
        cout<<"build person"<<endl;
    }
    ~Person(){
        cout<<"release person"<<endl;
    }
    void printInfo(){
        cout<<"print person"<<endl;
    }

    void incStrong(){
        count +=1;
    }

    void decStrong(){
        count -=1;
    }

    int getStrongCount(){
        return  count;
    }

};

class sp {
private:
    Person *p;

public:
    sp() : p(0) {
        cout<<"init sp"<<endl;
    }

    sp(Person *other)
    {
        cout<<"sp( Person *other)"<<endl;
        p = other;
        p->incStrong();
    }

    sp(const sp &other)
    {
        cout<<"sp(const Person *other)"<<endl;
        p = other.p;
        p->incStrong();
    }

    
    ~sp()
    {
        cout<<"~sp()"<<endl;
        if (p){
            p->decStrong();
            cout<<" ~"<<p->getStrongCount()<<endl;
            if( p->getStrongCount() ==0){
                delete p;
                p=NULL;
                cout<<"delete p"<<endl;
            }
        }
    }

    Person *operator->()
    {
        cout<<"->"<<endl;
        return p;
    }
};

void test_func(sp &other)
{
    sp s = other;
    s->printInfo();
}

int main(int argc, char **argv)
{

    sp other = new Person();
    test_func(other);

    cout<<"main() end"<<endl;
    return 0;
}

5. 将计数count 封装成一个基类。

6.在sp类中,引入模板。

复制代码
using namespace std;

class RefBase{
private:
    int count;
public:
    RefBase():count(0){}
    void incStrong(){count++;}
    void decStrong(){count--;}
    int getStrongCount(){return count;}
};

class Person:public RefBase{

public:
    
    Person(){
        cout<<"build person"<<endl;
    }
    ~Person(){
        cout<<"release person"<<endl;
    }
    void printInfo(){
        cout<<"print person"<<endl;
    }


};
template <typename T>
class sp {
private:
    T *p;

public:
    sp() : p(0) {
        cout<<"init sp"<<endl;
    }

    sp(T *other)
    {
        cout<<"sp( Person *other)"<<endl;
        p = other;
        p->incStrong();
    }

    sp(const sp &other)
    {
        cout<<"sp(const Person *other)"<<endl;
        p = other.p;
        p->incStrong();
    }

    
    ~sp()
    {
        cout<<"~sp()"<<endl;
        if (p){
            p->decStrong();
            cout<<" ~"<<p->getStrongCount()<<endl;
            if( p->getStrongCount() ==0){
                delete p;
                p=NULL;
                cout<<"delete p"<<endl;
            }
        }
    }

    T *operator->()
    {
        cout<<"->"<<endl;
        return p;
    }
};

void test_func(sp<Person> &other)
{
    sp<Person> s = other;
    s->printInfo();
}

int main(int argc, char **argv)
{

    sp<Person> other = new Person();
    test_func(other);

    cout<<"main() end"<<endl;
    return 0;
}

一个智能指针初见雏形了。

相关推荐
问道飞鱼8 分钟前
【移动端知识】移动端多 WebView 互访方案:Android、iOS 与鸿蒙实现
android·ios·harmonyos·多webview互访
aningxiaoxixi1 小时前
Android 之 audiotrack
android
枷锁—sha1 小时前
【DVWA系列】——CSRF——Medium详细教程
android·服务器·前端·web安全·网络安全·csrf
Cao_Shixin攻城狮4 小时前
Flutter运行Android项目时显示java版本不兼容(Unsupported class file major version 65)的处理
android·java·flutter
呼啦啦呼啦啦啦啦啦啦7 小时前
利用pdfjs实现的pdf预览简单demo(包含翻页功能)
android·javascript·pdf
idjl9 小时前
Mysql测试题
android·adb
游戏开发爱好者811 小时前
iOS App 电池消耗管理与优化 提升用户体验的完整指南
android·ios·小程序·https·uni-app·iphone·webview
人生游戏牛马NPC1号12 小时前
学习 Flutter (四):玩安卓项目实战 - 中
android·学习·flutter
星辰也为你祝福h13 小时前
Android原生Dialog
android
梁同学与Android14 小时前
Android ---【CPU优化】需要优化的原因及优化的地方
android