爬虫数据清洗可视化实战-就业形势分析

news2025/1/14 1:15:01

基于采集和分析招聘网站的数据的芜湖就业形势的调查研究

一、引言

本报告旨在分析基于大数据的当地就业形势,并提供有关薪资、工作地点、经验要求、学历要求、公司行业、公司福利以及公司类型及规模的详细信息。该分析是通过网络爬虫技术对招聘网站的数据进行采集和分析而得出的。
本文部分内容来自网上搜集与个人实践。如果任何信息存在错误,欢迎读者批评指正。本文仅用于学习交流,不用作任何商业用途。

二、薪资范围分布分析

1. 薪资范围分布直方图

在薪资范围分布直方图中,我们对薪资范围的分布情况进行了分析。以下是主要结果:

在这里插入图片描述

  • 图表内容:包括最低薪资和最高薪资的直方图,以薪资范围为 x 轴,招聘数量为 y 轴,以不同颜色表示最低薪资和最高薪资。
  • 本地数据存储:我们保存了薪资范围、最低薪资数量和最高薪资数量的文本文件 salary_distribution.txt,以及直方图图像文件 salary_distribution.png

根据提供的薪资范围和数量数据,我们可以分析出不同薪资范围的招聘情况。以下是对数据的分析:

  • 分析

    • 最低薪资数量大部分集中在0.0-25000.0的范围内,共计521个职位。
    • 最高薪资数量也主要分布在0.0-25000.0范围内,有444个职位。
    • 平均薪资数量在不同薪资范围内有较大差异,最高的平均薪资数量在0.0-25000.0范围内,为498。
    • 在其他薪资范围内,最低薪资数量、最高薪资数量和平均薪资数量相对较少。
  • 结论

    • 大多数招聘职位的薪资范围集中在较低的区间(0.0-25000.0),这可能是大多数职位的薪资水平。
    • 在更高薪资范围内的职位数量较少,但平均薪资水平可能更高。
    • 从薪资范围的分布情况来看,职位的薪资差异较大,求职者需要根据自身情况和期望进行选择。

这个分析可以帮助求职者了解不同薪资范围内的招聘情况,有助于他们做出更明智的职业决策。

三、工作地点分析

2. 不同地点的招聘数量柱状图

我们分析了不同工作地点的招聘数量情况。以下是主要结果:
在这里插入图片描述

  • 图表内容:柱状图展示了不同工作地点的招聘数量,x 轴表示工作地点,y 轴表示招聘数量。

  • 本地数据存储:我们保存了地点和对应招聘数量的文本文件 location_counts.txt,以及柱状图图像文件 location_counts.png

  • 分析

    • 芜湖是最活跃的招聘地点,共有279个招聘职位。
    • 芜湖的各个区域也有一定数量的招聘职位,包括芜湖·鸠江区(80个职位)、芜湖·弋江区(58个职位)、芜湖·芜湖县(23个职位)、芜湖·繁昌县(13个职位)、芜湖·镜湖区(10个职位)和芜湖·三山区(10个职位)。
    • 南京·江北新区、芜湖·无为市、滁州、芜湖·南陵县等地也有一些招聘职位,但数量较少。
    • 其他城市和地区,如马鞍山、宣城、黔南、合肥、安徽省、苏州·姑苏区、镇江、合肥·蜀山区、郑州·高新区、石家庄、上海·嘉定区、南京·鼓楼区、深圳·福田区、淮南、合肥·高新区、宿州、南京·溧水区、商丘、南京·高淳区、杭州·余杭区、上海·闵行区、大连·金州区、南京和郑州等地的招聘数量相对较少,每个地点只有1或2个职位。
  • 结论

    • 芜湖是招聘活动最为集中的地点,有着最多的招聘职位。
    • 除芜湖外,南京的江北新区也有一定数量的招聘职位。
    • 对于求职者来说,了解不同地点的招聘情况有助于选择适合自己的工作地点,以及更好地规划求职策略。

3. 不同地点的平均薪资柱状图

我们进一步分析了不同工作地点的平均薪资情况。以下是主要结果:

在这里插入图片描述

  • 图表内容:柱状图展示了不同工作地点的平均薪资,x 轴表示工作地点,y 轴表示平均薪资。

  • 本地数据存储:我们保存了工作地点和对应平均薪资的文本文件 avg_salary_by_location.txt,以及柱状图图像文件 avg_salary_by_location.png

  • 分析

    • 郑州·高新区的平均薪资最高,为150,000.0元。
    • 滁州、合肥、黔南和安徽省的平均薪资也在较高水平,分别为119,750.0元、82,500.0元、79,000.0元和53,000.0元。
    • 芜湖、南京、杭州、石家庄等地的平均薪资相对较低,不超过15,000.0元。
    • 其他地点的平均薪资也有所不同,显示了不同地区的薪资差异。
  • 结论

    • 工作地点对于平均薪资有显著影响,一些地区的平均薪资明显高于其他地区。
    • 郑州·高新区、滁州、合肥等地的平均薪资较高,可能反映了这些地区的经济发展水平和就业机会较多。
    • 薪资水平相对较低的地区可能需要考虑其他因素,如生活成本和就业机会。

这个分析可以帮助求职者了解不同地点的平均薪资情况,有助于他们在选择工作地点时做出更明智的决策。

四、经验和学历要求分析

4. 不同经验要求和学历要求的招聘数量柱状图

我们分析了不同经验要求和学历要求的招聘数量情况。以下是主要结果:
在这里插入图片描述

  • 图表内容:包括两个子图,左侧子图展示不同经验要求的招聘数量,右侧子图展示不同学历要求的招聘数量。

  • 本地数据存储:我们保存了经验要求和对应招聘数量、学历要求和对应招聘数量的文本文件 experience_education_counts.txt,以及柱状图图像文件 experience_education_counts.png

  • 经验要求分析

    • 最常见的经验要求是3-4年,共有180个职位要求此经验。
    • 其次是2年经验,有102个职位。
    • 1年经验和5-7年经验的职位数量相近,分别为101和88个职位。
    • 无需经验的职位也有一定数量,为52个职位。
    • 对于更高级别的经验要求,10年以上和8-9年的职位数量相对较少,分别为10和9个职位。
  • 学历要求分析

    • 最常见的学历要求是大专,共有257个职位要求此学历。
    • 本科学历要求的职位数量也相对较多,为188个职位。
    • 中技/中专、高中和初中及以下学历要求的职位数量依次减少,分别为52、33和9个职位。
    • 硕士学历要求的职位数量最少,仅有3个职位。
  • 结论

    • 大多数职位对于经验要求集中在1-4年范围内,以及无需经验。
    • 学历要求方面,大专和本科是最常见的学历要求,占据了绝大多数职位。
    • 对于求职者来说,了解不同经验和学历要求的职位分布有助于更好地匹配自己的背景和目标,以寻找适合的工作机会。

五、公司信息分析

5. 不同公司行业的招聘数量柱状图

我们分析了不同公司行业的招聘数量情况。以下是主要结果:
在这里插入图片描述

  • 图表内容:柱状图展示了不同公司行业的招聘数量,x 轴表示公司行业,y 轴表示招聘数量。

  • 本地数据存储:我们保存了公司行业和对应招聘数量的文本文件 industry_counts.txt,以及柱状图图像文件 industry_counts.png

  • 分析

    • 汽车行业
      是招聘数量最多的行业,拥有142个职位。
    • 新能源和电子技术/半导体/集成电路行业分别拥有59和56个职位,排名第二和第三。
    • 机械/设备/重工、汽车零配件、计算机软件等行业也有一定数量的职位,分别为46、39和28个。
    • 金融/投资/证券、多元化业务集团公司、环保、仪器仪表/工业自动化等行业的招聘数量相对较少,都在10个以下。
  • 结论

    • 不同公司行业的招聘情况差异较大,一些行业拥有大量的职位,而其他行业则拥有相对较少的职位。
    • 汽车行业、新能源和电子技术/半导体/集成电路行业可能是求职者关注的热门行业,因为它们有较多的职位机会。
    • 需要注意的是,金融/投资/证券、多元化业务集团公司等行业的招聘数量相对较少,可能竞争较激烈。

这个分析可以帮助求职者了解不同公司行业的招聘情况,有助于他们根据自己的兴趣和专业选择适合的行业和职位。

6. 不同公司行业的平均薪资柱状图

我们进一步分析了不同公司行业的平均薪资情况。以下是主要结果:
在这里插入图片描述

  • 图表内容:柱状图展示了不同公司行业的平均薪资,x 轴表示公司行业,y 轴表示平均薪资。

  • 本地数据存储:我们保存了公司行业和对应平均薪资的文本文件 avg_salary_by_industry.txt,以及柱状图图像文件 avg_salary_by_industry.png

  • 分析

    • 医疗/护理/卫生行业拥有最高的平均薪资,达到116,000元。这可能是因为医疗行业对专业人员有较高的薪资标准。
    • 电子技术/半导体/集成电路行业和计算机软件行业的平均薪资也相对较高,分别为24,696元和19,071元。
    • 金融/投资/证券行业的平均薪资为47,857元,属于高薪行业之一。
    • 汽车、快速消费品(食品、饮料、化妆品)、汽车零配件、建筑/建材/工程等行业的平均薪资也较高,都在10,000元以上。
    • 一些行业如广告、生活服务、中介服务等平均薪资较低,都在10,000元以下。
  • 结论

    • 不同公司行业的平均薪资差异很大,一些行业的平均薪资相对较高,而其他行业则平均薪资较低。
    • 求职者可以根据自己的职业目标和薪资期望选择适合的行业和公司。
    • 需要注意的是,平均薪资只是一个参考指标,实际薪资还会受到个人经验、职位级别、地区等多种因素的影响。

这个分析有助于求职者了解不同行业的薪资水平,以便做出更明智的职业选择。

六、公司福利和类型规模分析

7. 公司福利项出现次数柱状图

我们分析了不同公司福利项的出现次数情况。以下是主要结果:
在这里插入图片描述

  • 图表内容:柱状图展示了不同公司福利项的出现次数,x 轴表示公司福利项,y 轴表示出现次数。

  • 本地数据存储:我们保存了公司福利项和对应出现次数的文本文件 welfare_counts.txt,以及柱状图图像文件 welfare_counts.png

  • 分析:

    • 从柱状图中,你可以看到"五险一金"是最受欢迎的公司福利项,出现次数最多,表明公司普遍提供这项福利以满足员工的基本社会保障需求。

    • "绩效奖金"和"年终奖金"紧随其后,这表明公司也非常重视员工的绩效评估和激励措施。

    • 一些其他福利项,如"员工旅游"和"免费班车",出现次数相对较低,但仍然受到一定关注。

    • "免费员工餐"和"包吃包住"等福利项的出现次数较少,可能只有少数公司提供。

    • 常见福利项:出现次数最多的公司福利项是"五险一金",出现了372次,这表明这是大多数公司都提供的标准福利。其他常见福利项包括"绩效奖金"、“年终奖金”、“餐饮补贴"和"专业培训”。

    • 标配福利:除了"五险一金"外,“员工旅游”、“免费班车”、“交通补贴”、“定期体检”、"通讯补贴"等福利项也被多家公司采用,这些福利通常是吸引员工和提高员工满意度的常见选择。

    • 行业相关福利:某些福利项可能与公司所属行业有关,例如"汽车行业"和"汽车零部件"相关的福利项,包括"汽车"、“出厂检验报告"和"车补”。

    • 个性化福利:还有一些公司可能提供一些较为个性化的福利,如"中医平台推广"、“口吃矫正”、"深度学习"等,这些福利可能根据公司的特殊需求或员工的特殊要求而设定。

  • 结论

    • 福利改进:通过分析这些福利项,您可以了解到其他公司提供的福利,有助于您了解市场竞争和吸引员工的策略。您可以根据这些信息来考虑是否需要改进或新增公司的福利政策,以更好地吸引和留住人才。
    • 这些分析可以帮助你了解不同公司在福利方面的偏好,以及哪些福利项可能对员工招聘和保留产生积极影响。当然,具体的福利选择还可能受到公司规模、行业、地理位置等因素的影响。

8. 不同公司类型及规模的招聘数量柱状图

我们分析了不同公司类型及规模的招聘数量情况。以下是主要结果:

  • 图表内容:柱状图展示了不同公司类型及规模的招聘数量,x 轴表示公司类型及规模,y 轴表示招聘数量。

  • 本地数据存储:我们保存了公司类型及规模和对应招聘数量的文本文件 company_type_and_size_counts.txt,以及柱状图图像文件 company_type_and_size_counts.png。
    在这里插入图片描述

  • 分析

    • 民营公司在不同规模下都有较多的招聘需求,其中民营公司规模在150-500人之间的需求最高,为102个职位。
    • 其次是500-1000人规模的民营公司,招聘数量为66个职位。
    • 国企中,规模在10000人以上的需求较高,有26个职位。
    • 合资公司的需求也相对均衡,500-1000人规模的公司需求最多,为22个职位。
    • 有一些公司没有具体规模标识,招聘数量为16个职位,可能是小型的创业公司或者其他类型的公司。
    • 外资公司在不同规模下的需求相对较少,多数集中在150-500人规模的公司,有16个职位。
    • 已上市的公司需求较为有限,多数集中在10000人以上和500-1000人规模的公司。
  • 结论

    • 民营公司和国企是招聘需求较高的公司类型,尤其是规模在150-500人之间的民营公司。
    • 合资公司和外资公司的需求相对较少,主要集中在500-1000人和150-500人规模的公司。
    • 已上市公司的招聘需求相对较低,多数分布在较大规模的公司中。
    • 招聘者可以根据公司类型和规模的分布情况来更好地选择适合自己的职位和公司。

这个分析有助于求职者了解不同公司类型和规模下的招聘情况,以便做出更明智的职业决策。

七、关键词分析

9. 公司行业词云图

我们生成了公司行业的词云图,以展示公司行业的关键词。以下是主要结果:
在这里插入图片描述

  • 图表内容:词云图展示了不同公司行业的关键词,以词语频率为基础生成,文字颜色随机。

  • 本地数据存储:我们保存了公司行业文本数据到本地文件 industry_wordcloud.txt,以及词云图像文件 industry_wordcloud.png

  • 分析

    • 词云图中显示了不同行业的关键词,其中最常出现的词语包括"汽车"、“电子”、“技术”、“新能源”、"医疗"等。
    • "汽车"是出现频率最高的词语,表明汽车行业在招聘市场上占据重要地位。
    • "电子"和"技术"也是热门行业关键词,说明这两个行业也有较多的招聘机会。
  • 结论

    • 从词云图中可以看出,汽车、电子、技术等行业是当前招聘市场的热门领域,求职者可以关注这些行业的职业机会。
    • 词云图提供了对公司行业数据的可视化呈现,有助于直观地了解不同行业的招聘情况。

10. 公司福利词云图

我们还生成了公司福利的词云图,以展示公司福利的关键词。以下是主要结果:
在这里插入图片描述

  • 图表内容:词云图展示了不同公司福利的关键词,以词语频率为基础生成,文字颜色随机。

  • 本地数据存储:我们保存了公司福利文本数据到本地文件 welfare_wordcloud.txt,以及词云图像文件 welfare_wordcloud.png
    以下是根据福利文本数据生成的词云图和分析:

  • 分析

    • 词云图中显示了不同公司提供的福利待遇,其中最常出现的福利包括"五险一金"、“年终奖金”、“餐饮补贴”、“通讯补贴”、"员工旅游"等。
    • "五险一金"是出现频率最高的福利之一,表明大多数公司提供这一基本的社保福利。
    • "年终奖金"也是一个常见的福利,可能吸引了求职者的关注。
    • 餐饮补贴、通讯补贴等福利也受到一定程度的关注,这些福利有助于提高员工的生活质量。
  • 结论

    • 从词云图中可以看出,五险一金、年终奖金、餐饮补贴等福利是吸引求职者的亮点,公司在招聘时可以重点突出这些福利待遇。
    • 词云图提供了对公司福利数据的可视化呈现,有助于求职者了解不同公司的福利待遇。

八、总结与展望

通过对招聘数据的多维度分析和可视化呈现,我们得出了关于当地就业形势的一系列重要信息。这些信息不仅有助于学生了解就业市场的趋势,还可以为学校和社会提供有关就业形势的重要参考。

未来,我们可以进一步扩展和改进这项调研工作,包括更多数据源的采集、深入的数据分析方法以及更准确的预测模型的建立,以帮助更多人做出明智的职业选择。

九、致谢

感谢你的关注和阅读,希望这份报告对你的职业生涯规划有所帮助。如果有任何问题或需要进一步的咨询,请随时与我们联系。
这份报告代表了我们对就业形势的深入研究,希望这些分析结果对大家有所帮助。

十、相关代码

11. 数据收集代码

# -*- coding = utf-8 -*-
import csv
import time
import random
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.chrome.options import Options

option = Options()
option.add_experimental_option('excludeSwitches', ['enable-automation'])
option.add_argument('--disable-blink-features=AutomationControlled')

driver = webdriver.Chrome(options=option)
driver.execute_script("window.scrollTo(0, document.body.scrollHeight)")
headers = ['职位名称', '薪资范围', '工作地点', '经验要求', '学历要求', '公司名称', '公司类型及规模',
           '公司行业', '公司福利']

jobs_list = []
count = 1
error_time = 0
try:
    driver.get("https://we.51job.com/pc/search?jobArea=150300&keyword=&searchType=2&sortType=0&metro=")
    driver.implicitly_wait(10)
    for j in range(1, 30):
        for i in range(1, 21):
            try:
                job = driver.find_element(By.CSS_SELECTOR, f'div.j_joblist > div:nth-child({i})')
                # print(job.get_attribute('innerHTML'))
                job_name = job.find_element(By.CSS_SELECTOR, '.el > div > span').text
                salary = job.find_element(By.CSS_SELECTOR, '.el > p.info > span.sal').text
                location = job.find_element(By.CSS_SELECTOR, '.el > p.info > span.d.at > span:nth-child(1)').text
                exp = job.find_element(By.CSS_SELECTOR, '.el > p.info > span.d.at > span:nth-child(3)').text
                edu = job.find_element(By.CSS_SELECTOR, '.el > p.info > span.d.at > span:nth-child(5)').text
                company = job.find_element(By.CSS_SELECTOR, '.er > a').text
                company_type_scale = job.find_element(By.CSS_SELECTOR, '.er > p.dc.at').text
                industry = job.find_element(By.CSS_SELECTOR, '.er > p.int.at').text
                try:
                    tag = job.find_element(By.CSS_SELECTOR, '.el > p.tags').text
                except:
                    tag = ''
            except:
                print("error:" + str(i))
                error_time += 1
                if error_time >= 3:
                    input("网络可能断开,输入任意值继续")
                    error_time = 0
                continue
            job_item = {
                '职位名称': job_name,
                '薪资范围': salary,
                '工作地点': location,
                '经验要求': exp,
                '学历要求': edu,
                '公司名称': company,
                '公司类型及规模': company_type_scale,
                '公司行业': industry,
                '公司福利': tag
            }

            jobs_list.append(job_item)

            # 随机等待1-5秒,防止被识别
            time.sleep(random.randint(2, 6))
            print(j,i)
        error_time = 0
        driver.find_element(By.CSS_SELECTOR,
                            f'div.bottom-page > div > div > div> button.btn-next').click()
        time.sleep(random.randint(2, 6))
except Exception as e:
    print("Error:", e)
    input("网络可能断开,输入任意值继续")


finally:
    with open('51job.csv', 'w', newline='', encoding='utf-8-sig') as f:
        writer = csv.writer(f)
        writer.writerow(headers)

        for job_item in jobs_list:
            row = list(job_item.values())
            writer.writerow(row)

    driver.quit()



12.数据清洗与分析代码

from collections import Counter
import pandas as pd
import matplotlib.pyplot as plt
from matplotlib.font_manager import FontProperties
import jieba
from wordcloud import WordCloud
import re
# 设置中文字体
font = FontProperties(fname="C:/Windows/Fonts/simhei.ttf", size=12)
plt.rcParams['font.sans-serif'] = ['SimHei']
# 读取CSV文件
df = pd.read_csv('51job.csv')
# 定义提取薪资函数
def extract_salary(s):
    pattern1 = '([0-9]+\.?[0-9]*)千-([0-9]+\.?[0-9]*)万·([0-9]+\.?[0-9]*)薪'
    pattern2 = '([0-9]+\.?[0-9]*)-([0-9]+\.?[0-9]*)千'
    pattern3 = '([0-9]+\.?[0-9]*)千-([0-9]+\.?[0-9]*)万'
    pattern4 = '([0-9]+\.?[0-9]*)-([0-9]+\.?[0-9]*)千·([0-9]+\.?[0-9]*)薪'
    pattern5 = '([0-9]+\.?[0-9]*)-([0-9]+\.?[0-9]*)万'
    pattern6 = '([0-9]+\.?[0-9]*)-([0-9]+\.?[0-9]*)万·([0-9]+\.?[0-9]*)薪'
    pattern7 = '([0-9]+)元/天'
    match1 = re.search(pattern1, s)
    match2 = re.search(pattern2, s)
    match3 = re.search(pattern3, s)
    match4 = re.search(pattern4, s)
    match5 = re.search(pattern5, s)
    match6 = re.search(pattern6, s)
    match7 = re.search(pattern7, s)
    if match1:
        low, high, extra = match1.groups()
        low, high, extra = float(low), float(high), float(extra)
        return low * 1000, high * 10000, extra
    elif match2:
        low, high = match2.groups()
        low, high = float(low), float(high)
        return low * 1000, high * 1000
    elif match3:
        low, high = match3.groups()
        low, high = float(low), float(high)
        return low * 1000, high * 10000
    elif match4:
        low, high, extra = match4.groups()
        low, high, extra = float(low), float(high), float(extra)
        return low * 1000, high * 1000, extra
    elif match5:
        low, high = match5.groups()
        low, high = float(low), float(high)
        return low * 10000, high * 10000
    elif match6:
        low, high, extra = match6.groups()
        low, high, extra = float(low), float(high), float(extra)
        return low * 10000, high * 10000, extra
    elif match7:
        day = float(match7.group(1))
        return day
    else:
        print(s)
        return None


# 计算平均薪资
try:
    df['最低薪资'] = df['薪资范围'].apply(lambda x: extract_salary(x)[0])
    df['最高薪资'] = df['薪资范围'].apply(lambda x: extract_salary(x)[1])
    df['平均薪资'] = (df['最低薪资'].astype(float) + df['最高薪资'].astype(float)) / 2
except:
    df['日结'] = df['薪资范围'].apply(lambda x: extract_salary(x)[0])
    df['平均薪资'] = df['日结'] * 30

try:
    df['加成'] = df['薪资范围'].apply(lambda x: extract_salary(x)[2])
    df['平均薪资'] += df['加成'] * ((df['最低薪资'].astype(float) + df['最高薪资'].astype(float)) / 2) / 12
except:
    print('无加成')
# 1. 薪资范围分布直方图
plt.figure(figsize=(20, 10))
hist_low = plt.hist(df['最低薪资'], bins=10, alpha=0.5, color='blue', label='最低薪资')
hist_high = plt.hist(df['最高薪资'], bins=30, alpha=0.5, color='red', label='最高薪资')
hist_ave = plt.hist(df['平均薪资'], bins=20, alpha=0.5, color='yellow', label='平均薪资')
plt.xlabel('薪资范围 (k RMB)')
plt.ylabel('招聘数量')
plt.legend()
plt.title('薪资范围分布')

# 添加数据标签和保存数据到文本文件
with open('salary_distribution.txt', 'w', encoding='utf-8') as file:
    # 写入列名
    file.write('薪资范围\t最低薪资数量\t最高薪资数量\t平均薪资数量\n')
    # 遍历直方图的每个x轴区间
    for i in range(len(hist_low[0])):
        # 构造该区间的字符串
        bin_str = f'{hist_low[1][i]}-{hist_low[1][i + 1]}'
        # 获取该区间对应的最低薪资频数
        low_count = int(hist_low[0][i])
        # 获取该区间对应的最高薪资频数
        high_count = int(hist_high[0][i])
        # 获取该区间对应的平均薪资频数
        ave_count = int(hist_ave[0][i])
        # 拼接并写入行数据
        file.write(f'{bin_str}\t{low_count}\t{high_count}\t{ave_count}\n')
plt.savefig('salary_distribution.png')
# 显示图表
plt.show()

# 2. 不同地点的招聘数量柱状图
location_counts = df['工作地点'].value_counts()
# 绘制柱状图
plt.figure(figsize=(20, 10))
location_counts.plot(kind='bar', color='skyblue')
# 设置中文标签
plt.xlabel('工作地点', fontproperties=font)
plt.ylabel('招聘数量', fontproperties=font)
plt.title('不同地点的招聘数量', fontproperties=font)
plt.xticks(rotation=90)
# 添加数据标签和保存数据到文本文件
with open('location_counts.txt', 'w', encoding='utf-8') as file:
    file.write('地点\t招聘数量\n')
    for location, count in location_counts.items():
        file.write(f'{location}\t{count}\n')
plt.savefig('location_counts.png')
# 显示图表
plt.show()

# 3. 不同地点的平均薪资柱状图
avg_salary_by_location = df.groupby('工作地点')['最低薪资'].mean()
# 可以选择展示前几个地点的数据
top_avg_salary_by_location = avg_salary_by_location.sort_values(ascending=False)
# 绘制柱状图
plt.figure(figsize=(20, 10))
top_avg_salary_by_location.plot(kind='bar', color='lightgreen')
# 设置中文标签
plt.xlabel('工作地点', fontproperties=font)
plt.ylabel('平均薪资 (k RMB)', fontproperties=font)
plt.title('不同地点的平均薪资', fontproperties=font)
plt.xticks(rotation=90)
# 添加数据标签和保存数据到文本文件
with open('avg_salary_by_location.txt', 'w', encoding='utf-8') as file:
    file.write('工作地点\t平均薪资\n')
    for location, avg_salary in top_avg_salary_by_location.items():
        file.write(f'{location}\t{avg_salary}\n')
plt.savefig('avg_salary_by_location.png')
# 显示图表
plt.show()

# 4. 不同经验要求和学历要求的招聘数量柱状图
# 统计不同经验要求的招聘数量
exp_counts = df['经验要求'].value_counts()
# 统计不同学历要求的招聘数量
edu_counts = df['学历要求'].value_counts()
# 可视化经验要求的招聘数量
plt.figure(figsize=(20, 10))
plt.subplot(1, 2, 1)
exp_counts.plot(kind='bar', color='skyblue')
plt.xlabel('经验要求', fontproperties=font)
plt.ylabel('招聘数量', fontproperties=font)
plt.title('不同经验要求的招聘数量', fontproperties=font)
# 可视化学历要求的招聘数量
plt.subplot(1, 2, 2)
edu_counts.plot(kind='bar', color='lightgreen')
plt.xlabel('学历要求', fontproperties=font)
plt.ylabel('招聘数量', fontproperties=font)
plt.title('不同学历要求的招聘数量', fontproperties=font)
plt.tight_layout()
# 添加数据标签和保存数据到文本文件
with open('experience_education_counts.txt', 'w', encoding='utf-8') as file:
    file.write('经验要求\t招聘数量\n')
    for experience, count in exp_counts.items():
        file.write(f'{experience}\t{count}\n')

    file.write('学历要求\t招聘数量\n')
    for education, count in edu_counts.items():
        file.write(f'{education}\t{count}\n')
plt.savefig('experience_education_counts.png')
# 显示图表
plt.show()

# 5. 不同公司行业的招聘数量柱状图
# 统计不同公司行业的招聘数量
industry_counts = df['公司行业'].value_counts()

# 可视化不同公司行业的招聘数量
plt.figure(figsize=(20, 10))
industry_counts.plot(kind='bar', color='skyblue')
plt.xlabel('公司行业', fontproperties=font)
plt.ylabel('招聘数量', fontproperties=font)
plt.title('不同公司行业的招聘数量', fontproperties=font)
plt.xticks(rotation=90)

# 添加数据标签和保存数据到文本文件
with open('industry_counts.txt', 'w', encoding='utf-8') as file:
    file.write('公司行业\t招聘数量\n')
    for industry, count in industry_counts.items():
        file.write(f'{industry}\t{count}\n')
plt.savefig('industry_counts.png')
# 显示图表
plt.show()

# 6. 不同公司行业的平均薪资柱状图
# 计算不同公司行业的平均薪资水平
avg_salary_by_industry = df.groupby('公司行业')['平均薪资'].mean()
# 可视化不同公司行业的平均薪资水平
plt.figure(figsize=(20, 10))
avg_salary_by_industry.plot(kind='bar', color='lightgreen')
plt.xlabel('公司行业', fontproperties=font)
plt.ylabel('平均薪资 (k RMB)', fontproperties=font)
plt.title('不同公司行业的平均薪资', fontproperties=font)
plt.xticks(rotation=90)
# 添加数据标签和保存数据到文本文件
with open('avg_salary_by_industry.txt', 'w', encoding='utf-8') as file:
    file.write('公司行业\t平均薪资\n')
    for industry, avg_salary in avg_salary_by_industry.items():
        file.write(f'{industry}\t{avg_salary}\n')
plt.savefig('avg_salary_by_industry.png')
# 显示图表
plt.show()


# 7. 公司福利项出现次数柱状图
# 统计不同公司福利的出现次数
def extract_welfare(welfare_str):
    if isinstance(welfare_str, str):
        return welfare_str.split('\n')
    else:
        return []


# 将福利信息分割成列表
df['公司福利'] = df['公司福利'].apply(extract_welfare)
# 统计不同福利项的出现次数
welfare_counts = df['公司福利'].explode().str.strip().value_counts()
# 处理没有福利信息的公司
if '' in welfare_counts:
    no_welfare_count = welfare_counts['']
    welfare_counts = welfare_counts.drop('')
    welfare_counts['无福利'] = no_welfare_count
# 可视化不同福利项的出现次数
plt.figure(figsize=(20, 10))
welfare_counts.nlargest(20).plot(kind='bar', color='lightblue')
plt.xlabel('公司福利项', fontproperties=font)
plt.ylabel('出现次数', fontproperties=font)
plt.title('不同公司福利项的出现次数', fontproperties=font)
plt.xticks(rotation=90)
# 添加数据标签和保存数据到文本文件
with open('welfare_counts.txt', 'w', encoding='utf-8') as file:
    file.write('公司福利项\t出现次数\n')
    for welfare, count in welfare_counts.items():
        file.write(f'{welfare}\t{count}\n')
plt.savefig('welfare_counts.png')
# 显示图表
plt.show()

# 8. 不同公司类型及规模的招聘数量柱状图
# 统计不同公司类型及规模的招聘数量
company_type_and_size_counts = df['公司类型及规模'].value_counts()
# 可视化不同公司类型及规模的招聘数量
plt.figure(figsize=(20, 10))
company_type_and_size_counts.plot(kind='bar', color='skyblue')
plt.xlabel('公司类型及规模', fontproperties=font)
plt.ylabel('招聘数量', fontproperties=font)
plt.title('不同公司类型及规模的招聘数量', fontproperties=font)
plt.xticks(rotation=90)
# 添加数据标签和保存数据到文本文件
with open('company_type_and_size_counts.txt', 'w', encoding='utf-8') as file:
    file.write('公司类型及规模\t招聘数量\n')
    for company_type, count in company_type_and_size_counts.items():
        file.write(f'{company_type}\t{count}\n')
plt.savefig('company_type_and_size_counts.png')
# 显示图表
plt.show()

# 9. 公司行业词云图
# 合并公司行业文本
industry_text = ' '.join(df['公司行业'].dropna())
# 使用jieba分词
seg_list = jieba.cut(industry_text)
# 创建词云
wordcloud = WordCloud(width=800, height=400, background_color='white',
                      font_path='C:/Windows/Fonts/simhei.ttf').generate(' '.join(seg_list))
# 可视化词云
plt.figure(figsize=(20, 10))
plt.imshow(wordcloud, interpolation='bilinear')
plt.axis('off')
plt.title('公司行业词云', fontproperties=font)
# 将公司行业词频信息保存到本地文件
industry_words = list(jieba.cut(industry_text))
industry_word_counts = Counter(industry_words)
with open('industry_wordcloud.txt', 'w', encoding='utf-8') as file:
    for word, count in industry_word_counts.items():
        file.write(f'{word}: {count}\n')
# 保存词云图像
wordcloud.to_file('industry_wordcloud.png')
# 显示词云图像
plt.show()

# 10. 公司福利词云图
# 合并公司福利文本
welfare_text = ' '.join(df['公司福利'].explode().dropna())
# 使用jieba分词
seg_list = jieba.cut(welfare_text)
# 创建词云
wordcloud = WordCloud(width=800, height=400, background_color='white',
                      font_path='C:/Windows/Fonts/simhei.ttf').generate(' '.join(seg_list))
# 可视化词云
plt.figure(figsize=(20, 10))
plt.imshow(wordcloud, interpolation='bilinear')
plt.axis('off')
plt.title('公司福利词云', fontproperties=font)
# 保存词云图像
wordcloud.to_file('welfare_wordcloud.png')
# 将公司福利文本数据保存到本地文件
welfare_words = list(jieba.cut(welfare_text))
welfare_word_counts = Counter(welfare_words)
with open('welfare_wordcloud.txt', 'w', encoding='utf-8') as file:
    for word, count in welfare_word_counts.items():
        file.write(f'{word}: {count}\n')
# 显示词云图像
plt.show()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/986150.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

校园二手物品交易系统微信小程序设计

系统简介 本网最大的特点就功能全面,结构简单,角色功能明确。其不同角色实现以下基本功能。 服务端 后台首页:可以直接跳转到后台首页。 用户信息管理:管理所有申请通过的用户。 商品信息管理:管理校园二手物品中…

蓝牙服务功能

前言 这阵子用到蓝牙比较多,想写一个专栏专门讲解蓝牙协议及其应用,本篇是第二篇文章,讲解蓝牙服务。 参考网上各大神文章,及瑞萨的文章,参考GPT,并且加入了一些本人的理解。 图片部分源自网络&#xff…

树形控件加自定义图标样式及指引线

记录一下留用&#xff0c;有错误请指正。 效果图如下&#xff1a; 自定义图标及指引线 代码&#xff1a; <div class"head-container" style"margin-left: -15px;"><el-tree icon-class"none"style"height:100%; overflow-y: h…

ffmpeg-android studio创建jni项目

一、创建native项目 1.1、选择Native C 1.2、命名项目名称 1.3、选择C标准 1.4、项目结构 1.5、app的build.gradle plugins {id com.android.application }android {compileSdk 32defaultConfig {applicationId "com.anniljing.ffmpegnative"minSdk 25targetSdk 32…

浏览器进程,性能指标,性能优化

目录 浏览器进程&#xff1a;多进程 主进程&#xff1a;显示、交互&#xff0c;增删进程 UI进程&#xff1a;控制地址栏、书签、前进后退 存储进程&#xff1a;cookie&#xff0c;webstorage&#xff0c;indexDB 渲染进程&#xff1a;每个标签页或窗口都有一个独立的渲染进…

Android11编译第四弹:证书和资源内置

问题&#xff1a;我们智能货柜&#xff0c;终端与云端采用的是MQTT通信&#xff0c;为了更加安全&#xff0c;需要采用HTTPS进行通信。这样&#xff0c;中断需要内置证书。这就需要实现ROM中内置资源&#xff0c;将资源打包到系统中。 一、什么是内置资源 内置资源&#xff0…

LeetCode刷题笔记【29】:动态规划专题-1(斐波那契数、爬楼梯、使用最小花费爬楼梯)

文章目录 前置知识解题思路解题步骤动态规划的debug 509. 斐波那契数题目描述解题思路代码使用dp数组优化空间复杂度: 不用数组, 只用两个变量记录即可 70. 爬楼梯题目描述解题思路代码使用dp数组优化空间复杂度: 不用数组, 只用两个变量记录即可 746. 使用最小花费爬楼梯题目描…

Mybatis传递实体对象只能直接获取,不能使用对象.属性方式获取

mybatis的自动识别参数功能很强大&#xff0c;pojo实体类可以直接写进mapper接口里面&#xff0c;不需要在mapper.xml文件中添加paramType,但是加了可以提高mybatis的效率 不加Param注解&#xff0c;取值的时候直接写属性 //这里是单参数&#xff0c;可以不加param&#xff01…

YOLO的基本原理详解

YOLO介绍 YOLO是一种新的目标检测方法。以前的目标检测方法通过重新利用分类器来执行检测。与先前的方案不同&#xff0c;将目标检测看作回归问题从空间上定位边界框&#xff08;bounding box&#xff09;并预测该框的类别概率。使用单个神经网络&#xff0c;在一次评估中直接…

ARM的异常处理

概念 处理器在正常执行程序的过程中可能会遇到一些不正常的事件发生 这时处理器就要将当前的程序暂停下来转而去处理这个异常的事件 异常事件处理完成之后再返回到被异常打断的点继续执行程序 异常处理机制 不同的处理器对异常的处理的流程大体相似&#xff0c;但是不同的处理器…

VsCode备忘

上次简单学习了一下vscode的使用&#xff0c;结果好长时间没用&#xff0c;今天打开又全忘了。。。再记录一下吧 快捷键 CtrlShiftP 命令面板&#xff0c;查找命令&#xff0c;设置等等 Ctrl 打开集成终端&#xff0c;监视生成输出 Ctrl, 打开设置 CtrlP 转到文件,使用转到符…

提高使用VS Code工作效率的技巧

提高使用VS Code工作效率的技巧 时间轴视图&#xff1a;本地源代码控制 时间轴视图为我们提供了内置的源代码控制。 我们中的许多人都知道 Git 和其他源代码控制工具有多么有用&#xff0c;它们可以帮助我们轻松跟踪文件更改并在需要时恢复到之前的状态。 因此&#xff0c;…

网络威胁防御+资产测绘系统-Golang开发

NIPS-Plus 网络威胁防御资产测绘系统-Golang开发 项目地址&#xff1a;https://github.com/jumppppp/NIPS-Plus NIPS-Plus 是一款使用golang语言开发的网络威胁防御系统&#xff08;内置资产测绘系统&#xff09; 网络威胁流量视图网络威胁详细信息浏览列表网络威胁反制探测攻…

编程中的信号处理和系统 - 初学者指南

信号处理是工程和编程的一个重要领域。 基本上,它允许工程师和程序员改进数据,以便人们可以更有效地使用它。 例如,由于信号处理,电话中的大部分背景噪音都被消除了。这样,通话的另一端就只能听到您的声音。 其他例子有: 音频和音乐软件图像视频处理软件医学影像软件语…

【2023高教社杯】D题 圈养湖羊的空间利用率 问题分析、数学模型及MATLAB代码

【2023高教社杯】D题 圈养湖羊的空间利用率 问题分析、数学模型及MATLAB代码 1 题目 题目 D 题 圈养湖羊的空间利用率 规模化的圈养养殖场通常根据牲畜的性别和生长阶段分群饲养&#xff0c;适应不同种类、不同阶段的牲畜对空间的不同要求&#xff0c;以保障牲畜安全和健康&a…

微信小程序navigateTo进入页面后返回原来的页面需要携带数据回来

需求 如图&#xff1a;点击评论后会通过wx.navigateTo进入到评论页面&#xff0c;评论完返回count给原页面&#xff0c;重新赋值实现数量动态变化&#xff0c;不然要刷新这个页面才会更新最新的评论数量。 实现方式&#xff1a; 在评论页面通过wx.setStorageSync(‘data’…

上传ipa到appstore工具

登录app store connect上架的时候&#xff0c;苹果推荐了三个上传构建版本的工具&#xff0c;一个是xcodde&#xff0c;一个是transporter&#xff0c;一个是命令行工具。但是这几个工具在windows都没有对应的软件可以安装。 因此&#xff0c;假如使用windows电脑&#xff0c;…

1600*C. Maximum Set

解析&#xff1a; 尽可能的增大集合内的数&#xff0c;所以倍数要尽可能的小&#xff0c;所以让最小的数不断乘 2&#xff0c;即可找到最大的数量。 所以&#xff0c;每次计算 k log2&#xff08; y / x &#xff09;,这样可得出最小的 x&#xff0c;乘多少个 2&#xff0c;能…

数据分享|WEKA信贷违约预测报告:用决策树、随机森林、支持向量机SVM、朴素贝叶斯、逻辑回归...

完整报告链接&#xff1a;http://tecdat.cn/?p28579 作者&#xff1a;Nuo Liu 数据变得越来越重要&#xff0c;其核心应用“预测”也成为互联网行业以及产业变革的重要力量。近年来网络 P2P借贷发展形势迅猛&#xff0c;一方面普通用户可以更加灵活、便快捷地获得中小额度的贷…

【css面试题】 实现一个盒子的水平竖直居中对齐效果

面试题里有时还会强调 子盒子宽高是否已知&#xff0c;要注意一下 尝试一&#xff1a;给父盒子设置padding 或者子盒子设置margin <style>.father{width: 300px;height: 200px;overflow: hidden; /* 放坑爹现象&#xff0c;不信你删了试试 */background-color: #db7b7b…