MariaDB社区

 找回密码
 注册

QQ登录

只需一步,快速开始

搜索
查看: 3173|回复: 0
打印 上一主题 下一主题

求mysql 关于blob的接口

[复制链接]
跳转到指定楼层
1#
发表于 2008-8-15 17:33:02 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
BLOB是一个二进制大对象,可以容纳可变数量的数据。有4种BLOB类型:TINYBLOB、BLOB、MEDIUMBLOB和LONGBLOB。它们只是可容纳值的最大长度不同。
   mysql有些存储引擎是支持blob类型的,有些不支持。例如,MYISAM和falcon都是支持blob的。
   我现在的问题是,mysql是如何调用函数来处理blob这种特殊类型的。例如,在一个含有blob字段的表里面执行update,insert,delete这些操作的时候,函数是如何调用的。
   由于MYISAM是大多数表的默认存储引擎,所以我就研究了一下MYISAM。
我看了十多天MYISAM的源代码,始终没有搞明白里面的函数调用关系。因为结构太复杂了!在mysql-6.0.3/storage/myisam/目录下有个Mi_update.c文件,代码如下:
/* Update an old row in a MyISAM table */

#include "fulltext.h"
#include "rt_index.h"

int mi_update(register MI_INFO *info, const uchar *oldrec, uchar *newrec)
{//MI_INFO在myisamdef.h定义
   int flag,key_changed,save_errno;
   reg3 my_off_t pos;
   uint i;
   uchar old_key[MI_MAX_KEY_BUFF],*new_key;//#define MI_MAX_KEY_BUFF  (MI_MAX_KEY_LENGTH+MI_MAX_KEY_SEG*6+8+8)
   bool auto_key_changed=0;
   ulonglong changed;
   MYISAM_SHARE *share=info->s;
   ha_checksum old_checksum;
   DBUG_ENTER("mi_update");
   LINT_INIT(new_key);
   LINT_INIT(changed);
   LINT_INIT(old_checksum);

   DBUG_EXECUTE_IF("myisam_pretend_crashed_table_on_usage",
                   mi_print_error(info->s, HA_ERR_CRASHED);
                   DBUG_RETURN(my_errno= HA_ERR_CRASHED););
   if (!(info->update & HA_STATE_AKTIV))
   {
     DBUG_RETURN(my_errno=HA_ERR_KEY_NOT_FOUND);
   }
   if (share->options & HA_OPTION_READ_ONLY_DATA)
   {
     DBUG_RETURN(my_errno=EACCES);
   }
   if (info->state->key_file_length >= share->base.margin_key_file_length)
   {
     DBUG_RETURN(my_errno=HA_ERR_INDEX_FILE_FULL);
   }
   pos=info->lastpos;
   if (_mi_readinfo(info,F_WRLCK,1))
     DBUG_RETURN(my_errno);

   if (share->calc_checksum)
     old_checksum=info->checksum=(*share->calc_checksum)(info,oldrec);
   if ((*share->compare_record)(info,oldrec))
   {
     save_errno=my_errno;
     goto err_end;   /* Record has changed */
   }


   /* Calculate and check all unique constraints */
   key_changed=0;
   for (i=0 ; i < share->state.header.uniques ; i++)
   {
     MI_UNIQUEDEF *def=share->uniqueinfo+i;
     if (mi_unique_comp(def, newrec, oldrec,1) &&
  mi_check_unique(info, def, newrec, mi_unique_hash(def, newrec),
    info->lastpos))
     {
       save_errno=my_errno;
       goto err_end;
     }
   }
   if (_mi_mark_file_changed(info))
   {
     save_errno=my_errno;
     goto err_end;
   }

   /* Check which keys changed from the original row */

   new_key=info->lastkey2;
   changed=0;
   for (i=0 ; i < share->base.keys ; i++)
   {
     if (mi_is_key_active(share->state.key_map, i))
     {
       if (share->keyinfo.flag & HA_FULLTEXT )//128
       {
  if (_mi_ft_cmp(info,i,oldrec, newrec))//ft_update.c
  {
    if ((int) i == info->lastinx)
    {
    /*
      We are changeing the index we are reading on.  Mark that
      the index data has changed and we need to do a full search
      when doing read-next
    */
      key_changed|=HA_STATE_WRITTEN;//4//4
    }
    changed|=((ulonglong) 1 << i);
    if (_mi_ft_update(info,i, old_key,oldrec,newrec,pos))
      goto err;
  }
       }
       else
       {
  uint new_length=_mi_make_key(info,i,new_key,newrec,pos);//Mi_key.c
  uint old_length=_mi_make_key(info,i,old_key,oldrec,pos);

         /* The above changed info->lastkey2. Inform mi_rnext_same(). */
         info->update&= ~HA_STATE_RNEXT_SAME; //4096

  if (new_length != old_length ||
      memcmp((uchar*) old_key,(uchar*) new_key,new_length))
  {
    if ((int) i == info->lastinx)
      key_changed|=HA_STATE_WRITTEN; /* Mark that keyfile changed */
    changed|=((ulonglong) 1 << i);
    share->keyinfo.version++;
    if (share->keyinfo.ck_delete(info,i,old_key,old_length)) goto err;
    if (share->keyinfo.ck_insert(info,i,new_key,new_length)) goto err;
    if (share->base.auto_key == i+1)
      auto_key_changed=1;
  }
       }
     }
   }
   /*
     If we are running with external locking, we must update the index file
     that something has changed.
   */
   if (changed || !my_disable_locking)
     key_changed|= HA_STATE_CHANGED;// 1 database has changed

   if (share->calc_checksum)
   {
     info->checksum=(*share->calc_checksum)(info,newrec);
     /* Store new checksum in index file header */
     key_changed|= HA_STATE_CHANGED;
   }
   {
     /*
       Don't update index file if data file is not extended and no status
       information changed
     */
     MI_STATUS_INFO state; // st_mi_status_info Myisamdef.h
     ha_rows org_split;
     my_off_t org_delete_link;

     memcpy((char*) &state, (char*) info->state, sizeof(state));
     org_split=      share->state.split;
     org_delete_link= share->state.dellink;
     if ((*share->update_record)(info,pos,newrec))
       goto err;
     if (!key_changed &&
  (memcmp((char*) &state, (char*) info->state, sizeof(state)) ||
   org_split != share->state.split ||
   org_delete_link != share->state.dellink))
       key_changed|= HA_STATE_CHANGED;  /* Must update index file */
   }
   if (auto_key_changed)
     set_if_bigger(info->s->state.auto_increment,
                   retrieve_auto_increment(info, newrec));
   if (share->calc_checksum)
     info->state->checksum+=(info->checksum - old_checksum);

   info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV |
    key_changed);
   myisam_log_record(MI_LOG_UPDATE,info,newrec,info->lastpos,0);
   /*
     Every myisam function that updates myisam table must end with
     call to _mi_writeinfo(). If operation (second param of
     _mi_writeinfo()) is not 0 it sets share->changed to 1, that is
     flags that data has changed. If operation is 0, this function
     equals to no-op in this case.

     mi_update() must always pass !0 value as operation, since even if
     there is no index change there could be data change.
   */
   VOID(_mi_writeinfo(info, WRITEINFO_UPDATE_KEYFILE));
   allow_break();    /* Allow SIGHUP & SIGINT */
   if (info->invalidator != 0)
   {
     DBUG_PRINT("info", ("invalidator... '%s' (update)", info->filename));
     (*info->invalidator)(info->filename);
     info->invalidator=0;
   }
   DBUG_RETURN(0);

err:
   DBUG_PRINT("error",("key: %d  errno: %d",i,my_errno));
   save_errno=my_errno;
   if (changed)
     key_changed|= HA_STATE_CHANGED;
   if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_OUT_OF_MEM ||
       my_errno == HA_ERR_RECORD_FILE_FULL)
   {
     info->errkey= (int) i;
     flag=0;
     do
     {
       if (((ulonglong) 1 << i) & changed)
       {
  if (share->keyinfo.flag & HA_FULLTEXT)
  {
    if ((flag++ && _mi_ft_del(info,i, new_key,newrec,pos)) ||
        _mi_ft_add(info,i, old_key,oldrec,pos))
      break;
  }
  else
  {
    uint new_length=_mi_make_key(info,i,new_key,newrec,pos);
    uint old_length= _mi_make_key(info,i,old_key,oldrec,pos);
    if ((flag++ && _mi_ck_delete(info,i,new_key,new_length)) ||
        _mi_ck_write(info,i,old_key,old_length))
      break;
  }
       }
     } while (i-- != 0);
   }
   else
   {
     mi_print_error(info->s, HA_ERR_CRASHED);
     mi_mark_crashed(info);
   }
   info->update= (HA_STATE_CHANGED | HA_STATE_AKTIV | HA_STATE_ROW_CHANGED |
    key_changed);

  err_end:
   myisam_log_record(MI_LOG_UPDATE,info,newrec,info->lastpos,my_errno);
   VOID(_mi_writeinfo(info,WRITEINFO_UPDATE_KEYFILE));
   allow_break();    /* Allow SIGHUP & SIGINT */
   if (save_errno == HA_ERR_KEY_NOT_FOUND)
   {
     mi_print_error(info->s, HA_ERR_CRASHED);
     save_errno=HA_ERR_CRASHED;
   }
   DBUG_RETURN(my_errno=save_errno);
} /* mi_update */


看代码没有blob这个单词,而是通过调用了一些其他函数,例如:_mi_ft_update:

/* update a document entry */
int _mi_ft_update(MI_INFO *info, uint keynr, uchar *keybuf,
                  const uchar *oldrec, const uchar *newrec, my_off_t pos)
{
  int error= -1;
  FT_WORD *oldlist,*newlist, *old_word, *new_word;
  CHARSET_INFO *cs=info->s->keyinfo[keynr].seg->charset;
  uint key_length;
  int cmp, cmp2;
  DBUG_ENTER("_mi_ft_update");
  if (!(old_word=oldlist=_mi_ft_parserecord(info, keynr, oldrec,
                                            &info->ft_memroot)) ||
      !(new_word=newlist=_mi_ft_parserecord(info, keynr, newrec,
                                            &info->ft_memroot)))
    goto err;
  error=0;
  while(old_word->pos && new_word->pos)
  {
    cmp= mi_compare_text(cs, (uchar*) old_word->pos,old_word->len,
                             (uchar*) new_word->pos,new_word->len,0,0);
    cmp2= cmp ? 0 : (fabs(old_word->weight - new_word->weight) > 1.e-5);
    if (cmp < 0 || cmp2)
    {
      key_length=_ft_make_key(info,keynr,keybuf,old_word,pos);
      if ((error=_mi_ck_delete(info,keynr,(uchar*) keybuf,key_length)))
        goto err;
    }
    if (cmp > 0 || cmp2)
    {
      key_length=_ft_make_key(info,keynr,keybuf,new_word,pos);
      if ((error=_mi_ck_write(info,keynr,(uchar*) keybuf,key_length)))
        goto err;
    }
    if (cmp<=0) old_word++;
    if (cmp>=0) new_word++;
}
if (old_word->pos)
   error=_mi_ft_erase(info,keynr,keybuf,old_word,pos);
else if (new_word->pos)
   error=_mi_ft_store(info,keynr,keybuf,new_word,pos);
err:
  free_root(&info->ft_memroot, MYF(MY_MARK_BLOCKS_FREE));
  DBUG_RETURN(error);
}
还有很多其他代码!
我们项目小组想写一个支持blob操作的存储引擎,不过我是个菜鸟。希望各位大侠出手相助!
这些函数究竟是怎样调用的呢?可以说一下某种引擎(如myisam)的某个操作(如update,write_row,delete)。当然讲得越多越详细越好!
  谢谢!


[ 本帖最后由 oken 于 2008-8-15 18:38 编辑 ]
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 转播转播 分享分享 分享淘帖 顶 踩
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|Archiver|小黑屋|手机版|MariaDB社区 ( 京ICP备07012489号    |
业务联系: QQ:48474881; 邮箱: 48474881@qq.com; 电话:13911732319
声明:本站部分文章是网友转载,若未经作者同意或署名有误,请联系网站管理员。

GMT+8, 2024-9-20 00:28 , Processed in 0.060255 second(s), 19 queries .

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表