ZooKeeper实践方案:(7) 分布式锁

ZooKeeper实践方案:(7) 分布式锁

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

1.基本介绍

分布式锁是控制分布式系统之间同步訪问共享资源的一种方式,须要相互排斥来防止彼此干扰来保证一致性。

利用Zookeeper的强一致性能够完毕锁服务。Zookeeper的官方文档是列举了两种锁。独占锁和共享锁。

独占锁保证不论什么时候都仅仅有一个进程能或者资源的读写权限。共享锁能够同一时候有多个读,可是同一时刻最多仅仅能有一个写,读和写是相互排斥的。

2.场景分析

我们准备来实现相互排斥的锁,依照官网的思路,给定一个锁的路径,如/Lock,全部要申请这个锁的进程都在/Lock文件夹下创建一个/Lock/lock-的暂时序列节点,并监控/Lock的子节点变化事件。当子节点发送变化时用get_children()获取子节点的列表,假设发现进程发现自己拥有最小的一个序号,则获得锁。

处理业务完成后须要释放锁,此时仅仅须要删除该暂时节点就可以。简单来说就是永远是拥有最小序号的进程获得锁。

3.场景实践

使用锁有两个主要的函数,就是lockunlock.定义为

  • Lock *lock(zhandle_t *zkhandle,const char *path)
    lock函数有两个參数,一个是zookeeper_init返回的句柄zkhandle,还有一个是锁的路径,假设成功则返回一个Lock的结构体指针。并同一时候获得锁,否则返回NULL。
  • int unlock(zhandle_t *zkhandle,Lock * *lock)
    unlock函数也有两个參数。一个是zookeeper_init返回的句柄zkhandle,还有一个是lock函数返回的结构体指针的指针

接下来在看详细的实现。

Lock *lock(zhandle_t *zkhandle,const char *path){    Lock *lock = create_lock(zkhandle,path);    if(lock != NULL){        while(try_lock(zkhandle,lock) == 0){            sleep(1);        }    }else{        fprintf(stderr,"error when create lock %s.\n",path);    }    return lock;}

  • create_lock:负责锁的初始化。主要功能是负责创建{path}的节点已经{path}/lock-的暂时序列节点。

    {path}假设存在则不再创建。

  • try_lock:尝试加锁,这个函数不会等待,失败和成功都马上返回。其主要功能是获取{path}的子节点列表,并查看自己是否是拥有最小序列号的节点。假设是则返回1,否则返回0;

lock函数初始化锁后,会持续的尝试加锁,直到成功。尽管我是这样实现的。可是过于简单粗暴(哈哈)。假设拿不到锁的话。持续就会堵塞在lock函数。

int unlock(zhandle_t *zkhandle,Lock * *lock){    if(*lock){        int ret = zoo_delete(zkhandle,(*lock)->selfpath,-1);        if(ret != ZOK){            fprintf(stderr,"error when release lock %s.\n",(*lock)->selfpath);        }        free(*lock);        *lock = NULL;        return ret;    }    return ZOK;}

unlock函数就很easy了。就是将create_lock中创建的暂时序列节点删除就能够了。

接下来在看下模拟程序的功能。

> ./mylock -hUsage : [mylock] [-h]  [-p path][-s ip:port]         -h Show help        -p lock path        -s zookeeper server ip:portFor example:    mylock -s 172.17.0.36:2181 -p /Lock

模拟程序有3个选项。

当中
-s:为Zookeeper的server的ip:port.
-p: 为锁的路径。

分别同一时候执行多个mylock程序,就能够看到各个程序之间是怎样获取锁的了。

最后是完整的代码:

#include<stdio.h>  #include<string.h>  #include<unistd.h>#include"zookeeper.h"  #include"zookeeper_log.h"  char g_host[512]= "172.17.0.36:2181";  char g_path[512]= "/Lock";typedef struct Lock{    char lockpath[1024];    char selfpath[1024];}Lock;void print_usage();void get_option(int argc,const char* argv[]);/**********unitl*********************/  void print_usage(){    printf("Usage : [mylock] [-h]  [-p path][-s ip:port] \n");    printf("        -h Show help\n");    printf("        -p lock path\n");    printf("        -s zookeeper server ip:port\n");    printf("For example:\n");    printf("    mylock -s172.17.0.36:2181 -p /Lock\n");}void get_option(int argc,const char* argv[]){    extern char    *optarg;    int            optch;    int            dem = 1;    const char    optstring[] = "hp:s:";    while((optch = getopt(argc , (char * const *)argv , optstring)) != -1 )    {        switch( optch )        {        case 'h':            print_usage();            exit(-1);        case '?':            print_usage();            printf("unknown parameter: %c\n", optopt);            exit(-1);        case ':':            print_usage();            printf("need parameter: %c\n", optopt);            exit(-1);        case 's':            strncpy(g_host,optarg,sizeof(g_host));            break;        case 'p':            strncpy(g_path,optarg,sizeof(g_path));            break;        default:            break;        }    }} Lock *create_lock(zhandle_t *zkhandle,const char *path){    char path_buffer[512]={0};    int bufferlen = sizeof(path_buffer);    Lock * lock = NULL;    int ret = zoo_exists(zkhandle,path,0,NULL);     if(ret != ZOK){        ret = zoo_create(zkhandle,path,"1.0",strlen("1.0"),                            &ZOO_OPEN_ACL_UNSAFE,0,                            path_buffer,bufferlen);          if(ret != ZOK){            fprintf(stderr,"failed to create the path %s!\n",path);        }else{            printf("create path %s successfully!\n",path);        }    }    if(ret == ZOK){        char child_path[512];        sprintf(child_path,"%s/lock-",path);        ret = zoo_create(zkhandle,child_path,"1.0",strlen("1.0"),                            &ZOO_OPEN_ACL_UNSAFE,ZOO_SEQUENCE|ZOO_EPHEMERAL,                            path_buffer,bufferlen);          if(ret != ZOK){            fprintf(stderr,"failed to create the path %s!\n",path);        }else{            printf("create path %s successfully!\n",path);        }    }    if(ret == ZOK){        lock = (Lock *)malloc(sizeof(Lock));        strcpy(lock->lockpath,path);        strcpy(lock->selfpath,path_buffer);    }    return lock;}int try_lock(zhandle_t *zkhandle,Lock *lock){    struct String_vector children;    int i = 0;    int ret = zoo_get_children(zkhandle,lock->lockpath,0,&children);    if(ret != ZOK){        fprintf(stderr,"error when get children of path %s\n",lock->lockpath);        ret = -1;    }else{        char *myseq = rindex(lock->selfpath,'/');        if (myseq != NULL) myseq += 1;        ret = 1;        for(i = 0; i < children.count; ++i){            if(strcmp(children.data[i],myseq) < 0){                ret = 0;                break;            }                    }        for(i = 0; i < children.count; ++i){            free(children.data[i]);            children.data[i] = NULL;        }    }    return ret;}Lock *lock(zhandle_t *zkhandle,const char *path){    Lock *lock = create_lock(zkhandle,path);    if(lock != NULL){        while(try_lock(zkhandle,lock) == 0){            sleep(1);        }    }else{        fprintf(stderr,"error when create lock %s.\n",path);    }    return lock;}int unlock(zhandle_t *zkhandle,Lock * *lock){    if(*lock){        int ret = zoo_delete(zkhandle,(*lock)->selfpath,-1);        if(ret != ZOK){            fprintf(stderr,"error when release lock %s.\n",(*lock)->selfpath);        }        free(*lock);        *lock = NULL;        return ret;    }    return ZOK;}int main(int argc, const char *argv[])  {      int timeout = 30000;      char path_buffer[512];      int bufferlen=sizeof(path_buffer);      zoo_set_debug_level(ZOO_LOG_LEVEL_WARN); //设置日志级别,避免出现一些其它信息      get_option(argc,argv);    zhandle_t* zkhandle = zookeeper_init(g_host,NULL, timeout, 0, (char *)"lock Test", 0);      if (zkhandle ==NULL)      {          fprintf(stderr, "Error when connecting to zookeeper servers...\n");          exit(EXIT_FAILURE);      }      int ret = zoo_exists(zkhandle,g_path,0,NULL);     if(ret != ZOK){        ret = zoo_create(zkhandle,g_path,"1.0",strlen("1.0"),                            &ZOO_OPEN_ACL_UNSAFE,0,                            path_buffer,bufferlen);          if(ret != ZOK){            fprintf(stderr,"failed to create the path %s!\n",g_path);        }else{            printf("create path %s successfully!\n",g_path);        }    }    if(ret == ZOK ){       Lock *mylock = lock(zkhandle,g_path);        if(mylock){            printf("get lock of %s.\n",g_path);            printf("self path is %s.\n",mylock->selfpath);            printf("do something....\n");            getchar();            unlock(zkhandle,&mylock);        }        }    zookeeper_close(zkhandle);     return 0;}

版权声明:本文博客原创文章,博客,未经同意,不得转载。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/117603.html原文链接:https://javaforall.net

(0)
上一篇 2022年1月3日 上午11:00
下一篇 2022年1月3日 上午11:00


相关推荐

  • JavaScript 闭包详解

    JavaScript 闭包详解JavaScript闭包文章目录JavaScript闭包一、为什么要闭包二、外部得以访问函数内变量三、某些变量得以常驻内存1.垃圾回收机制对闭包的处理2.结合立即执行函数来保存某些变量总结#前言##1.什么是闭包函数闭包函数是声明在另一个函数内的函数,是被嵌套在父函数内部的子函数,在《JS高级程序设计-第3版》中对闭包解释是:”闭包是指有权访问另外一个函数作用域中的变量的函数.”闭包函数可以访问[包裹其的函数]内的各种参数和变量,即便外部函数已经执行完毕.(至于为什么请看下文).一、为什么

    2022年6月28日
    24
  • kafka零拷贝原理_kafka常用命令

    kafka零拷贝原理_kafka常用命令Kafka除了具备消息队列MQ的特性和使用场景外,它还有一个重要用途,就是做存储层。用kafka做存储层,为什么呢?一大堆可以做数据存储的MySQL、MongoDB、HDFS……因为kafka数据是持久化磁盘的,还速度快;还可靠、支持分布式……啥!用了磁盘,还速度快!!!没错,kafka就是速度无敌,本文将探究kafka无敌性能背后的秘密。首先要有个概念,kafka高性能的背后,是多方面协同后、最终的结果,kafka从宏观架构、分布式partition存储、ISR数据同步、以及“无孔不

    2026年2月7日
    8
  • vb程序设计第四版实验报告答案_vb程序设计第四版课后答案刘炳文

    vb程序设计第四版实验报告答案_vb程序设计第四版课后答案刘炳文您所在位置:网站首页>海量文档&nbsp>&nbsp计算机&nbsp>&nbspVisualBasicVisualBasic程序设计教程(第4版)实验答案.doc129页本文档一共被下载:次,您可全文免费在线阅读后下载本文档。下载提示1.本站不保证该用户上传的文档完整性,不预览、不比对内容而直接下载产生的反…

    2022年10月7日
    4
  • git回退版本命令

    git回退版本命令如果你在本地做了错误提交 那么回退版本的方法很简单 1 先用下面命令找到要回退的版本的 commitid gitreflog2 接着回退版本 gitresethard 就是你要回退的版本的 commitid 的前面几位 远程分支版本回退的方法如果你的错误提交已经推送到自己的远程分支了 那么就需要回滚远程分支了 1 首先要回退本地分支 gitrefloggit 紧接着强制推送到远程分支 gi

    2025年6月19日
    6
  • 大话数据结构学习心得

    大话数据结构学习心得想重温一下数据结构和算法,选择了大话数据结构这本书。本书用趣味的方式介绍了数据结构起源、算法设计,线性表、栈与队列、串、树、图、查找、排序。对于当前用高级语言(java,c#,python等)开发的软件开发人员来说可能相关内容涉及不到,因为高级语言已经封装好了相关方法。但是了解了计算机内存存储、查找、排序等算法对于开发人员来说会有一个新的认识:例如如何优化方法提高存储速度、查询速度等。附:…

    2022年6月24日
    30
  • 辛星PHP教程之yii和ci教程已经写完,望与朋友们交流[通俗易懂]

    辛星PHP教程之yii和ci教程已经写完,望与朋友们交流

    2022年1月22日
    50

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号