`

HashTable 构建与思考

 
阅读更多

1)

typedef struct hashnode_struct{

struct hashnode_struct *next;

const char *key;

void *val;

}* hashnode, _hashnode;

 

_hashnode obj;

hashnode p = &obj;  p是一个指针

 

HashTable.c:201: warning: control reaches end of non-void function

它的意思是:控制到达非void函数的结尾。就是说你的一些本应带有返回值的函数到达结尾后可能并没有返回任何值。这时候,最好检查一下是否每个控制流都会有返回值。

assignment makes pointer from integer without a cast

 

 

 

犯了一个愚蠢的错误:

   int i=1; i<<2;  得出的值为4, 但是i本身的值还是1,4是表达式i<<2的值 我一直以为i被修改了

 

 

 

#include<stdio.h>
#include<stdlib.h>
#include<string.h> 

#ifndef __GHASH_H_
#define __GHASH_H_

#define D_HASHSIZE 512
typedef struct ST_Node
{
   char * key;
   char * value;
   struct ST_Node * next;
} Node;



void HashInit();
Node *  HashInsert(char * key, char * value);
int  HashRemove(char * key);
Node *  HashSearch(char * key);
void FreeGHash();
void HashTraversal();

#endif
 


static Node* hashtab[D_HASHSIZE];
 

void HashInit()
{
 	 int i=0;
 	 for(i=0; i<D_HASHSIZE; i++)
 	 {
	  	hashtab[i]=NULL;
     }
}

Node *  HashInsert(char * key,char * value)
{
 	 Node * np;
 	 unsigned int hashval;
 	 if((np=HashSearch(key))==NULL) /*如果没有相同的key则直接插入*/
 	 {
         np=(Node *)malloc(sizeof(Node));
         if(NULL==np || NULL ==(np->key = strdup(key))  	 /*strdup:复制一个字符串副本*/
					 || NULL ==(np->value = strdup(value)) 
		   )
         {
             return NULL;
         }
         hashval = _Hash(key);
         np->next = (Node *)hashtab[hashval]; /*解决不同的key算出相同hash值的collision*/
         hashtab[hashval] = np;
     }
     else	/*如果有则覆盖*/
     {
	  	 free(np->value); 
	  	 if((np->value=strdup(value))== NULL)
	  	 {
		    return NULL;
         }
 	 }
 	 return np;
}


int  HashRemove(char * key)
{
	Node * np = NULL;
	np = HashSearch(key);
    if(np == (Node *)hashtab[_Hash(key)]) /*如果是数组所指向的链表的头指针则让数组指向next*/
    {
        freeNode(np);
        hashtab[_Hash(key)] = np->next;
    }

	  
	return 0; /*正常返回0*/
}

/*
	确保_Hash(key) 值相同
	&& key字符串也相同
	return: 所定位的指针
*/
Node *  HashSearch(char * key)
{
    Node  *np;

    for(np = (Node *)hashtab[_Hash(key)]; /*定位HashTable所指向的链表 进行循环比较*/
        np != NULL;
        np = np->next)
      if(strcmp(key, np->key) == 0) 
           return np;
    return NULL;
}

void FreeGHash()
{ 
 	 for(int i=0; i<D_HASHSIZE; i++)
 	 {
	     if(hashtab[i]!=NULL)
	     {
		    Node * tmp;
		    Node * deln;
		    for(tmp=hashtab[i]; tmp!=NULL; tmp=hashtab[i])
		    {
			     hashtab[i]=tmp->next;
			     freeNode(tmp);
			}
		 }
     }
}

void HashTraversal()
{
     printf("Print Hash:\n");
     int i=0;
     for(i=0; i<D_HASHSIZE; i++)
     {
         if(hashtab[i] !=NULL )
         {
            printf("%d---",i);
            Node * tmp;
            for(tmp=hashtab[i]; tmp!=NULL; tmp=tmp->next)
            {
                 printf("%s ==> %s;",tmp->key,tmp->value);
            }
            printf("\n");
         }
     }
}   

static unsigned int _Hash(char *key)
{
    return _ELFHash(key)%D_HASHSIZE;
}

// ELF Hash Function
static unsigned int _ELFHash(char *str)
{
       unsigned int hash = 0;
       unsigned int x = 0;

       while (*str)
      {
           hash = (hash << 4) + (*str++);//hash左移4位,当前字符ASCII存入hash
           if ((x = hash & 0xF0000000L) != 0)
           {//如果最高的四位不为0,则说明字符多余7个,如果不处理,再加第九个字符时,第一个字符会被移出,因此要有如下处理。

           //该处理,如果对于字符串(a-z 或者A-Z)就会仅仅影响5-8位,否则会影响5-31位,因为C语言使用的算数移位
           hash ^= (o >> 24);
           //清空28-31位。
           hash &= ~x;	/*将hash的值5-31 原封保留*/
           }
       }

      //返回一个符号位为0的数,即丢弃最高位,以免函数外产生影响。(我们可以考虑,如果只有字符,符号位不可能为负)
     return (hash & 0x7FFFFFFF);

}

static void freeNode(Node * item)
{
   free(item->key);
   free(item->value);
   free(item);
}
 
 
int test2(void)
{
//  Node *nodePtr;

    HashInit();

    HashInsert("tom", "18");
    HashInsert("jerry", "20");
    HashInsert("spike", "22");

    HashTraversal();

    printf("\nupdate spike age:\n");
    HashInsert("spike", "0");
    HashTraversal();


    printf("\nremove tom:\n");
    HashRemove("tom");
    HashTraversal();

    return 0;


}

 
int main(void)
{
	 test2(void);
	 return 0; 
}

 

  

       while (*str)
      {
           hash = (hash << 4) + (*str++);//hash左移4位,当前字符ASCII存入hash
           if ((x = hash & 0xF0000000L) != 0)
           {//如果最高的四位不为0,则说明字符多余7个,如果不处理,再加第九个字符时,第一个字符会被移出,因此要有如下处理。

			   //该处理,如果对于字符串(a-z 或者A-Z)就会仅仅影响5-8位,否则会影响5-31位,因为C语言使用的算数移位
			   hash ^= (x >> 24);  
			   /**
					因为x除了5-8位为!0其余全部为0
					hash ^ 0X000000F0 ===>hash中除了5-8位其余位均无影响,
											相当于位与1操作(&0X111111) 效果是一样的
											其余的值原封保留
											
					
			   */
			   
			   //清空28-31位。
			   hash &= ~x;	
			   /*
					:(x虽然右移了24位但是x的值并未改变) 
				   x取之与hash的高4位,用之hash的高4位(x = hash & 0xF0000000L)
                   x取反后与hash进行&操作,则会把hash的高4位清0
			   */
           }
       }

 

 

一个数和0进行异或运算 == 这个数和1进行位运算

 

 

分享到:
评论

相关推荐

    用Hashtable构建Jtree,使各节点按输入顺序显示

    这是因为`Hashtable`在内部使用了哈希表(Hash Table)数据结构,它的插入顺序并不保证与元素添加的顺序一致。 为了解决这个问题,开发者创建了一个名为`MyHashtable`的新类,该类继承自`Hashtable`。`MyHashtable`...

    C# json 转hashtable

    与此相关的,`Hashtable`是.NET框架中的一个古老的集合类,用于存储键值对,它在早期的.NET应用中十分常见。然而,随着.NET Framework的发展,`Dictionary, TValue&gt;`逐渐取代了`Hashtable`,因为后者不支持泛型,且...

    hashmap与hashtable区别

    ### HashMap与Hashtable的区别 在Java编程语言中,`HashMap`和`Hashtable`是两种非常重要的数据结构,它们都用于存储键值对。然而,在实际应用过程中,这两种数据结构有着本质的不同,下面将详细介绍这些差异。 ##...

    WinFormHashTable最简单用法,.net hashtable ,hashtable ,hashtable用法

    在WinForm应用中,Hashtable常用于存储用户界面控件与数据之间的关联,或者在事件处理中暂存数据。例如,可以创建一个Hashtable来存储窗体控件与数据库字段的对应关系,方便数据绑定和操作。 5. **注意点** - **...

    HashMap与HashTable区别

    ### HashMap与HashTable的区别 在Java编程语言中,`HashMap`和`HashTable`是两种非常重要的数据结构,它们都实现了`Map`接口,并提供了键值对的存储方式。这两种数据结构虽然相似,但在实现细节和使用场景上存在...

    HashTable

    3. 插入与查找:插入操作首先计算键的哈希值,然后根据哈希值找到对应的位置进行插入。查找操作同样计算哈希值,找到对应位置后,如果是链地址法,则需要遍历链表查找匹配的键。 4. 删除:删除操作涉及到如何找到要...

    hashtable序列化与反序列化

    本文将详细探讨标题所提及的“hashtable序列化与反序列化”,并提供一个基本的示例。 首先,让我们理解什么是序列化。序列化是将对象的状态转换为可存储或可传输的形式的过程。在Java中,对象序列化允许我们将一个...

    HashMap与HashTable和HashSet的区别

    ### HashMap与HashTable和HashSet的区别 #### 一、概述 在Java集合框架中,`HashMap`, `HashTable` 和 `HashSet` 是三个重要的数据结构,它们分别实现了`Map`接口和`Set`接口,提供了不同的功能来满足不同的编程...

    HashMap和HashTable的区别和不同

    ### HashMap与HashTable的区别详解 #### 引言 在Java编程中,`HashMap`与`HashTable`作为两种常用的数据结构,经常被用来存储键值对数据。尽管它们在功能上相似,但在实现细节、性能表现以及使用场景方面存在显著...

    asp.net遍历hashtable

    在ASP.NET中,Hashtable是一种常用的数据结构,它是一个键值对集合,允许程序员存储和检索对象。本篇文章将深入探讨如何在ASP.NET中遍历Hashtable,以及相关的重要知识点。 首先,理解Hashtable的基本概念至关重要...

    hashtable存储数据.rar

    在Java编程语言中,`Hashtable`是一个非常基础且重要的数据结构,它属于集合框架的一部分,提供了键值对(key-value pairs)的存储功能。`Hashtable`类是线程安全的,意味着在多线程环境下,它能确保数据的一致性和...

    hashMap和hashTable的区别

    #### 一、简介与基本概念 `HashMap` 和 `HashTable` 都是 Java 集合框架中非常重要的数据结构,它们都实现了 `Map` 接口,用于存储键值对。尽管它们在功能上有很多相似之处,但在实现细节和性能特性上存在显著差异...

    C#-Hashtable应用

    除了基本操作,Hashtable还提供了其他有用的方法,如Clear()用于清除所有元素,CopyTo()用于将Hashtable复制到数组,以及GetHashCode()和Equals()方法,这些方法与对象的比较和身份验证有关。 在处理集合时,遍历是...

    HashMap和HashTable底层原理以及常见面试题

    HashMap和HashTable底层原理以及常见面试题 HashMap和HashTable是Java中两个常用的数据结构,都是基于哈希表实现的,但它们之间存在着一些关键的区别。本文将深入探讨HashMap和HashTable的底层原理,并总结常见的...

    经典讲解List和ArrayList和Vector和HashTable和HashMap区别

    `Vector`也是`List`接口的实现,与`ArrayList`类似,但它是线程安全的。这意味着在多线程环境下,多个线程可以同时操作`Vector`而不会产生数据不一致的问题。但是,由于其同步机制,性能通常低于`ArrayList`。 4. ...

    hashtable和hashmap的区别

    ### Hashtable和HashMap的区别 在Java编程语言中,`Hashtable`和`HashMap`是两种非常重要的数据结构,它们都实现了`Map`接口,用于存储键值对。尽管它们有着相似的功能,但在实现细节和应用场景上存在显著差异。接...

    java Hashtable的泛型化

    3. **自动装箱与拆箱**:对于基本类型对应的包装类,泛型`Hashtable`支持自动装箱(如`Integer`与`int`之间)和拆箱,简化了代码。 4. **可读性**:通过明确指定类型,代码的意图更加清晰,阅读和维护起来更容易。 ...

    hashtable和dictionary的探讨

    在编程领域,哈希表(Hashtable)和字典(Dictionary)是两种常用的数据结构,它们在存储和检索键值对时提供了高效的性能。本文将深入探讨这两种数据结构的原理、性能差异以及实际应用中的考虑因素。 哈希表,通常...

    Hashtable的使用

    - **不允许`null`键和`null`值**:与`HashMap`不同,`Hashtable`不允许插入`null`键或`null`值。如果你尝试这样做,会抛出`NullPointerException`。 - **非有序性**:`Hashtable`中的元素顺序并不是按照插入的顺序...

    HashMap与HashTable的区别(含源码分析)

    在Java编程语言中,`HashMap`和`HashTable`都是实现键值对存储的数据结构,但它们之间存在一些显著的区别,这些区别主要体现在线程安全性、性能、null值处理以及一些方法特性上。以下是对这两个类的详细分析: 1. ...

Global site tag (gtag.js) - Google Analytics