在日常开发中。数据集合是我们不可缺少的重要工具之中的一个。在C#中,.Net Framework也为我们提供了种类繁多,功能多样的数据集工具。在此,我基于List<T> 和 HashTable制作了一个功能强大的数据集,我将其命名为HashList。他集二者的优势于一身,既支持数组索引的操作,同一时候也支持键值对操作。
我更在此基础上封装了一些经常使用的工具函数,使其可以极大的方便我们日常开发中的数据管理。
HashList 核心功能例如以下:
1、类型安全的数据集合。省去了强制转换的繁冗操作
2、依据数组索引。哈希键值来加入,删除。读取数据
3、从序列的指定索引处截取指定长度
4、使用实现了ICompare接口的排序类进行排序
5、查找指定谓语条件的一组数据或单个数据
6、连接两个HashList数据集
不足之处:
1、原本打算使用双向链接表来替代List<T>,可是发现LinkedList无法实现数据集的过滤和排序,所以不得已又换回了了List<T>。假设哪位大神知道双向连接表的过滤和排序实现方法。还请指教。
2、因为C#是强类型语言。所以某些操作,特别是删除操作。对于HashList中的HashTable部分就显得比較困难,因为我无法去动态获取对象的key属性。(当然。反射可以做到。可是因为它的效率较低。所以我不打算使用)。眼下实现的方法就是在HashTable中遍历比較,然后再删除。假设哪位大神有更好的方法。请指教。
另外,本代码中对于对象的打印使用的是Jayrock。
下面源代码及Jayrock下载地址,请点这里。
废话少说,直接上源代码:
HashList.cs
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System;
using UnityEngine;
//************************************************************************************
//* 命名空间:xyxk;
//* 版本: V1.0.0.0;
//* 创建人: 枫叶天空 (孙伟航);
//* QQ : 1569423066;
//* 电子邮箱:SunixSky@gmail.com
//* 创建时间:2014-06-14;
//* 描写叙述 : 功能强大的自己定义集合,同一时候兼容list与hashtable的核心功能;
//*************************************************************************************
namespace xyxk
{
/// <summary>
/// 哈希列表;
/// </summary>
public class HashList<T> : IDisposable
{
//哈希表对象,用来进行键值数据处理;
public Hashtable hash = new Hashtable();
//泛型列表,用来进行有序数据处理;
public List<T> list = new List<T>();
//字符串格式化标志;
//true 打印list;
//false 打印hashTabel;
public bool typeList = true;
/////////////////////////////////////// public method ////////////////////////////////////////////////////////////
/// <summary>
/// 获取一个Array结构的数据集;
/// </summary>
/// <returns></returns>
public T[] array
{
get
{
T[] array = new T[count];
list.CopyTo(array);
return array;
}
}
/// <summary>
/// 从当前列表中指定的索引位置处截取num个对象。并返回;
/// </summary>
/// <param name="start">截取的起始索引</param>
/// <param name="num">截取数量</param>
/// <returns></returns>
public List<T> splice(int start, int num)
{
T[] array = new T[num];
list.CopyTo(start,array,0,num);
removeFromHashByArray(array);
return array.ToList();
}
/// <summary>
/// 连接两个hashList;
/// </summary>
/// <param name="value"></param>
public void concat(HashList<T> value)
{
list = list.Concat(value.list).ToList(); ;
addFromHash(value.hash);
}
/// <summary>
/// 通过键值加入对象;
/// </summary>
/// <param name="key"></param>
/// <param name="value"></param>
public void addElement(object key, T value)
{
list.Add(value);
hash.Add(key, value);
}
/// <summary>
/// 向指定的索引处加入对象;
/// </summary>
/// <param name="value"></param>
/// <param name="index"></param>
public void addElement(object key, T value, int index)
{
list.Insert(index, value);
hash.Add(key, value);
}
/// <summary>
/// 获取指定索引处的对象;
/// </summary>
/// <param name="index"></param>
/// <returns></returns>
public T getElement(int index)
{
return list.ElementAtOrDefault(index);
}
/// <summary>
/// 获取指定键值的对象;
/// </summary>
/// <param name="key"></param>
/// <returns></returns>
public T getElement(object key)
{
if (hash.ContainsKey(key))
return (T)hash[key];
return default(T);
}
/// <summary>
/// 删除指定的对象;
/// </summary>
/// <param name="value"></param>
public void removeElement(T value)
{
list.Remove(value);
removeFromHash(value);
}
/// <summary>
/// 删除指定键值的对象;
/// </summary>
/// <param name="key"></param>
public void removeElement(object key)
{
T value = (T)hash[key];
list.Remove(value);
hash.Remove(key);
}
/// <summary>
/// 删除指定索引处的对象;
/// </summary>
/// <param name="index"></param>
public void removeElement(int index)
{
T value = list.ElementAtOrDefault(index);
list.Remove(value);
removeFromHash(value);
}
/// <summary>
/// 获取list中的最后一个对象,并将其从list中删除;
/// </summary>
/// <returns></returns>
public T pop()
{
int index = list.Count - 1;
T value = list.ElementAt(index);
removeElement(index);
return value;
}
/// <summary>
/// 获取list中的第一个对象,并将其从list中删除;
/// </summary>
/// <returns></returns>
public T shift()
{
T value = list.ElementAt(0);
removeElement(0);
return value;
}
/// <summary>
/// 依据指定的接口排序类,对list进行排序,并将排序后的list返回;
/// </summary>
/// <param name="compare"></param>
/// <returns></returns>
public List<T> sort(IComparer<T> compare)
{
list.Sort(compare);
return list;
}
/// <summary>
/// 搜索与指定谓词所定义的条件相匹配的元素,并返回整个 List<T> 中的第一个匹配元素;
/// </summary>
/// <param name="match"></param>
/// <returns></returns>
public T find(Predicate<T> match)
{
return list.Find(match);
}
/// <summary>
/// 检索与指定谓词定义的条件匹配的全部元素;
/// </summary>
/// <param name="match"></param>
/// <returns></returns>
public List<T> findAll(Predicate<T> match)
{
return list.FindAll(match);
}
/// <summary>
/// 返回指定对象的索引;
/// </summary>
/// <param name="value"></param>
/// <returns></returns>
public int indexOf(T value)
{
return list.IndexOf(value);
}
/// <summary>
/// 清空列表中全部数据;
/// </summary>
public void clear()
{
list.Clear();
hash.Clear();
}
/// <summary>
/// 将列表转化成字符串;
/// </summary>
/// <returns></returns>
public override string ToString()
{
if (typeList)
return Jayrock.Json.Conversion.JsonConvert.ExportToString(list);
else
return Jayrock.Json.Conversion.JsonConvert.ExportToString(hash);
}
/// <summary>
/// 获取list中包括的元素数;
/// </summary>
public int count
{
get
{
return list.Count;
}
}
/// <summary>
/// 释放;
/// </summary>
public void Dispose()
{
Dispose(true);//释放全部的资源;
GC.SuppressFinalize(this);//不须要再调用本对象的Finalize方法
}
///////////////////////////////// protected method ///////////////////////////////////////////////////////
/// <summary>
/// 从hash表中删除指定对象;
/// </summary>
/// <param name="value"></param>
protected void removeFromHash(T value)
{
object key = null;
foreach (DictionaryEntry de in hash)
{
if (de.Value.Equals(value))
{
key = de.Key;
break;
}
}
if (key != null)
hash.Remove(key);
}
/// <summary>
/// 从指定的数组中删除对象;
/// </summary>
/// <param name="array"></param>
protected void removeFromHashByArray(T[] array)
{
int length = array.Length;
for (int i = 0; i < length; i++ )
removeElement(array[i]);
}
/// <summary>
/// 释放对象;
/// </summary>
/// <param name="disposing">是否清理托管资源</param>
protected virtual void Dispose(bool disposing)
{
clear();
//清理托管资源;
if (disposing)
{
list = null;
hash = null;
}
}
/// <summary>
/// 从指定的hash表中加入数据;
/// </summary>
/// <param name="value"></param>
protected void addFromHash(Hashtable value)
{
foreach (DictionaryEntry de in value)
{
hash.Add(de.Key, de.Value);
}
}
}
}
同一时候附上測试代码,直接绑定在U3D的一个GameObject上就可以。
HashListTest.cs
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Jayrock.Json;
namespace xyxk
{
public class HashListTest : MonoBehaviour
{
public HashList<CustomeData> hashList = new HashList<CustomeData>();
void Start()
{
testAddElement();
testConcat();
testSplice();
testValues();
testSort();
testIndexOf(hashList.getElement(3));
testFind();
testFindAll();
testShift();
testPop();
testGetElement();
testRemoveElement();
}
/// <summary>
/// 加入元素測试用例;
/// </summary>
public void testAddElement()
{
Debug.Log("====================testAddElement==========================");
for (int i = 0; i < 3; i++)
{
//直接加入对象引用;
CustomeData data = new CustomeData();
data.key = "s" + i;
data.name = "name" + i;
data.level = i;
data.Type = i % 2;
hashList.addElement(data.key,data);
}
print("直接加入对象引用;-------result:" + hashList.ToString());
//将对象引用加入到指定的索引处;
CustomeData data3 = new CustomeData();
data3.key = 3;
data3.name = "name" + 3;
data3.level = 3;
data3.Type = -1;
hashList.addElement(data3.key,data3, 0);
print("将对象引用加入到指定的索引处;-------result:" + hashList.ToString());
//键值对方式加入;
CustomeData data4 = new CustomeData();
data4.key = "4";
data4.name = "name" + 4;
data4.level = 4;
data4.Type = -1;
hashList.addElement(data4.key, data4);
print("键值对方式加入;-------result:" + hashList.ToString());
}
/// <summary>
/// 排序測试;
/// </summary>
public void testSort()
{
Debug.Log("====================testSort==========================");
hashList.sort(new SortLevel());
print("sort result:" + hashList.ToString());
}
/// <summary>
/// 获取元素測试用例;
/// </summary>
public void testGetElement()
{
Debug.Log("====================testGetElement==========================");
CustomeData data1 = hashList.getElement(0);
print("通过索引获取对象; data1:" + data1.ToString());
CustomeData data2 = hashList.getElement("state2");
print("通过键值获取对象; data2:" + data2.ToString());
}
/// <summary>
/// 删除元素測试用例;
/// </summary>
public void testRemoveElement()
{
Debug.Log("====================testRemoveElement==========================");
//通过对象引用删除对象;
CustomeData data = hashList.getElement(1);
hashList.removeElement(data);
print("通过对象引用删除对象;-------result:" + hashList.ToString());
//通过索引删除对象;
hashList.removeElement(0);
print("通过索引删除对象;-------result:" + hashList.ToString());
//通过键值删除对象;
hashList.removeElement("s1");
print("通过键值删除对象;-------result:" + hashList.ToString());
}
/// <summary>
/// 获取列表中最后一个数据;
/// </summary>
public void testPop()
{
Debug.Log("====================testPop==========================");
CustomeData data = hashList.pop();
print("移除列表中的最后一个对象; data:" + data.ToString());
print("终于结果;-------result:" + hashList.ToString());
}
/// <summary>
/// 获取列表中第一个数据;
/// </summary>
public void testShift()
{
Debug.Log("====================testShift==========================");
CustomeData data = hashList.shift();
print("移除列表中的第一个对象; data:" + data.ToString());
print("终于结果;-------result:" + hashList.ToString());
}
/// <summary>
/// 查找指定谓语条件的单条数据;
/// </summary>
public void testFind()
{
Debug.Log("====================testFind==========================");
CustomeData data = hashList.find(CustomeData.findOne);
print("查找结果;-------result:" + data.ToString());
}
/// <summary>
/// 查找指定谓语条件的数据集合;
/// </summary>
public void testFindAll()
{
Debug.Log("====================testFindAll==========================");
List<CustomeData> list = hashList.findAll(CustomeData.findByType);
print("查找结果;-------result:" + Jayrock.Json.Conversion.JsonConvert.ExportToString(list));
}
/// <summary>
/// 查找给定元素的索引;
/// </summary>
/// <param name="data"></param>
public void testIndexOf(CustomeData data)
{
Debug.Log("====================testIndexOf==========================");
int index = hashList.indexOf(data);
print("索引值;------:" + index);
}
/// <summary>
/// 数据的多种获取方式;
/// </summary>
public void testValues()
{
Debug.Log("====================testValues==========================");
List<CustomeData> list = hashList.list;
Debug.Log("List 结构数据;-------:" + Jayrock.Json.Conversion.JsonConvert.ExportToString(list));
Hashtable hashTable = hashList.hash;
Debug.Log("HashTable 结构数据;-------:" + Jayrock.Json.Conversion.JsonConvert.ExportToString(hashTable));
CustomeData[] array = hashList.array;
Debug.Log("Array 结构数据;-------:" + Jayrock.Json.Conversion.JsonConvert.ExportToString(array));
}
/// <summary>
/// 连接两个hashList;
/// </summary>
public void testConcat()
{
Debug.Log("====================testConcat==========================");
HashList<CustomeData> list = new HashList<CustomeData>();
for (int i = 0; i < 3; i++)
{
//直接加入对象引用;
CustomeData data = new CustomeData();
data.key = "state" + i;
data.name = "nameK" + i;
data.level = i;
data.Type = i % 2;
list.addElement(data.key, data);
}
Debug.Log("新表; ----" + list.ToString());
hashList.concat(list);
Debug.Log("连接新表; ----" + hashList.ToString());
}
/// <summary>
/// 截取hashList;
/// </summary>
public void testSplice()
{
Debug.Log("====================testSplice==========================");
hashList.typeList = false;
List<CustomeData> list = hashList.splice(3, 3);
Debug.Log("截取数据; ----" + Jayrock.Json.Conversion.JsonConvert.ExportToString(list));
Debug.Log("剩余数据; ----" + hashList.ToString());
}
}
public class CustomeData
{
public string name;
public int level;
private int type = 0;
public int Type
{
get { return type; }
set { type = value; }
}
private object _key;
public object key
{
get
{
return _key;
}
set
{
_key = value;
}
}
public override string ToString()
{
return Jayrock.Json.Conversion.JsonConvert.ExportToString(this);
}
/// <summary>
/// 查找等级为3,且type为-1的数据;
/// </summary>
/// <param name="data"></param>
/// <returns></returns>
public static bool findOne(CustomeData data)
{
return (data.level) == 3 && (data.type == -1);
}
/// <summary>
/// 查找等级为3,且type为-1的数据;
/// </summary>
/// <param name="data"></param>
/// <returns></returns>
public static bool findByType(CustomeData data)
{
return data.type == -1;
}
}
/// <summary>
/// 依据数据的level字段进行倒序排列;
/// </summary>
public class SortLevel : IComparer<CustomeData>
{
public int Compare(CustomeData a, CustomeData b)
{
if (a.level <= b.level)
return 1;
else
return -1;
}
}
}
U3D开发交流QQ群 345305437 欢迎您的加入