V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
timerring
V2EX  ›  问与答

佬们,像这种 F12 一直调用 debugger 不让调试的网站有什么通用解开限制的解决办法吗?

  •  
  •   timerring · 45 天前 · 1104 次点击
    这是一个创建于 45 天前的主题,其中的信息可能已经有所发展或是发生改变。

    网站 什么值得买: https://www.smzdm.com/

    5 条回复    2025-06-03 22:03:34 +08:00
    linauror
        1
    linauror  
       45 天前   ❤️ 1
    一般切换到 source ,会自动定位到“debugger”代码的位置,右键添加到忽略列表,基本就可以了
    hafuhafu
        2
    hafuhafu  
       45 天前   ❤️ 1
    直接工具栏里点停用断点 Ctrl+F8 ,然后继续...
    覆盖源码也行。
    timerring
        3
    timerring  
    OP
       45 天前
    @linauror 还真是,感谢佬!
    strugglers
        4
    strugglers  
       45 天前   ❤️ 1
    // ==UserScript==
    // @name 移除开发者工具防护
    // @namespace http://tampermonkey.net/
    // @version 2025-04-02
    // @description 移除页面中监听开发者工具打开的行为
    // @author struggler
    // @match *://*/*
    // @icon 
    // @grant none
    // @run-at document-start
    // ==/UserScript==

    ;(function () {
    "use strict"

    // 主防护函数
    const applyProtections = () => {
    // 1. 重写关键函数和构造函数
    overrideFunctionConstructors()

    // 2. 修改定时器行为
    modifyTimers()

    // 3. 处理 eval 和错误捕获
    handleEvalAndErrors()

    // 4. 反开发者工具检测
    antiDevToolsDetection()

    // 5. 移除事件监听
    //removeEventListeners()

    // 6. 覆盖常见检测函数
    overrideDetectionFunctions()

    // 7. 阻止 debugger 检测
    preventDebuggerDetection()

    console.log("[开发者工具防护] 所有防护措施已激活")
    }

    // 1. 重写关键函数和构造函数
    const overrideFunctionConstructors = () => {
    // 重写 Function 构造函数
    Function.prototype.constructor_ = Function.prototype.constructor
    Function.prototype.constructor = function (code) {
    if (code === "debugger") {
    return function () {}
    }
    return Function.prototype.constructor_.call(this, code)
    }
    }

    // 2. 修改定时器行为
    const modifyTimers = () => {
    // 重写 setInterval 函数
    const originalSetInterval = window.setInterval
    window.setInterval = function (callback, delay) {
    if (callback && callback.toString) {
    var funString = callback.toString()
    if (funString.indexOf("debugger") > -1) return
    if (funString.indexOf("window.close") > -1) return
    }
    delay = delay * 10000
    return originalSetInterval(callback, delay)
    }
    }

    // 3. 处理 eval 和错误捕获
    const handleEvalAndErrors = () => {
    // 保存原始的 try-catch 逻辑
    const originalTryCatch = {
    try: window.eval,
    catch: window.onerror,
    }

    // 劫持 eval
    window.eval = function (code) {
    return originalTryCatch.try.apply(this, arguments)
    }

    // 劫持全局错误处理
    window.onerror = function () {
    return true
    }
    }

    // 4. 反开发者工具检测
    const antiDevToolsDetection = () => {
    // 禁用基于 debugger 的检测
    window.debugger = function () {}
    window.debugger.toString = () => "function debugger() { [native code] }"
    window.console.clear = () => {}

    // 禁用基于 console.log 的检测
    if (!window.console) window.console = {}

    // 禁用基于时间差的检测
    const originalDate = window.Date
    window.Date = new Proxy(window.Date, {
    construct(target, args) {
    if (args.length === 0) {
    return new originalDate(originalDate.now())
    }
    return new originalDate(...args)
    },
    })

    // 禁用基于 Function.toString 的检测
    const originalToString = Function.prototype.toString
    Function.prototype.toString = function () {
    if (this === window.console.log || this === window.debugger) {
    return "function() { [native code] }"
    }
    return originalToString.call(this)
    }

    // 禁用基于窗口大小的检测
    Object.defineProperty(window, "outerWidth", {
    get: () => window.innerWidth + 100,
    configurable: false,
    })
    Object.defineProperty(window, "outerHeight", {
    get: () => window.innerHeight + 100,
    configurable: false,
    })
    }

    // 5. 移除事件监听
    const removeEventListeners = () => {
    // 覆盖 addEventListener
    const originalAdd = EventTarget.prototype.addEventListener
    EventTarget.prototype.addEventListener = function (type) {
    if (["keydown", "resize"].includes(type)) {
    return
    }
    originalAdd.apply(this, arguments)
    }

    // 清除特定事件属性
    ;["onkeydown", "onresize"].forEach((prop) => {
    if (window[prop]) window[prop] = null
    if (document[prop]) document[prop] = null
    })
    }

    // 6. 覆盖常见检测函数
    const overrideDetectionFunctions = () => {
    window.__detectDevTools = function () {
    return false
    }
    window.__CHROME_DEVTOOLS_EXTENSION_DETECTED__ = false
    window.devtools = { isOpen: false }
    window.performance.now = function () {
    return Date.now()
    }
    }

    // 7. 阻止 debugger 检测
    const preventDebuggerDetection = () => {
    const originalDebugger = Function.prototype.constructor
    Function.prototype.constructor = function () {
    const fn = originalDebugger.apply(this, arguments)
    if (arguments[0] && arguments[0].includes("debugger")) {
    return function () {}
    }
    return fn
    }
    }

    // 立即执行所有防护措施
    applyProtections()
    })()


    写了个油猴脚本解决
    Sdyhgc
        5
    Sdyhgc  
       44 天前 via Android
    16 进制修改 debugger 字符串 替换成其他的,一劳永逸
    关于   ·   帮助文档   ·   自助推广系统   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   2578 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 23ms · UTC 12:22 · PVG 20:22 · LAX 05:22 · JFK 08:22
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.