粗糙的刑事量刑模型-随机森林算法

  • 一、效果
    • (一)特征重要性
    • (二)预测精度
    • (三)结果
  • 二、大致思路
  • (一)数据爬取
  • (二)数据处理
    • 1、解压缩
    • 2、去重
    • 3、格式转换
    • 4、文件移动
    • 5、法条分割为匹配的数据集
    • 6、选择罪名和法定刑
    • 7、选择量刑情节
    • 8、加重构成要件的去除
    • 9、模糊匹配相应数据并写入excel中
    • 10、数据调整
    • 11、机器学习
  • 三、总结

一、效果

(一)特征重要性

(二)预测精度

(三)结果

监督学习的几个算法都试过,只有随机森林的测试集效果较好可以达到30%,其余大多只有20%多,当然如果继续调参可能会更好,但是整体而言,大致也仅在30%左右;训练集最好的时候可以到80%以上,但是测试集最高只能30%左右
结论:模型泛化能力不足,拟合能力也不是非常好
可能原因:特征使用的量刑情节使用的是0、1判断,可能过于稀疏而且提取时也可能存在一定问题,可能没有和特定的人匹配完全。需要重新调整参照标签

二、大致思路

(一)数据爬取

数据爬取使用的是selenium+xpath,尝试过使用requests,也进行了js反反爬
部分js反反爬

def __RequestVerificationToken():
"""cipher参数解密"""
start_logger()
with open('J:\\PyCharm项目\\项目\\中国裁判文书网\\requests路线\\cipher.js', encoding='utf-8') as f:jsdata_1 = f.read()
js_1 = execjs.compile(jsdata_1, cwd=r"C:\\Users\\Administrator\\node_modules")
__RequestVerificationToken = js_1.call('random',24)
return __RequestVerificationToken
logging.debug(__RequestVerificationToken())


有关js反反爬需要感谢大佬"越学越害怕"的帮助,虽然最后由于瑞数没有成功。

request没有成功采用的是selenium,链接: 中国裁判文书下载:selenium路线.

最后成功获取28万份裁判文书!!!!!!

但是原本准备获取58万份裁判文书的,失败的原因部分在于网络,有时因为网络延迟导致翻页失败;但是还有一些不知道为什么拿不到,可能是由于通过全国区县的限定词进行全文检索限定的不够精确将部分文书忽略掉了

(二)数据处理

1、解压缩

解压:

def unzip_file(zip_file_name, destination_path):
try:archive = zipfile.ZipFile(zip_file_name, mode='r')for file in archive.namelist():try:archive.extract(file, destination_path)except Exception as e:print(e)continue
except:passdef zipfile_name(file_dir):
# 读取文件夹下面的文件名.zip
L = []
index = 1
for root, dirs, files in os.walk(file_dir):#rint(files)for file in files:if os.path.splitext(file)[1] == '.zip':  # 读取带zip 文件L.append(os.path.join(root, file))print("正在解压缩-----------------第%d个压缩包-------------------------------------" % index)index += 1
return L
# 入口函数
def main():
fn = zipfile_name(input_files.strip('\u202a'))
for file in fn:unzip_file(file, output_files.strip('\u202a'))
if __name__ == "__main__":main()

这是是从csdn上复制下来的,自己用的也是这个,应该是来自博主:DesolatePoison的《https://blog.csdn.net/qq_40695642/article/details/100170316?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522159357139119725211911252%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fall.%2522%257D&request_id=159357139119725211911252&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2allfirst_rank_ecpm_v1~rank_ctr_v4-3-100170316.ecpm_v1_rank_ctr_v4&utm_term=%E6%96%87%E4%BB%B6%E6%89%B9%E9%87%8F%E8%A7%A3%E5%8E%8B%E7%BC%A9python》

整体解压时间似乎也较长大体上应该花了5-6个小时

2、去重

由于爬取时使用全文检索,不可避免的会存在下载的裁判文书重复,因而写了去重的程序

def repeat_file_find():
repeat_files = {}
for file_name in file_names:

    if file_name in repeat_files:repeat_files[str(file_name)] += 1else:repeat_files[str(file_name)] = 1
'''排序'''
repeat_files_ = sorted(repeat_files.items(), key=lambda x: x[1], reverse=True)
pprint.pprint(repeat_files_[:100])
with open("重复文件有哪些.json", "w+")as f:json.dump(repeat_files_,f)
#文件dump,数据dumps
#repeat_file_find()
data = open("重复文件有哪些.json", "r", encoding='utf-8',      errors='Egnore').read()
datas = json.loads(data)
for i in datas:print(i)

但是奇怪的是:似乎没有重复的文件??????、

该内容已经经过排序了,最多的排在前面,但是这里没有重复的文件

疑惑:似乎解压后不用去除,可以自动去重?
好像确实是这样,在后面移动也是如此

3、格式转换

裁判文书下载下来的是压缩包,压缩包内文件格式为doc

python不识别doc,python-docx只能使用docx格式的,但是下载的文书格式均为doc格式的,因为需要转为docx格式的

比较坑的是:
csdn上转换的docx,都是使用word实现的,但是我用到是wps,用word也能转换为docx,不报错。
但是读取时就会无法读取,csdn上说加空格,这是没有什么用的,原因就在于缺失了word,应该是这个原因,因而转为能够读取的docx需要使用先用的wps程序

代码:

def doc_to_docx(path):
"""
DOC转为docx,wps格式
"""index_ = 1
"""打开wps"""
#word = wc.gencache.EnsureDispatch('kwps.application')#有缓存之类的
#重复跳过一:
with open("J:\PyCharm项目\项目\中国裁判文书网\selenium路线\数据清洗模块\文书由doc转为docx\docx已经有的.txt","r",encoding="utf-8")as f:list_ = f.readlines()
# 重复跳过二:
with open("J:\PyCharm项目\项目\中国裁判文书网\selenium路线\数据清洗模块\docx移动至新文件夹中\已经存在的docx.txt","r",encoding="utf-8")as f:list_2 = f.readlines()
"""遍历doc文档"""
for index,path_ in enumerate(glob.glob(os.path.join(path, "*"))):path1 = path_ + "\n"if path1 in list_ or path1 in list_2:#print("重复跳过中-----------------------------------")passelse:try:word = wc.Dispatch('kwps.application')doc = word.Documents.Open(path_)# 错在没有加入文件名,位置加名字加格式"""保存在同一文件夹下命名为docx,注意此处必须为docx否则转换不成功,必须为12否则无法读取"""doc.SaveAs2(path_.split('.')[0] + ".docx", 12)# print(path_.split('.')[0] + ".docx")# 保存在其他地方的 文件夹# doc.SaveAs(output + path_.split("/")[-1].split(".")[0] + ".docx", 16)print('正在进行格式转换--------------------------%d---------------------------------' % index)print("页面关闭第%d次---------------------------------------"%index_)index_ += 1doc.Close()if index_ < 10000:passelse:word.Quit()print("wps程序退出中--------------------%d---------"%index_)index_ = 1except Exception as e:print(e)

关键:

word = wc.Dispatch('kwps.application')#是kwps而不是worddoc.SaveAs2(path_.split('.')[0] + ".docx", 12)#参数为12

这样转换比较忙,用了二天多,而这过程中无法使用wps,否者会乱

如果有更好的方法请一定私信我!!!!!!!!

4、文件移动

当然这个可以在第3步的时候直接docx保存到指定的文件夹中
文件移动过程中发现,确实可以去重,因为重复文件不会移动到指定的文件夹中去

5、法条分割为匹配的数据集

当然这有个前提就是先要读取docx文档内容
我用python-docx有时能读有时不能,就换成了docx2txt
主要代码:

 text = docx2txt.process(path)print(text)

通过逗号、句号进行分割形成匹配数据集
def split_docx_judments(path):
“”“刑事诉讼法、民事诉讼法、行政诉讼法基本可以共用一套函数”""
writer = []
index = 1
list_ = []
text = read(path).replace("\n", “”).replace("\u3000", “”).replace(" ", “”).strip()

pattern = re.compile("判决如下:被告人(.*?)犯(.*?),判处有期徒刑(.*?),")
match = re.findall(pattern,text)
if len(match) > 0:#print("文书内容:",match)for content_ in match:if "强奸罪" in content_:#print("命中文书内容:", i)content = text.split(",")"""一个一个切分"""for i in content:for i_ in i.split("。"):for i_1 in i_.split(";"):for i_2 in i_1.split(":"):for i_3 in i_2.split("、"):writer.append(i_2.strip())return writer,content_
else:  # 补上这一句是为了防止匹配失败时的处理也是非len(match) > 0content = "非正常裁判文书"return content

分割完成后为:

当然也可以通过jieba进行切词,但是切词的用来作为匹配时的错误可能性就会比较高,因而我选用的是这种方法

6、选择罪名和法定刑

不能进行全罪名,这样有些量刑情节会出现问题

用正则提取,主要内容

 pattern = re.compile("【(.*?)】.*?的,处(.*?)。")content_need = re.findall(pattern, writer)pattern_ = re.compile("【(.*?)】.*?的,对(.*?)。")

提取后放入csv

7、选择量刑情节

通过《量刑指导意见》进行提取,核心的正则匹配

 pattern = re.compile("量刑要素的细化(.*?)第二十七条")

结果为:

此处部分内容进行了人工填补

8、加重构成要件的去除

选择的罪名为:强奸罪,原因相对重要的罪名,且其基础法定刑为3-10年,可能较好处理

去除加重构成要件核心在于模糊匹配

def clear_high_simple(path):
"""简单判断是否超过十年"""
try:if sp(path) == "非正常裁判文书":message = "非正常裁判文书"return messageelse:text, match = sp(path)Arabic_numerals = cm(str(match))if Arabic_numerals >= 10:with open("J:\PyCharm项目\项目\中国裁判文书网\selenium路线\法条内容提取\加重构成要件排除\强奸罪加重.txt", "r")as f:for i in f.readlines():for i1 in text:# print(fuzz.partial_ratio(i.strip("\n"), i1))fraction = fuzz.partial_ratio(i.strip("\n"), i1)if fraction > 80 and len(i1) > 2:number = 1return numberelse:number = 0return number
except Exception as e:print(e)

9、模糊匹配相应数据并写入excel中

模糊匹配前需要先对文书内容进行特点罪名的读取,只选择选定大多罪名
选择选定的罪名进行相应文书的选择

def split_docx_judments(path):
"""刑事诉讼法、民事诉讼法、行政诉讼法基本可以共用一套函数"""
writer = []
index = 1
list_ = []
text = read(path).replace("\n", "").replace("\u3000", "").replace(" ", "").strip()pattern = re.compile("判决如下:被告人(.*?)犯(.*?),判处有期徒刑(.*?),")
match = re.findall(pattern,text)
if len(match) > 0:#print("文书内容:",match)for content_ in match:if "强奸罪" in content_:#print("命中文书内容:", i)content = text.split(",")"""一个一个切分"""for i in content:for i_ in i.split("。"):for i_1 in i_.split(";"):for i_2 in i_1.split(":"):for i_3 in i_2.split("、"):writer.append(i_2.strip())return writer,content_
else:  # 补上这一句是为了防止匹配失败时的处理也是非len(match) > 0content = "非正常裁判文书"return content

这种选择相对比较成功,因为裁判文书的判决基本格式为:

即其格式化相对模型,正则提取后的人与罪能够在一起

然后是整篇文书相应内容的写入

 for i in text:# print(i)"""一直出现问题在于txt路径是相对路径"""with open("J:\PyCharm项目\项目\中国裁判文书网\selenium路线\法条内容提取\量刑情节.txt", "r", encoding="utf-8")as f:condition = f.readlines()for i_ in condition:if fuzz.partial_ratio(i_.strip("\n"), i) > 90:element_lists[title] = 1dataframes[i_.strip("\n")] = element_lists"""注意此处的缩进程度需到此处即for处,继续缩进存在无法更新过快"""for i_ in condition:element_list[title] = 0dataframe[i_.strip("\n")] = element_list

这里有个问题就是字典的更新,直接if.else会对内容进行覆盖,此处原因不知道是字典本身的原因还是由于我使用了多线程导致的

进行字典更新,比较复杂的写法

 """字典更新"""dataframe_ = dict(dataframe, **dataframes)"""法定刑更新"""dataframe_1 = {"法定刑": {title: 6}}dataframe_2 = dict(dataframe_, **dataframe_1)"""文书长度更新"""dataframe_3 = {"裁判文书长度": {title: len(''.join(text))}}dataframe_4 = dict(dataframe_2, **dataframe_3)"""宣告刑更新"""dataframe_5 = {"宣告刑": {title: match[2]}}dataframe_6 = dict(dataframe_4, **dataframe_5)pprint.pprint("更新后的字典 :")pprint.pprint(dataframe_6["宣告刑"][title])pprint.pprint(dataframe_6)

之后就是excel的写入

10、数据调整

合并excel,核心代码:

pd.concat(data_frames, axis=0, ignore_index=True)

数据内容处理,主要是分组,将机器学习的参数标签与预测值进行分开,如果写入做的比较好就不需要进行这样的数据处理

def data_site_change():
content = pd.read_excel(r"F:\360下载\提取后的excel文件(测试集)\合并的excel\合并的量刑模型数据(原始).xls".strip('\u202a'))
# 位置调换
dataframe_2 = content.宣告刑
# df已经变更为content
df = content.drop('宣告刑', axis=1)
#print("修改时:    ")
#print(df)
df.insert(1, '宣告刑', dataframe_2)
#print("修改后:    ")
x = df.iloc[:, 2:]
#print(x)
y = df.iloc[:, 1:2]
df["宣告刑"] = df["宣告刑"].map(lambda x:cm(x))
df["法定刑"] = df["法定刑"].map(lambda x:p(x))
print(df)
writer = pd.ExcelWriter(r"‪F:\360下载\提取后的excel文件(测试集)\合并的excel\合并的量刑模型数据_2.xls".strip('\u202a'))
df.to_excel(writer, sheet_name='强奸罪量刑模型修改', index=False)
writer.save()
data_site_change()

这里的关键在于中文数字转为阿拉伯数字数字

转换代码:将csdn上的一个博主的中文转数字进行了加工

def chinese_math(need_change):
"""注意单位:此时是月"""
time_year = 0
time_month=0
for i, key in enumerate(list(need_change)):if key == "年":number = list(need_change)[i - 1]# print(list(need_change)[i-1])if number in word.keys():time_year = change(number) *12elif key == "月":number_1 = list(need_change)[i - 2]if number_1 in word.keys():time_month = change(number_1)time = time_year + time_month
#print(str(time) + "年")
return time

need_change = “三年七个月。(刑期从判决执行之日起计算。判决执行以前先行羁押的”
print(chinese_math(need_change))

全部转换为月份,因为全部转为年份,在训练时汇报数据类型错误(实测比较麻烦,所以转为月份),使用int()强制转换会造成宣告刑的不准确,因而转为月份较好。

修改前:

修改后:

当然此处的法定刑并不是进行了转换,而是修改,因为之前使用中线理论设置为6年,但是拟合能力不足,于是通过随机数增加权重的方法进行了随机化以模拟幅度理论(其实影响不大,几乎没有什么影响)

11、机器学习

基本都是使用监督学习算法
首先进行数据类型转换

df = pd.read_excel(file)
X_ = df.iloc[:,2:]
y_ = df.iloc[:,1:2]
"""变成列表"""
X = X_.values.tolist()
y_1 = y_.values.tolist()
"""两层列表变成一层"""
y = sum(y_1,[])
"""数组化"""
X= np.array(X)
y = np.array(y)

最终结果需要变为数组,即

sklearn的数据集类型也是这样

<class 'numpy.ndarray'>

否者报错:

ValueError: Number of features of the model must match the input. Model n_features is 44 and input n_features is 2

第一个是k近邻算法的拟合得分,第二个是随机森林的得分

拟合效果以及测试集的拟合效果均不是很好,特别是测试集的拟合
最后一个list格式的是预测值,k近邻算法反而比拟合效果好的随机森林要准确,正确的值为40个月

github仓库中的一个量刑辅助的测试集得分只有0.16

可见可能是相关标签的抽取的多少以及准确度问题

三、总结

机器学习效果不太好的原因可能是量刑情节设置的不太合理,但是也有本身不完善的原因,
等待下一步调整

粗糙的量刑模型-随机森林算法相关推荐

  1. R语言使用randomForest包构建随机森林模型的步骤和流程、随机森林算法包括抽样样本(观察)和变量来创建大量的决策树(多个树,构成了森林,而且通过样本抽样和变量抽样,让多个树尽量不同)

    R语言使用randomForest包中的randomForest函数构建随机森林模型的步骤和流程(Random forests).随机森林算法包括抽样样本(观察)和变量来创建大量的决策树(多个树,构成 ...

  2. RandomForest 随机森林算法与模型参数的调优

    公号:码农充电站pro 主页:https://codeshellme.github.io 目录 1,集成算法之 bagging 算法 2,随机森林算法 3,随机森林算法的实现 4,随机森林算法的使用 ...

  3. Python数据挖掘项目:构建随机森林算法模型预测分析泰坦尼克号幸存者数据

    作者CSDN:进击的西西弗斯 本文链接:https://blog.csdn.net/qq_42216093/article/details/120196972 版权声明:本文为作者原创文章,未经作者同 ...

  4. 利用mysql建立随机森林_随机森林算法实例 - osc_4imme0wh的个人空间 - OSCHINA - 中文开源技术交流社区...

    根据成年人数据集来预测一个人的收入 1.准备数据集 我下载好了一个成年人数据集,从百度云下载 链接:https://pan.baidu.com/s/10gC8U0tyh1ERxLhtY8i0bQ 提取 ...

  5. Bagging与随机森林算法原理小结

    点击上方"小白学视觉",选择加"星标"或"置顶" 重磅干货,第一时间送达 在集成学习原理小结中,我们讲到了集成学习有两个流派,一个是boos ...

  6. 随机森林算法demo python spark

    关键参数 最重要的,常常需要调试以提高算法效果的有两个参数:numTrees,maxDepth. numTrees(决策树的个数):增加决策树的个数会降低预测结果的方差,这样在测试时会有更高的accu ...

  7. spark 随机森林算法案例实战

    随机森林算法 由多个决策树构成的森林,算法分类结果由这些决策树投票得到,决策树在生成的过程当中分别在行方向和列方向上添加随机过程,行方向上构建决策树时采用放回抽样(bootstraping)得到训练数 ...

  8. 随机森林算法4种实现方法对比测试:DolphinDB速度最快,XGBoost表现最差

    随机森林是常用的机器学习算法,既可以用于分类问题,也可用于回归问题.本文对scikit-learn.Spark MLlib.DolphinDB.XGBoost四个平台的随机森林算法实现进行对比测试.评 ...

  9. (转载)机器学习知识点(十七)Bagging与随机森林算法原理小结

    集成学习有两个流派,一个是boosting派系,它的特点是各个弱学习器之间有依赖关系.另一种是bagging流派,它的特点是各个弱学习器之间没有依赖关系,可以并行拟合.本文就对集成学习中Bagging ...

最新文章

  1. 面试官:说说什么是Java内存模型?
  2. 布局覆盖 超出一部分_Android 布局优化
  3. 网易盖楼回复的简易实现
  4. wordpress 常用函数-wpdb类
  5. 2021-03-15 Lp信号
  6. IE6下PNG图片透明效果(PNG图片做背景也可以)
  7. 运算符重载为类的友元函数
  8. iPhone:你知道这 13 年我是怎么过的吗?
  9. Looksery Cup 2015 B. Looksery Party 暴力
  10. acm 算法竞赛 时间
  11. android设备多屏互动,大神教你玩转多屏互动-PC投屏安卓篇
  12. 安川焊接机器人做圆弧运动编程_安川机器人示教器编程插补方式详细介绍--安川机器人...
  13. pubg解封验证计算机怎么验证,绝地求生解封教程
  14. Collecting package metadata (current_repodata.json): fail亲测成功
  15. Docker 1.12实践:Docker Service、Stack与分布式应用捆绑包
  16. [日推荐] 『紫砂壶平台』一把紫砂,一种品味
  17. UML画图总结以及浅谈UNL九种图
  18. 学大伟业:学长是如何对待数学竞赛的
  19. 海康威视网络摄像头配置本地存储服务器(远程连接查看回放)
  20. time_wait,不要乱用tcp_tw_recycle和net.ipv4.tcp_tw_reuse

热门文章

  1. Android ListView滑动删除及响应事件详解
  2. win10网络连接不上问题解决方法
  3. 一位新竹科学园区朋友的来信
  4. Arduino控制土壤湿度传感器
  5. python生成微信个性签名的词云图
  6. js 时分秒比较大小
  7. 关于处理npm ERR! Unexpected token in JSON at position...
  8. 将网站、网页变成灰白色调的哀悼风格界面的CSS代码
  9. Kafka(三):kafka消费者
  10. wps高亮怎么取消_一组WPS表格小技巧,简单实用