from selenium.webdriver.chrome.options import Options
from selenium import webdriver
from selenium.webdriver.common.by import By
import time
import os
import random


def get_driver():
    # 检查是否已经打开浏览器
    browser_opened = False
    for handle in webdriver.Chrome().window_handles:
        browser_opened = True
        break

    # 创建浏览器实例或在已有浏览器中操作
    if browser_opened:
        options = Options()
        options.debugger_address = '127.0.0.1:9222'
        driver = webdriver.Chrome(options=options)
    else:
        os.system(r'start chrome --remote-debugging-port=9222 --user-data-dir="D:\评阅用"')
        options = Options()
        options.add_experimental_option('debuggerAddress', '127.0.0.1:9222')
        driver = webdriver.Chrome(options=options)

    return driver


driver = get_driver()

# 找到有'内蒙古开放大学'字样的标签页
while True:
    for handle in driver.window_handles:
        driver.switch_to.window(handle)
        if '内蒙古开放大学' in driver.title:
            print('登录成功')
            break
    else:
        time.sleep(3)
        continue
    break

while True:
    # 等待题目组元素出现
    while True:
        try:
            elements = driver.find_elements(By.CLASS_NAME, 'card-div')
            if elements:
                break
        except:
            pass
        time.sleep(3)

    score_list = []
    subject_number_list = []
    for element in elements:
        # 查找待批改元素
        card_score_element = element.find_element(By.CLASS_NAME, 'card-score')
        if '待批改' not in card_score_element.text:
            continue
        # 查找题号元素
        subject_number_element = element.find_element(By.CLASS_NAME, 'subject-number')
        subject_number_text = subject_number_element.text
        # 提取题号
        subject_number = ''
        for char in subject_number_text:
            if char.isdigit():
                subject_number += char
        subject_number_list.append(subject_number)
        print('待批改的题目号为:', subject_number)

        # 判断是否作答
        answered = False
        elements = driver.find_elements(By.ID, f'subject-{subject_number}')
        for element in elements:
            answer_content_element = element.find_elements(By.CLASS_NAME, 'answer-content')
            if answer_content_element:
                answered = True
                break
        if not answered:
            print(f'第{subject_number}题未作答,不获取分值')
            continue

        # 查找题目分值
        summary_numbers = []
        for element in elements:
            summary_element = element.find_element(By.CLASS_NAME, 'summary-sub-title')
            summary_text = summary_element.text
            summary_number = ''
            for char in summary_text:
                if char.isdigit():
                    summary_number += char
            summary_numbers.append(int(summary_number))
            print(f'第{subject_number}题的分值为:{summary_number}')

        # 将题目分值存入列表
        score_list.extend(summary_numbers)

    # 如果只有一个题目,则对这道题的分值减去1-2分
    if len(score_list) == 1:
        score_list[0] -= random.randint(1, 2)
        print(f'第{subject_number_list[0]}题减{score_list[0]}分,已填入{score_list[0]}分')
    else:
        # 随机选择2-5个题目减分
        # 确保减分数量不超过总题目数量
        reduce_index_list = random.sample(range(len(score_list)), min(random.randint(2, 5), len(score_list)))

        # 对选择的题目减分
        for i in reduce_index_list:
            subject_number = subject_number_list[i]
            reduce_score = random.randint(1, 2)
            score_list[i] -= reduce_score
            print(f'第{subject_number}题减{reduce_score}分,已填入{score_list[i]}分,其他题满分')

    # 将分值填入分值框中
    for i, subject_number in enumerate(subject_number_list):
        elements = driver.find_elements(By.ID, f'subject-{subject_number}')
        for element in elements:
            score_element = element.find_element(By.NAME, 'score')
            score_element.clear()
            score_element.send_keys(str(score_list[i]))
            time.sleep(0.5)

    # 等待提交成功
    while True:
        try:
            driver.find_element(By.CLASS_NAME, 'toast-message')
            time.sleep(1)
        except:
            break

    # 点击下一个学生
    next_student_element = driver.find_element(By.CLASS_NAME, 'icon-student-right-narrow')
    try:
        next_student_element.click()
    except:
        print('已到达最后一个学生')

# 处理运行报错:'Sample larger than population or is negative'
# 在选择减分题目数量时,确保不超过总题目数量
reduce_index_list = random.sample(range(len(score_list)), min(random.randint(2, 5), len(score_list)))

代码解析:

  1. 获取浏览器驱动:

    • 使用 get_driver() 函数获取 Chrome 浏览器驱动,支持在已有浏览器中操作或新创建浏览器。
  2. 找到目标网页:

    • 通过循环遍历所有标签页,找到包含 '内蒙古开放大学' 字样的标签页,并进行登录操作。
  3. 寻找待批改题目:

    • 通过 card-div 类名找到所有题目组元素。
    • 通过 card-score 类名找到待批改的题目。
    • 通过 subject-number 类名找到题号,并提取数字部分。
  4. 判断题目是否作答:

    • 通过 answer-content 类名判断题目是否已经作答。
  5. 获取题目分值:

    • 通过 summary-sub-title 类名找到题目分值,并提取数字部分。
  6. 随机减分:

    • 如果只有一个题目,则随机减去 1-2 分。
    • 如果有多个题目,则随机选择 2-5 个题目进行减分,确保减分数量不超过总题目数量。
    • 随机减去 1-2 分。
  7. 填写分值:

    • 找到分值输入框 (score 属性),清空并填入分数。
  8. 等待提交成功:

    • 通过 toast-message 类名等待提交成功提示。
  9. 点击下一个学生:

    • 通过 icon-student-right-narrow 类名找到并点击下一个学生按钮。

错误处理:

  • 针对 ValueError: Sample larger than population or is negative 错误,在随机选择减分题目数量时,使用 min(random.randint(2, 5), len(score_list)) 确保选择的数量不超过总题目数量,避免出现错误。

优化建议:

  • 可以根据实际需要调整减分逻辑,例如根据不同题型的难度进行不同的减分操作。
  • 可以添加一些异常处理机制,避免程序在遇到意外情况时崩溃。
  • 可以将代码封装成函数,方便重复使用。
  • 可以使用更加专业的 Selenium 库,例如 SeleniumBase,简化代码编写。
  • 可以使用 requests 库代替 Selenium 获取网页内容,提高效率。

使用说明:

  1. 安装必要的 Python 库:pip install selenium
  2. 将代码保存为 *.py 文件。
  3. 修改代码中的 user-data-dir 路径,指向 Chrome 浏览器配置文件所在的文件夹。
  4. 运行代码,即可自动批改作业。

注意:

  • 本代码仅供参考,请根据实际情况进行修改和调整。
  • 使用代码前,请确保已了解 Selenium 库的基本用法。
  • 请勿使用本代码进行任何非法或不道德的操作。
  • 使用本代码需谨慎,并承担相应的风险。

希望以上信息能帮到您!

Selenium 自动化批改:随机减分并填写分数

原文地址: https://www.cveoy.top/t/topic/oKKA 著作权归作者所有。请勿转载和采集!

免费AI点我,无需注册和登录