目录:

  • 0、需求
  • 1、前言
    • 1.1、开发环境
    • 1.1、整体流程
    • 1.2、类
    • 1.3、核心算法
  • 2、接口的设计和实现
    • 2.1、读写 txt 文件的模块
    • 2.2、SimHash 模块(核心模块)
    • 2.3、海明距离模块
    • 2.4、main 主模块
  • 3、接口部分的性能改进
    • 3.1、性能分析
  • 4、单元测试
    • 4.1、读写 txt 文件的模块 的测试
    • 4.2、SimHash 模块 的测试
    • 4.3、海明距离模块 的测试
    • 4.4、主测试 MainTest
  • 5、异常处理
    • 5.1、设计与实现
    • 5.2、测试
  • 6、PSP 表格
  • 7、参考

0、需求

题目:论文查重

描述如下:

设计一个论文查重算法,给出一个原文文件和一个在这份原文上经过了增删改的抄袭版论文的文件,在答案文件中输出其重复率。

  • 原文示例:今天是星期天,天气晴,今天晚上我要去看电影。
  • 抄袭版示例:今天是周天,天气晴朗,我晚上要去看电影。

要求输入输出采用文件输入输出,规范如下:

  • 命令行参数给出:论文原文的文件的绝对路径
  • 命令行参数给出:抄袭版论文的文件的绝对路径
  • 命令行参数给出:输出的答案文件的绝对路径

测试样例使用方法是:orig.txt 是原文,其他 orig_add.txt 等均为抄袭版论文。

  • 测试文件的下载链接

也可以自己创建测试文件。

注意:答案文件中输出的答案为浮点型,精确到小数点后两位

1、前言

  • 注:此项目是博主的软件工程课程的个人编程作业。

  • github地址(点击即可跳转)

1.1、开发环境

  • 编程语言:Java 14

  • IDE:Intellij IDEA 2020.1

  • 项目构建工具:maven

  • 单元测试:JUnit-4.12

  • 性能分析工具:JProfiler 9.2

  • 依赖的外部 jar 包:汉语言处理包

    <dependency><groupId>com.hankcs</groupId><artifactId>hanlp</artifactId><version>portable-1.5.4</version>
    </dependency>
    

1.1、整体流程

1.2、类

  • MainPaperCheck:main 方法所在的类
  • HammingUtils:计算海明距离的类
  • SimHashUtils:计算 SimHash 值的类
  • TxtIOUtils:读写 txt 文件的工具类
  • ShortStringException:处理文本内容过短的异常类

1.3、核心算法

  • simhash+海明距离

  • 具体可参考:

    SimHash 原理与实现

2、接口的设计和实现

2.1、读写 txt 文件的模块

类:TxtIOUtils

包含了两个静态方法:

1、readTxt:读取txt文件

2、writeTxt:写入txt文件

实现:都是调用 Java.io 包提供的接口,比较简单,这里省略。

2.2、SimHash 模块(核心模块)

类:SimHashUtils

包含了两个静态方法:

1、getHash:传入String,计算出它的hash值,并以字符串形式输出,(使用了MD5获得hash值)

2、getSimHash:传入String,计算出它的simHash值,并以字符串形式输出,(需要调用 getHash 方法)

getSimHash 是核心算法,主要流程如下:

1、分词(使用了外部依赖 hankcs 包提供的接口)

List<String> keywordList = HanLP.extractKeyword(str, str.length());//取出所有关键词

2、获取 hash 值

String keywordHash = getHash(keyword);if (keywordHash.length() < 128) {// hash值可能少于128位,在低位以0补齐int dif = 128 - keywordHash.length();for (int j = 0; j < dif; j++) {keywordHash += "0";}}

3、加权、合并

for (int j = 0; j < v.length; j++) {// 对keywordHash的每一位与'1'进行比较if (keywordHash.charAt(j) == '1') {//权重分10级,由词频从高到低,取权重10~0v[j] += (10 - (i / (size / 10)));} else {v[j] -= (10 - (i / (size / 10)));}}

4、降维

String simHash = "";// 储存返回的simHash值for (int j = 0; j < v.length; j++) {// 从高位遍历到低位if (v[j] <= 0) {simHash += "0";} else {simHash += "1";}}

2.3、海明距离模块

  • 类:HammingUtils

包含了两个静态方法:

1、getHammingDistance:输入两个 simHash 值,计算出它们的海明距离 distance

for (int i = 0; i < simHash1.length(); i++) {// 每一位进行比较if (simHash1.charAt(i) != simHash2.charAt(i)) {distance++;}}

2、getSimilarity:输入两个 simHash 值,调用 getHammingDistance 方法得出海明距离 distance,在由 distance 计算出相似度。

return 0.01 * (100 - distance * 100 / 128);

2.4、main 主模块

  • main 方法的主要流程:
  1. 从命令行输入的路径名读取对应的文件,将文件的内容转化为对应的字符串
  2. 由字符串得出对应的 simHash值
  3. 由 simHash值求出相似度
  4. 把相似度写入最后的结果文件中
  5. 退出程序

3、接口部分的性能改进

3.1、性能分析

  • Overview

  • 方法的调用情况

  • 从分析图可以看到:

    调用次数最多的是com.hankcs.hanlp包提供的接口, 即分词、取关键词与计算词频花费了最多的时间。

    所以在性能上基本没有什么需要改进的。

4、单元测试

4.1、读写 txt 文件的模块 的测试

  • 基本思路:

    1、测试正常读取

    2、测试正常写入

    3、测试错误读取

    4、测试错误写入

public class TxtIOUtilsTest {@Testpublic void readTxtTest() {// 路径存在,正常读取String str = TxtIOUtils.readTxt("D:/test/orig.txt");String[] strings = str.split(" ");for (String string : strings) {System.out.println(string);}}@Testpublic void writeTxtTest() {// 路径存在,正常写入double[] elem = {0.11, 0.22, 0.33, 0.44, 0.55};for (int i = 0; i < elem.length; i++) {TxtIOUtils.writeTxt(elem[i], "D:/test/ans.txt");}}@Testpublic void readTxtFailTest() {// 路径不存在,读取失败String str = TxtIOUtils.readTxt("D:/test/none.txt");}@Testpublic void writeTxtFailTest() {// 路径错误,写入失败double[] elem = {0.11, 0.22, 0.33, 0.44, 0.55};for (int i = 0; i < elem.length; i++) {TxtIOUtils.writeTxt(elem[i], "User:/test/ans.txt");}}
}
  • 测试结果:

  • 代码覆盖率:

4.2、SimHash 模块 的测试

public class SimHashUtilsTest {@Testpublic void getHashTest(){String[] strings = {"余华", "是", "一位", "真正", "的", "作家"};for (String string : strings) {String stringHash = SimHashUtils.getHash(string);System.out.println(stringHash.length());System.out.println(stringHash);}}@Testpublic void getSimHashTest(){String str0 = TxtIOUtils.readTxt("D:/test/orig.txt");String str1 = TxtIOUtils.readTxt("D:/test/orig_0.8_add.txt");System.out.println(SimHashUtils.getSimHash(str0));System.out.println(SimHashUtils.getSimHash(str1));}
}
  • 测试结果:

  • 代码覆盖率:

    (SimHashUtils 的 getHash 方法的异常 catch 测试不了)

4.3、海明距离模块 的测试

  • 部分代码:
public class HammingUtilsTest {@Testpublic void getHammingDistanceTest() {String str0 = TxtIOUtils.readTxt("D:/test/orig.txt");String str1 = TxtIOUtils.readTxt("D:/test/orig_0.8_add.txt");int distance = HammingUtils.getHammingDistance(SimHashUtils.getSimHash(str0), SimHashUtils.getSimHash(str1));System.out.println("海明距离:" + distance);System.out.println("相似度: " + (100 - distance * 100 / 128) + "%");}
}
  • 测试结果:

  • 代码覆盖率:

4.4、主测试 MainTest

部分测试代码:

public class MainTest {@Testpublic void origAndAllTest(){String[] str = new String[6];str[0] = TxtIOUtils.readTxt("D:/test/orig.txt");str[1] = TxtIOUtils.readTxt("D:/test/orig_0.8_add.txt");str[2] = TxtIOUtils.readTxt("D:/test/orig_0.8_del.txt");str[3] = TxtIOUtils.readTxt("D:/test/orig_0.8_dis_1.txt");str[4] = TxtIOUtils.readTxt("D:/test/orig_0.8_dis_10.txt");str[5] = TxtIOUtils.readTxt("D:/test/orig_0.8_dis_15.txt");String ansFileName = "D:/test/ansAll.txt";for(int i = 0; i <= 5; i++){double ans = HammingUtils.getSimilarity(SimHashUtils.getSimHash(str[0]), SimHashUtils.getSimHash(str[i]));TxtIOUtils.writeTxt(ans, ansFileName);}}
}
  • 测试结果:

    • 结果文件:

5、异常处理

5.1、设计与实现

当文本长度太短时,HanLp无法取得关键字,需要抛出异常。

        try{if(str.length() < 200) throw new ShortStringException("文本过短!");}catch (ShortStringException e){e.printStackTrace();return null;}

实现了一个处理这个异常的类:ShortStringException(继承了Exception)

public ShortStringException(String message) {super(message);}

5.2、测试

  • 测试结果:

6、PSP 表格

PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟)
Planning 计划 60 60
· Estimate · 估计这个任务需要多少时间 60 60
Development 开发 1120 1290
· Analysis · 需求分析 (包括学习新技术) 300 360
· Design Spec · 生成设计文档 70 60
· Design Review · 设计复审 30 30
· Coding Standard · 代码规范 (为目前的开发制定合适的规范) 30 30
· Design · 具体设计 90 90
· Coding · 具体编码 300 360
· Code Review · 代码复审 60 60
· Test · 测试(自我测试,修改代码,提交修改) 240 300
Reporting 报告 240 250
· Test Repor · 测试报告 60 70
· Size Measurement · 计算工作量 60 60
· Postmortem & Process Improvement Plan · 事后总结, 并提出过程改进计划 120 120
· 合计 1420 1600

7、参考

https://blog.csdn.net/lance_yan/article/details/10304747

https://www.cnblogs.com/huilixieqi/p/6493089.html

Java 简单论文查重程序(SimHash+海明距离算法)相关推荐

  1. Java实现论文查重系统

    1.论文查重系统背景 日常生活中投票是非常常见的一件事,随着现在科技的发展,电子投票都是以线上操作的形势完成,本系统为模仿电子投票系统所作. a.本系统有管理员登录,个人信息修改,用户管理,投票管理, ...

  2. 个人项目作业-论文查重

    目录 1.作业基本信息 2.PSP表格 3.计算模块接口的设计与实现过程 3.1类和方法 3.2类.方法之间的调用关系 3.3 关键方法computeTxtSimilar() 4.计算模块接口部分的性 ...

  3. 目前需要开发出一个功能,对比查找并标注出两篇文章中类似的段落或者词句,有什么开源项目有这个功能吗? 其实有点像论文查重的功能,有论文查重的比较通用的开源项目推荐吗?...

    是的,你可以使用论文查重的工具来对比查找并标注两篇文章之间的相似段落或词句. 你可以尝试使用这些开源项目: MOSS (Measure Of Software Similarity):这是一个用于检测 ...

  4. 计算机论文查重修改吗,程序符号换了知网查重能过吗

    程序符号换了知网查重未必能过的,对于程序代码来说知网也是检测的.对于最新版本的检测系统来说,它对比的数据库中还新增了源代码数据库.所以说,程序代码一样容易出现非常高的重复率,与其它论文一样也是需要降重 ...

  5. 用java设计一个文件查重程序,输入两个文本文件,输出两个文本文件的重复率(最长公共子序列的应用)...

    你可以使用java代码来设计一个文件查重程序.首先,你需要读取两个文本文件的内容,将它们存储在字符串变量中.然后,你可以使用最长公共子序列(LCS)算法来计算两个字符串的重复率. LCS算法的实现方法 ...

  6. (精品)JAVA SSM框架黄淮学院食堂仓库管理系统的设计与实现源码+论文+查重报告+效果、安装视频+ppt模板(已降重)

    项目介绍: (精品)JAVA SSM框架黄淮学院食堂仓库管理系统的设计与实现源码+论文+查重报告+效果.安装视频+ppt模板(已降重) 高清视频演示: https://www.bilibili.com ...

  7. 个人项目——论文查重

    文章目录 项目概述 0. Gitcode链接 1.PSP表格 2.题目描述 3.算法实现基本思路 3.1simHash算法原理 3.2余弦定理查找相似度 4.模块接口部分 5.执行结果 6.代码测试 ...

  8. 个人项目-论文查重/3120005470

    文章目录 1.作业的基本信息 2.作业地址 3.PSP表格 4.算法的设计与实现过程 4.1 分词 4.2 hash 4.3 加权 4.4 合并 4.5 降维 4.6 通过海明距离计算simhash的 ...

  9. 第一次个人编程作业——论文查重

    文章目录 0 作业基本信息 1 作业地址:[https://gitcode.net/willwill0915/papercheck](https://gitcode.net/willwill0915/ ...

  10. lucene配置动态域_学位论文查重中如何使用Lucene全文检索

    本系统用的是Lucene全文检索架构,Lucene作为一个全文检索引擎框架,在构建机制上有众多明显的优点:首先,它可以对任何可以转化成文本格式的数据进行索引的创建,而学术论文包含的doc.PDF和CA ...

最新文章

  1. WPF:自动执行机器人程序若干注意事项
  2. packettotal.com - PacketTotal - A Free Online PCAP Analysis Engine
  3. java中构造器快捷方式_java 构造器 (构造方法)
  4. 技术 Leader 怎样带跨一个团队?
  5. jzoj2137-(GDKOI2004)城市统计【二维前缀和,bfs】
  6. 【工作记录】android手势事件操作记录
  7. C#面向对象17 23种设计模式
  8. UITableView 系列三 :分类显示和改变外观 (实例)
  9. 基于CNN目标检测方法(RCNN,Fast-RCNN,Faster-RCNN,Mask-RCNN,YOLO,SSD)行人检测
  10. log4net用法实例
  11. ISO 9001是什么?ISO 9001 质量管理体系详细介绍
  12. echarts 饼图文字图例多种颜色
  13. 流体力学——漩涡运动
  14. 韩国职业选手风格分析
  15. Newt Scamander的恐惧
  16. 【2016.5.27】再见,软件工程,你好,软件工程。
  17. 洛谷P2141珠心算测验 C++解法
  18. 和USB网络线有关的资料和我遇到的一些问题及解决
  19. 元素故事系列1:土壤团聚体的氮磷循环
  20. ​前端VueRouter解析

热门文章

  1. 计算机网络水晶头闪,网线水晶头坏了怎么办 小妙招一分钟解决你的问题
  2. 会计科目(定义,分类)
  3. Chrome和360浏览器“无法翻译此网页”解决方法
  4. 【HTML】HTML属性
  5. 秋天远程控制V1.0源码(易语言)
  6. docker file镜像分层
  7. 禾多科技与RTI达成合作,加速自动驾驶在中国量产落地
  8. 161张Menhera酱表情包 无水印汉化版
  9. 华为交换机eth口作用_华为5700交换机eth接口做什么用的?怎么使用它?
  10. iOS苹果开发者账号-企业账号、公司账号和个人账号的区别