[关闭]
@Dubyoo 2014-06-13T19:27:19.000000Z 字数 3793 阅读 3078

2014-05-04 项目一:SpellCorrect项目文档

项目文档


说明

  1. 开发环境:Linux、 c++、 UDP、 多线程
  2. 程序框架:客户端通过UDP发送关键词到服务器,服务器接收后放入线程池进行处理,与文本库中单词进行编辑距离计算比较,向客户端返回编辑距离最小的匹配单词。

第一阶段:搭建框架 0504

1.目录结构

  1. bin //存放可执行文件和执行脚本
  2. conf //配置文件,包括词库文件路径、磁盘 cache 路径、服务器地址等
  3. data //存放词库文件和 cache 磁盘文件
  4. include //存放头文件
  5. log //存放系统日志文件
  6. src //存放 cpp 代码文件
  7. Makefile //Makefile

2.搭建UDP服务器

  负责接收客户端查询词,处理后发回客户端

3.加入ThreadPool

  线程池运行后负责处理查询任务,并向客户端发回处理后的单词


第二阶段:编辑距离核心算法 0506

  由最长公共子序列开始,引入编辑距离算法:

  1. //递归算法计算编辑距离(效率极慢)
  2. int EditDistance(const string &a, const string &b, int i, int j)
  3. //比较两个字符串的编辑距离
  4. {
  5. if(i == 0)
  6. {
  7. return j;
  8. }
  9. if(j == 0)
  10. {
  11. return i;
  12. }
  13. if(a[i - 1] == b[j - 1]) //当前字符相等,编辑距离不变
  14. {
  15. return EditDistance(a, b, i - 1, j - 1) ;
  16. }
  17. else
  18. {
  19. return get_min( EditDistance(a, b, i - 1, j - 1) + 1, //改
  20. EditDistance(a, b, i, j - 1) + 1, //增
  21. EditDistance(a, b, i - 1, j) + 1 ); //删
  22. }
  23. }
  1. //非递归算法计算编辑距离
  2. int ED(const string &a, const string &b)
  3. {
  4. //对待比较的两个单词构造二阶矩阵
  5. int **memo = new int * [a.size() + 1];
  6. for(string::size_type ix = 0; ix != a.size() + 1; ++ix)
  7. {
  8. memo[ix] = new int [b.size() +1];
  9. }
  10. //矩阵初始化第一行和第一列
  11. for(string::size_type ix = 0; ix != a.size() + 1; ++ix)
  12. {
  13. memo[ix][0] = ix;
  14. }
  15. for(string::size_type jx = 0; jx != b.size() +1; ++jx)
  16. {
  17. memo[0][jx] = jx;
  18. }
  19. //行优先遍历整个矩阵,按 增删改 权值+1 计算编辑距离
  20. for(string::size_type ix = 1; ix != a.size() +1; ++ix)
  21. {
  22. for(string::size_type jx = 1; jx != b.size() +1; ++jx)
  23. {
  24. if(a[ix - 1] == b[jx - 1])
  25. {
  26. memo[ix][jx] = memo[ix -1][jx -1];
  27. }
  28. else
  29. {
  30. memo[ix][jx] = get_min(memo[ix - 1][jx] + 1,
  31. memo[ix][jx - 1] + 1,
  32. memo[ix - 1][jx - 1] +1);
  33. }
  34. }
  35. }
  36. int distance = memo[a.size()][b.size()]; //计算得到的编辑距离
  37. for(string::size_type ix = 0; ix != a.size(); ++ix)
  38. {
  39. delete [] memo[ix];
  40. }
  41. delete [] memo;
  42. return distance;
  43. }

第三阶段:切词统计 0507

1.英文

  对英文语料库,去符号,大写转小写,然后读入内存统计词频,最后写入文件,格式为

  1. word freq
  2. word freq
  3. ...

2.中文

  对于中文语料库,由于语料库原编码是gbk格式,用iconv转换函数将语料库转换为utf-8格式,然后读入文章使用jieba切词工具进行切词(也可切词后转换编码)并统计词频。格式同上。


第四阶段:扩展中文 utf-8 编码 0508

  对中文的扩展首先需要搞清楚utf-8编码的格式

  1. /* 英文字符 */
  2. //只含1个字节 格式为:
  3. 0xxxxxxx
  4. /* gbk编码 */
  5. //包含2个字节 格式为:
  6. 1xxxxxxx 1xxxxxxx
  7. /* utf-8编码 */
  8. //包含字节数不定 格式:
  9. 110xxxxx 10xxxxxx
  10. 1110xxxx 10xxxxxx 10xxxxxx //中文字符一般是这种格式(3字节)
  11. 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx
  12. ... //2-6字节不等
  1. //判断当前字节是否英文字符
  2. bool is_en(unsigned char c)
  3. {
  4. return ((c & 0x80) == 0);
  5. }
  6. //判断当前字节是否utf-8中文的第1个字节
  7. bool is_utf8_ch_first(unsigned char c)
  8. {
  9. return ((c & 0xE0) == 0xE0);
  10. }
  11. //判断当前字节是否utf-8中文的第2或第3个字节
  12. bool is_utf8_ch_second_third(unsigned char c)
  13. {
  14. return ((c & 0x80) == 0x80);
  15. }
  16. /*
  17. * 将每个字(英文字符或中文汉字)转换成一个32位整型数
  18. * 这样将一个单词转换为一个32位整型数数组
  19. * 用数组代替单词来进行编辑距离的计算
  20. */
  21. void parse_utf8_string(const string &s, vector<uint32_t> &vec)
  22. {
  23. for(string::size_type ix = 0; ix != s.size(); ++ix)
  24. {
  25. if(is_en(s[ix]))
  26. {
  27. vec.push_back((uint32_t)(s[ix]));
  28. //一个英文字符转换为32位整型数
  29. }
  30. else if(is_utf8_ch_first(s[ix]))
  31. {
  32. if(is_utf8_ch_second_third(s[ix + 1]) && is_utf8_ch_second_third(s[ix + 2]))
  33. {
  34. vec.push_back((uint32_t)(s[ix] << 16) + (uint32_t)(s[ix + 1] << 8) +(uint32_t)(s[ix + 2]));
  35. //一个中文字转换为32位整型数
  36. ix++;
  37. ix++;
  38. }
  39. else
  40. throw runtime_error("utf-8");
  41. }
  42. }
  43. }
  44. int ED_utf8(const string &a, const string &b)
  45. {
  46. vector<uint32_t> vec_a;
  47. vector<uint32_t> vec_b;
  48. parse_utf8_string(a, vec_a); //将字符串a转化为数组vec_a
  49. parse_utf8_string(b, vec_b); //将字符串b转化为数组vec_b
  50. int len_a = vec_a.size();
  51. int len_b = vec_b.size();
  52. int **memo = new int * [len_a + 1];
  53. for(int ix = 0; ix != len_a + 1; ++ix)
  54. {
  55. memo[ix] = new int [len_b + 1];
  56. }
  57. for(int ix = 0; ix != len_a + 1; ++ix)
  58. {
  59. memo[ix][0] = ix;
  60. }
  61. for(int jx = 0; jx != len_b +1; ++jx)
  62. {
  63. memo[0][jx] = jx;
  64. }
  65. for(int ix = 1; ix != len_a +1; ++ix)
  66. {
  67. for(int jx = 1; jx != len_b +1; ++jx)
  68. {
  69. if(vec_a[ix - 1] == vec_b[jx - 1])
  70. {
  71. memo[ix][jx] = memo[ix -1][jx -1];
  72. }
  73. else
  74. {
  75. memo[ix][jx] = get_min(memo[ix - 1][jx] + 1,
  76. memo[ix][jx - 1] + 1,
  77. memo[ix - 1][jx - 1] +1);
  78. }
  79. }
  80. }
  81. int distance = memo[len_a][len_b];
  82. for(int ix = 0; ix != len_a; ++ix)
  83. {
  84. delete [] memo[ix];
  85. }
  86. delete [] memo;
  87. return distance;
  88. }

第五阶段:优化扩展

1.cache缓存(难点在于线程间的cache同步策略)

  1. //存储cache采用的数据类型
  2. std::unordered_map<std::string, std::string> cache;
  3. //存储记录搜索词 search_word 和计算结果 result_word
  4. //如果再次有同样的搜索词search_word查询,直接由cache内的result_word返回结果,无需遍历单词库计算比较编辑距离

2.index索引

  1. 利用索引来缩小候选词范围,达到提高计算性能的目的:
  2. querynike时,我们只需取出包含nike的所有词进行编辑距离的计算:
  3. 索引niphone, nike, noodle, ...
  4. 索引iiphone, index, nike, ...
  5. 索引kkindle, nike, ...
  6. 索引eapple, nike, ...
  7. 只需计算很少的词就能获得查询结果,大大提高了计算效率
  8. //存储index采用的数据类型
  9. std::unordered_map<uint32_t, std::map<std::string, int> > index;
  10. //key 值是一个32位整型数,这个数表示词库中每个单词中的一个字(英文字符或中文字)
  11. //value 值是一个map,存储了包含key关键字的所有单词(std::string)及其词频(int)

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注