评论

收藏

[NoSQL] (1)redis下载编译

数据库 数据库 发布于:2021-07-30 18:28 | 阅读数:488 | 评论:0

一、redis下载编译这里没什么好说的
用的版本是redis-2.8.17
1)redis-server是可执行程序
2)mian函数在redis.c里面
3)如果要修改调试 这届在src目录下   修改后make或者make clean;make 就行
从main函数说起这里先说两个部分一个是  redis里面的回调函数  还有一个是redis里面的log日志
二、redis里的回调函数先看下代码;这是把redis里面的回调函数拿出来修改下
/*
redis里的回调函数
*/
#include<stdio.h>
#include<stdlib.h>
  
  
static void zmalloc_default_oom(size_t size) 
{
  printf("zmalloc_default_oom\n");
  fprintf(stderr, "zmalloc: Out of memory trying to allocate %d bytes\n",size);
  fflush(stderr);
}
  
static void (*zmalloc_oom_handler)(size_t) = zmalloc_default_oom;
  
  
void zmalloc_set_oom_handler(void (*oom_handler)(size_t)) 
{
  printf("zmalloc_set_oom_handler\n");
  zmalloc_oom_handler = oom_handler;
}
  
void redisOutOfMemoryHandler(size_t allocation_size) 
{
  printf("redisOutOfMemoryHandler------:%d\n",allocation_size);
}
  
int main(void)
{
  //zmalloc_set_oom_handler(redisOutOfMemoryHandler);
  zmalloc_oom_handler(10);
  getchar();
  return 0;
}
运行结果
zmalloc_default_oom
zmalloc:Out of memory trying to allocate 10 bytes
我们可以看到默认情况下,在没有注册回调函数的情况下zmalloc_oom_handler是指向  zmalloc_default_oom函数的
假如注册了回调函数的情况下,则调用的是 注册了的回调函数
int main(void)
{
  zmalloc_set_oom_handler(redisOutOfMemoryHandler);
  zmalloc_oom_handler(10);
  getchar();
  return 0;
}
运行结果
zmalloc_set_oom_handler
redisOutOfMemoryHandler----------:10
现在看看redis的代码
int main(int argc, char **argv) 
{
  struct timeval tv;
         
  /* We need to initialize our libraries, and the server configuration. */
#ifdef INIT_SETPROCTITLE_REPLACEMENT
  //初始化参数
  spt_init(argc, argv);
#endif
  setlocale(LC_COLLATE,"");
  /*
  zmalloc_enable_thread_safeness()
  开启了内存分配管理的线程安全变量,当内存分配时,
  redis会统计一个总内存分配量,这是一个共享资源,
  所以需要原子性操作,在redis的内存分配代码里,
  当需要原子操作时,就需要打开线程安全变量。
  */
  zmalloc_enable_thread_safeness();
  /*
  zmalloc_set_oom_handler()
  是一个内存分配错误处理,
  当无法得到需要的内存量时,
  会调用redisOutOfMemoryHandler函数。
  */
  zmalloc_set_oom_handler(redisOutOfMemoryHandler);
  srand(time(NULL)^getpid());
  gettimeofday(&tv,NULL);
  dictSetHashFunctionSeed(tv.tv_sec^tv.tv_usec^getpid());
  server.sentinel_mode = checkForSentinelMode(argc,argv);
  initServerConfig();
  ..........
}
zmalloc_set_oom_handler注册回调函数
redisOutOfMemoryHandler主要是个log日志打印,即在内存分配失败的时候触发回调函数,打印log。
void *zmalloc(size_t size) {
  void *ptr = malloc(size+PREFIX_SIZE);
 
  if (!ptr) zmalloc_oom_handler(size);
#ifdef HAVE_MALLOC_SIZE
  update_zmalloc_stat_alloc(zmalloc_size(ptr));
  return ptr;
#else
  *((size_t*)ptr) = size;
  update_zmalloc_stat_alloc(size+PREFIX_SIZE);
  return (char*)ptr+PREFIX_SIZE;
#endif
}
在分配内存失败的时候,触发回调函数
三、redis的log日志由于redis是单线程的  所以在redis.c里面的log没有做成多线程
这样的log,在单线程下 速度很快,因为无锁。但是在多线程下是不安全
简化了下  redis的log  单是大抵就是这样
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
 
/* Log levels */
#define REDIS_DEBUG 0
#define REDIS_VERBOSE 1
#define REDIS_NOTICE 2
#define REDIS_WARNING 3
 
 
#define REDIS_MAX_LOGMSG_LEN  1024 /* 默认信息长度 */
 
void redisLogRaw(int level, const char *msg);
void redisLog(int level, const char *fmt, ...);
 
/*
verbosity表示开启log的级别
需要写log的时候,log级别小于等于verbosity写log
否则不会写log
 
*/
struct redisServer {
  int verbosity;          /* 日志级别*/
  char *logfile;          /* Path of log file */
};
 
struct redisServer server; /* server global state */
 
 
void redisLog(int level, const char *fmt, ...) 
{
  //如果level级别大于verbosity则不打印
  if (level> server.verbosity) 
  {
    return;
  }
  va_list ap;
  char msg[REDIS_MAX_LOGMSG_LEN];
 
 
  va_start(ap, fmt);
  vsnprintf(msg, sizeof(msg), fmt, ap);
  va_end(ap);
 
  redisLogRaw(level,msg);
}
 
void redisLogRaw(int level, const char *msg) 
{
#if 1
   
  FILE *fp;
  char buf[64];
//  int rawmode = (level & REDIS_LOG_RAW);
  //int log_to_stdout = server.logfile[0] == '\0';
 
  //level &= 0xff; /* clear flags */
  //if (level < server.verbosity) return;
  if(server.logfile != NULL)
  {
    fp=fopen(server.logfile,"a");
  }
  else
  {
    fp=stdout;
  }
 
   
  int off;
//  struct timeval tv;
 
  //gettimeofday(&tv,NULL);
  //off = strftime(buf,sizeof(buf),"%d %b %H:%M:%S.",localtime(&tv.tv_sec));
  //snprintf(buf+off,sizeof(buf)-off,"%03d",(int)tv.tv_usec/1000);
  //fprintf(fp,"[%d] %s %c %s\n",(int)getpid(),buf,c[level],msg);
  fprintf(fp," %s\n",msg);
  fflush(fp);
 
  if(server.logfile != NULL)
  {
    fclose(fp);
  }
   
 
#endif
}
int main(void)
{
  server.verbosity=2;
  server.logfile=NULL;
  redisLog(1,"11111111\n");
  redisLog(2,"22222\n");
  redisLog(3,"333\n");
  getchar();
   
  return 0;
}
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
 
/* Log levels */
#define REDIS_DEBUG 0
#define REDIS_VERBOSE 1
#define REDIS_NOTICE 2
#define REDIS_WARNING 3
 
 
#define REDIS_MAX_LOGMSG_LEN    1024 /* 默认信息长度 */
 
void redisLogRaw(int level, const char *msg);
void redisLog(int level, const char *fmt, ...);
 
/*
verbosity表示开启log的级别
需要写log的时候,log级别小于等于verbosity写log
否则不会写log
 
*/
struct redisServer {
    int verbosity;                  /* 日志级别*/
    char *logfile;                  /* Path of log file */
};
 
struct redisServer server; /* server global state */
 
 
void redisLog(int level, const char *fmt, ...)
{
    //如果level级别大于verbosity则不打印
    if (level> server.verbosity)
    {
        return;
    }
    va_list ap;
    char msg[REDIS_MAX_LOGMSG_LEN];
 
 
    va_start(ap, fmt);
    vsnprintf(msg, sizeof(msg), fmt, ap);
    va_end(ap);
 
    redisLogRaw(level,msg);
}
 
void redisLogRaw(int level, const char *msg)
{
#if 1
    
    FILE *fp;
    char buf[64];
//  int rawmode = (level & REDIS_LOG_RAW);
    //int log_to_stdout = server.logfile[0] == '\0';
 
    //level &= 0xff; /* clear flags */
    //if (level < server.verbosity) return;
    if(server.logfile != NULL)
    {
        fp=fopen(server.logfile,"a");
    }
    else
    {
        fp=stdout;
    }
 
    
    int off;
//  struct timeval tv;
 
    //gettimeofday(&tv,NULL);
    //off = strftime(buf,sizeof(buf),"%d %b %H:%M:%S.",localtime(&tv.tv_sec));
    //snprintf(buf+off,sizeof(buf)-off,"%03d",(int)tv.tv_usec/1000);
    //fprintf(fp,"[%d] %s %c %s\n",(int)getpid(),buf,c[level],msg);
    fprintf(fp," %s\n",msg);
    fflush(fp);
 
    if(server.logfile != NULL)
    {
        fclose(fp);
    }
    
 
#endif
}
int main(void)
{
    server.verbosity=2;
    server.logfile=NULL;
    redisLog(1,"11111111\n");
    redisLog(2,"22222\n");
    redisLog(3,"333\n");
    getchar();
    
    return 0;
}


关注下面的标签,发现更多相似文章