当前位置: 首页 > news >正文

chrome源码中WeakPtr 跨线程使用详解:原理、风险与最佳实践

        base::WeakPtr 在 Chromium 中 不能安全地跨线程使用。这是一个很关键的点,下面详细解释原因及正确用法。

🔍原理与使用

✅ 先说答案:

  • base::WeakPtr 本质上是**线程绑定(thread-affine)**的。
  • 不能在多个线程之间创建或使用它,特别是不能在一个线程创建 WeakPtr,然后在另一个线程中使用它。
  • 如果你非要跨线程判断对象是否有效,应使用更底层或线程安全的机制(例如 RefCountedThreadSafe + scoped_refptr)。

📌 官方说明

Chromium 的 base::WeakPtr 文档中有明确说明(可在源码注释中看到):

WeakPtr is not thread-safe. It must be used and dereferenced on the same thread (or Sequence) on which it was obtained from a WeakPtrFactory.

这段代码是chrome源码里面的WeakPtr实现:

WeakPtr<T> GetWeakPtr() { return WeakPtr<T>(weak_reference_owner_.GetRef(), reinterpret_cast<T*>(ptr_)); }

是否线程安全?答案是:不是线程安全的,除非你明确知道 weak_reference_owner_ 是线程安全的,而且 ptr_ 的生命周期也受线程安全保护(比如引用计数等)。


🔍 逐行分析:

1. weak_reference_owner_.GetRef()

这看起来是一个类似 base::internal::WeakReferenceOwner 或你自定义的 WeakReferenceOwner 类型。Chromium 中的 WeakReferenceOwner 默认是 线程绑定的(Thread-affine),只能在创建它的线程中使用(否则在 Debug 模式下会崩)。

✅ 所以:GetRef() 调用必须在原始线程中,不是线程安全的。


2. reinterpret_cast<T*>(ptr_)

这个操作没做线程安全保护的话,也是不安全的,尤其是如果 ptr_ 不是线程安全引用类型(如 scoped_refptr<T>)。


✅ 正确的 WeakPtr 创建姿势(来自 Chromium)

Chromium 推荐用 base::WeakPtrFactory<T>,并且只能在绑定线程上使用它:

class MyClass { public: base::WeakPtr<MyClass> GetWeakPtr() { return weak_factory_.GetWeakPtr(); // ✅ 安全:只能在同一个线程上用 } private: base::WeakPtrFactory<MyClass> weak_factory_{this}; };

  • WeakPtrFactory<T> 本身有一个 SequenceChecker,确保你只能在创建线程中用 GetWeakPtr()

  • 即使你多线程中有多个任务,都只能在同一个线程中访问和使用这个 WeakPtrFactory


❌ 错误使用示例

如果你把 GetWeakPtr() 这个函数暴露给了多个线程,并且多个线程中调用了它,比如这样:

thread1: my_object->GetWeakPtr(); thread2: my_object->GetWeakPtr();

即便 GetWeakPtr() 本身没有 crash,也会存在竞争风险或触发 DCHECK。


✅ 如果你真的想跨线程构建 WeakPtr...

你可以:

  1. 在初始化线程上调用 GetWeakPtr()

  2. 然后把这个 WeakPtr 通过 PostTask 发送到其他线程使用(读取、检查是否有效)。

  3. 不能在其他线程上继续调用 GetWeakPtr()


✅ 总结表格

行为是否线程安全建议
同一线程调用 GetWeakPtr()✅ 安全✅ 推荐
不同线程调用 GetWeakPtr()❌ 不安全❌ 禁止
一个线程创建 WeakPtr,另一个线程只用它⚠️ 有条件安全⚠️ 小心使用
自己实现 GetWeakPtr() 如你代码中那样❌ 不建议❌ 风险高


如你有多线程需求,建议考虑:

  • 使用 scoped_refptr<T>(跨线程完全安全);

  • 或将 WeakPtrFactory 迁移到后台线程中,并保证所有访问都在同一个线程;

  • 或用 CrossThreadWeakPtr 模式(Chromium中个别模块使用,较复杂)。

这段代码的作用是 在回调或异步任务执行时,确保 weak_self 所指向的对象仍然有效(未被销毁),以防止访问已经释放的对象从而导致 野指针或崩溃


🔍 代码解析:

if (!weak_self || weak_self.WasInvalidated()) { return; }

这是一个典型的 base::WeakPtr<T> 使用场景,常用于异步任务或跨线程操作中。

✅ 判断目的:
  1. !weak_self

    • 检查 weak_self 是否为空(即未绑定任何对象)。

    • 这相当于检查 WeakPtr 是否有效初始化。

  2. weak_self.WasInvalidated()

    • 判断该 WeakPtr 是否已经 失效(即指向的原始对象已经销毁)

    • 即使指针本身不为空,如果 T 已被析构,也不能再访问。


🛡️ 为什么这么写?

异步/延迟任务常见问题:

举个例子:

// 假设这是某个类中的成员函数 PostTask(FROM_HERE, base::BindOnce(&MyClass::DoWork, weak_factory_.GetWeakPtr()));

  • 如果 MyClass 在任务执行前被析构,那么 DoWork() 中的 this 就是悬空指针。

  • WeakPtr 就是防止“任务晚于对象生命周期”这种典型 bug。

  • 所以在 DoWork() 内部首先要判断 weak_self 是否有效。


🧠 小结:

if (!weak_self || weak_self.WasInvalidated()) { return; // 安全退出,防止访问已销毁对象 }

这个判断非常关键,用于:

  • 保证异步回调安全性;

  • 防止 use-after-free;

  • 是 C++ 客户端编程(尤其是 Chromium 项目)中的推荐模式。

🛡️ 工程案例:

cookie_manager->SetCanonicalCookie(*canonical_cookie, cookie_info.url, options, base::BindOnce(&InProcessImporterBridge::SetCanonicalCookieCallback, weak_factory_.GetWeakPtr())); 这个代码在ui线程,SetCanonicalCookieCallback这个回调要是在其他线程调用会有什么问题\

在 Chromium 的线程模型中,SetCanonicalCookieCallback 回调若在其他线程(非 UI 线程)调用可能导致严重问题。以下是详细分析:


问题根源

  1. 线程安全性违例

    • InProcessImporterBridge 类及其成员变量通常非线程安全,假设仅在 UI 线程访问。

    • 若回调在其他线程(如 IO 线程)执行,直接访问成员变量会导致数据竞争(Data Race)。

  2. weak_factory_.GetWeakPtr() 的线程安全性

    • WeakPtrFactory 本身线程安全(GetWeakPtr() 可跨线程调用),但不保证回调执行时的对象存活

    • 若 UI 线程已销毁 InProcessImporterBridge,其他线程调用回调会访问无效内存。

  3. UI 线程依赖的操作

    • 回调中若操作 UI 相关资源(如更新界面、访问 Profile),必须在 UI 线程执行。


具体风险场景

// 错误示例:回调可能在其他线程执行
cookie_manager->SetCanonicalCookie(*canonical_cookie, cookie_info.url, options,base::BindOnce(&InProcessImporterBridge::SetCanonicalCookieCallback,weak_factory_.GetWeakPtr()));  // 若回调在非UI线程运行
  • 崩溃风险

    • 其他线程调用时,若 InProcessImporterBridge 已销毁,weak_factory_ 返回 nullptr,回调被丢弃(安全)。

    • 但若对象存活,回调直接访问成员变量(如 std::vectorstd::map)会触发未定义行为。

  • 逻辑错误

    • 非 UI 线程修改 UI 相关状态(如界面标志位),导致界面显示异常。


解决方案

1. 强制回调到 UI 线程

使用 base::BindPostTask 将回调派发到 UI 线程的任务队列:

cookie_manager->SetCanonicalCookie(*canonical_cookie,cookie_info.url,options,base::BindOnce(base::BindPostTask(base::SequencedTaskRunner::GetCurrentDefault(),  // 获取UI线程TaskRunnerbase::BindOnce(&InProcessImporterBridge::SetCanonicalCookieCallback,weak_factory_.GetWeakPtr())))
);
2. 回调中显式检查线程

在回调内部验证是否运行在 UI 线程:

void InProcessImporterBridge::SetCanonicalCookieCallback(...) {DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);// 安全操作成员变量
}
3. 使用 ThreadSafeRefCounted(若适合)

若 InProcessImporterBridge 需跨线程共享,改为继承 base::RefCountedThreadSafe,但需确保所有成员访问均线程安全(通常不推荐用于 UI 相关类)。


为什么 WeakPtr 不能完全解决问题?

  • WeakPtr 仅保证对象销毁后不执行回调,但若对象存活:

    • 不保证回调线程安全。

    • 不自动切换线程上下文。

  • 必须配合 BindPostTask 或线程检查使用。


Chromium 的线程模型要求

  1. UI 线程专属对象

    • 如 ProfileWebContents、UI 控件等,必须通过 UI 线程访问。

  2. 线程跳转代价

    • 跨线程任务提交(如 PostTask)有开销,但比崩溃/数据竞争更可接受。


正确代码示例

// 方法1:使用 BindPostTask(推荐)
cookie_manager->SetCanonicalCookie(*canonical_cookie,cookie_info.url,options,base::BindPostTask(base::SequencedTaskRunner::GetCurrentDefault(),base::BindOnce(&InProcessImporterBridge::SetCanonicalCookieCallback,weak_factory_.GetWeakPtr()))
);// 方法2:回调内手动跳转(备用)
class InProcessImporterBridge {public:void SetCanonicalCookieCallback(...) {if (!base::SequencedTaskRunner::GetCurrentDefault()->RunsTasksInCurrentSequence()) {base::SequencedTaskRunner::GetCurrentDefault()->PostTask(FROM_HERE,base::BindOnce(&InProcessImporterBridge::SetCanonicalCookieCallback,weak_factory_.GetWeakPtr(), ...));return;}DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);// 实际处理逻辑}
};

总结

问题风险等级解决方案
非UI线程访问成员变量高(崩溃)用 BindPostTask 跳转回 UI 线程
WeakPtr 对象已销毁低(安全)自动跳过回调
未同步的 UI 状态修改中(逻辑错误)添加 DCHECK_CALLED_ON_VALID_SEQUENCE

最佳实践

  • 默认假设回调可能在任何线程执行。

  • 对 UI 相关对象,始终通过 BindPostTask 或 PostTask 确保线程安全。

下面是个跨线程场景使用的一个完整例子

class QQBrowserImporter : public Importer {
public:QQBrowserImporter();explicit QQBrowserImporter(bool first_run);void StartImport(const importer::SourceProfile& source_profile,uint16_t items,ImporterBridge* bridge) override;static void ImportCookies(const std::wstring& localPath, base::WeakPtr<QQBrowserImporter> client);protected:friend class base::RefCountedThreadSafe<QQBrowserImporter>;~QQBrowserImporter() override;private:scoped_refptr<base::SequencedTaskRunner> GetdbTaskRunner();bool ScheduleTask(const base::RepeatingClosure& task);scoped_refptr<base::SequencedTaskRunner> db_thread_runner_;base::WeakPtrFactory<QQBrowserImporter> weak_factory_;DISALLOW_COPY_AND_ASSIGN(QQBrowserImporter);
};QQBrowserImporter::QQBrowserImporter(bool first_run) : first_run_(first_run), weak_factory_(this) {}QQBrowserImporter::~QQBrowserImporter(void){}void QQBrowserImporter::StartImport(const importer::SourceProfile& source_profile,uint16_t items,ImporterBridge* bridge) {bridge_ = bridge;bridge_->NotifyStarted();std::wstring source_path = source_profile.source_path.value();if ((items & importer::COOKIES) && !cancelled()) {ScheduleTask(base::BindRepeating(&QQBrowserImporter::ImportCookies, source_path, weak_factory_.GetWeakPtr()));}
}// TODO extract to base class
void QQBrowserImporter::ImportCookies(const std::wstring& localPath, base::WeakPtr<QQBrowserImporter> client) {std::wstring local_data_path = localPath;std::string desc;local_data_path += L"\\Default\\Network\\Cookies";base::FilePath qq_login_path(local_data_path);sql::Database db;if (!db.Open(qq_login_path)) {content::GetUIThreadTaskRunner({})->PostTask(FROM_HERE,base::BindOnce([](base::WeakPtr<QQBrowserImporter> weak_self) {if (!weak_self || weak_self.WasInvalidated()) {return;}if (weak_self->bridge_) {base::Value::Dict import_res;import_res.Set("successcount", 0);import_res.Set("faileddesc", "Failed to open QQ login data DB");std::string value_str;base::JSONWriter::Write(import_res, &value_str);weak_self->bridge_->NotifyEnded(false, importer::TYPE_COOKIES_QQ, value_str);}},client));return;}const char* kQuery = "SELECT host_key, encrypted_value, name, path, creation_utc, expires_utc, last_access_utc, is_secure, is_httponly, samesite, priority FROM cookies";sql::Statement stmt(db.GetUniqueStatement(kQuery));if (!stmt.is_valid()) {content::GetUIThreadTaskRunner({})->PostTask(FROM_HERE,base::BindOnce([](base::WeakPtr<QQBrowserImporter> weak_self) {if (!weak_self || weak_self.WasInvalidated()) {return;}if (weak_self->bridge_) {base::Value::Dict import_res;import_res.Set("successcount", 0);import_res.Set("faileddesc", "Failed to prepare login query statement");std::string value_str;base::JSONWriter::Write(import_res, &value_str);weak_self->bridge_->NotifyEnded(false, importer::TYPE_COOKIES_QQ, value_str);}},client));return;}std::string localPathUtf8 = base::WideToUTF8(localPath);auto aesKey = chrome::GetDecryptedKey(localPathUtf8);std::vector<importer::CookiesInfo> cookies_info;while (stmt.Step()) {std::string host_key  = stmt.ColumnString(0);std::string name = stmt.ColumnString(2);std::string path = stmt.ColumnString(3);int64_t creation_utc = stmt.ColumnInt64(4);int64_t expires_utc = stmt.ColumnInt64(5);int64_t last_access_utc = stmt.ColumnInt64(6);// compatible with old browser kernels, new kernels will add SameSite=None by defaultint is_http_only = true; // stmt.ColumnInt(8);int is_secure = true; // stmt.ColumnInt(7);int samesite_val = stmt.ColumnInt(9);int priority_val = stmt.ColumnInt(10);base::span<const uint8_t> blob_span = stmt.ColumnBlob(1);std::vector<unsigned char> encrypted(blob_span.begin(), blob_span.end());std::string value;if (encrypted.size() > 15 && encrypted[0] == 'v' && encrypted[1] == '1') {std::vector<unsigned char> nonce(encrypted.begin() + 3, encrypted.begin() + 15);std::vector<unsigned char> ciphertext(encrypted.begin() + 15, encrypted.end() - 16);std::vector<unsigned char> tag(encrypted.end() - 16, encrypted.end());auto decrypted = chrome::AESGCMDecrypt(aesKey, nonce, ciphertext, tag);value = std::string(decrypted.begin(), decrypted.end());}else {continue;}std::string cookie_str = FormatCookieString(name, value, host_key, path, expires_utc, is_secure == 0 ? false : true, is_http_only == 0 ? false : true, samesite_val);std::string scheme = is_secure ? "https://" : "http://";GURL url(scheme + ConvertHostKeyToDomain(host_key) + path);cookies_info.emplace_back(std::move(cookie_str), std::move(url), is_http_only);}if (cookies_info.empty()) {content::GetUIThreadTaskRunner({})->PostTask(FROM_HERE,base::BindOnce([](base::WeakPtr<QQBrowserImporter> weak_self) {if (!weak_self || weak_self.WasInvalidated()) {return;}if (weak_self->bridge_) {base::Value::Dict import_res;import_res.Set("successcount", 0);import_res.Set("faileddesc", se_import_user_info::kEmptyData);std::string value_str;base::JSONWriter::Write(import_res, &value_str);weak_self->bridge_->NotifyEnded(false, importer::TYPE_COOKIES_QQ, value_str);}},client));return;}auto task = base::BindOnce([](base::WeakPtr<QQBrowserImporter> weak_self, std::vector<importer::CookiesInfo> cookies_info) {if (!weak_self || weak_self.WasInvalidated()) {return;}if (weak_self->bridge_) {weak_self->bridge_->SetCookie(cookies_info, se_import_user_info::BrowserType::kQQ, importer::TYPE_COOKIES_QQ);}}, client, cookies_info);content::GetUIThreadTaskRunner({})->PostTask(FROM_HERE, std::move(task));
}scoped_refptr<base::SequencedTaskRunner> QQBrowserImporter::GetdbTaskRunner() {if (!db_thread_runner_) {db_thread_runner_ = base::ThreadPool::CreateSingleThreadTaskRunner({base::TaskPriority::HIGHEST, base::MayBlock()});}return db_thread_runner_;
}bool QQBrowserImporter::ScheduleTask(const base::RepeatingClosure& task) {scoped_refptr<base::SequencedTaskRunner> task_runner(GetdbTaskRunner());if (task_runner.get())return task_runner->PostTask(FROM_HERE, task);return false;
}

相关文章:

chrome源码中WeakPtr 跨线程使用详解:原理、风险与最佳实践

base::WeakPtr 在 Chromium 中 不能安全地跨线程使用。这是一个很关键的点&#xff0c;下面详细解释原因及正确用法。 &#x1f50d;原理与使用 ✅ 先说答案&#xff1a; base::WeakPtr 本质上是**线程绑定&#xff08;thread-affine&#xff09;**的。不能在多个线程之间创建…...

vue2使用three.js实现一个旋转球体

vue页面中 <div ref"container"></div>data声明 scene: null, camera: null, renderer: null, controls: null, rotationType: sphere, rotationTimer: null,backgroundImageUrl: https://mini-app-img-1251768088.cos.…...

社交平台推出IP关联机制:增强用户体验与网络安全的新举措

社交平台为我们提供与亲朋好友保持联系、分享生活点滴的便捷渠道&#xff0c;还成为了信息传播、观点交流的重要平台。然而&#xff0c;随着社交平台的普及&#xff0c;网络空间中的虚假信息、恶意行为等问题也日益凸显。为了应对这些挑战&#xff0c;许多社交平台相继推出IP关…...

sherpa-ncnn:音频处理跟不上采集速度 -- 语音转文本大模型

目录 1. 问题报错2. 解决方法 1. 问题报错 报错&#xff1a; An overrun occurred, which means the RTF of the current model on your board is larger than 1. You can use ./bin/sherpa-ncnn to verify that. Please select a smaller model whose RTF is less than 1 fo…...

【android bluetooth 协议分析 01】【HCI 层介绍 8】【ReadLocalVersionInformation命令介绍】

1. HCI_Read_Local_Version_Information 命令介绍 1. 功能&#xff08;Description&#xff09; HCI_Read_Local_Version_Information 命令用于读取本地 Bluetooth Controller 的版本信息&#xff0c;包括 HCI 和 LMP 层的版本&#xff0c;以及厂商 ID 和子版本号。 这类信息用…...

android13以太网静态ip不断断开连上问题

总纲 android13 rom 开发总纲说明 文章目录 1.前言2.log记录3.问题分析4.代码修改5.彩蛋1.前言 android13以太网静态ip不断断开连上,具体情况为保存静态以太网成功后,可以看到以太网链接成功的图标,但是几秒后会消失,出现断网,几秒后又出现,反复出现和消失。 2.log记录…...

UA 编译和建模入门教程(zhanzhi学习笔记)

一、使用SIOME免费工具建模 从西门子官网下载软件SIOS&#xff0c;需要注册登录&#xff0c;下载安装版就行。下载后直接安装就可以用了&#xff0c;如图&#xff1a; 安装完成后打开&#xff0c;开始建模&#xff0c;如图左上角有新建模型的按钮。 新建了新工程后&#xff0c…...

系统架构设计-案例分析总结

系统架构设计-案例分析总结 2024年下半年系统架构设计师案例第1题 2022年下半年系统架构设计师案例第1题第2题 2021年下半年系统架构设计师案例第1题第2题 2024年下半年系统架构设计师案例 题&#xff1a;效用树可用性中ping/echo策略和心跳策略比较 第1题 阅读以下关于面向质…...

【QT】一个界面中嵌入其它界面(三)

在 Qt 中&#xff0c;通过 UI 设计 或 代码布局 实现界面 A 中同时显示界面 B 和 C&#xff0c;并精确指定它们的位置&#xff0c;可以通过以下两种方式实现。以下是详细步骤和完整代码&#xff1a; 方法 0&#xff1a;使用 Qt Designer 可视化布局 通过 Qt Designer 拖拽控件…...

实战教程:影刀RPA采集闲鱼商品并分享钉钉

1.实战目标 采集字段&#xff1a; 采集时间商品ID商品标题标价商品链接 采集的第一个品 可通过钉钉分享给好友 也可以通过钉钉群通知指令&#xff0c;发送到指定群 2.实战代码 2.1 主体代码 2.2 采集初始化 先初始化环境 这一步骤主要是连接手机&#xff0c;能使用影刀RPA操…...

多模态大语言模型arxiv论文略读(八十二)

Emotion-LLaMA: Multimodal Emotion Recognition and Reasoning with Instruction Tuning ➡️ 论文标题&#xff1a;Emotion-LLaMA: Multimodal Emotion Recognition and Reasoning with Instruction Tuning ➡️ 论文作者&#xff1a;Zebang Cheng, Zhi-Qi Cheng, Jun-Yan H…...

常见排序算法整理(Java实现)

1.冒泡排序&#xff08;Bubble Sort&#xff09; 原理 重复遍历数组&#xff0c;比较相邻元素&#xff0c;若顺序错误则交换。每趟将最大元素"冒泡"到末尾。 每次遍历保证了最大元素被放在最后&#xff0c;所以内层循环不需要遍历到最后的位置。 代码实现 public …...

c++字符串常用语法特性查询示例文档(二)

在 C中&#xff0c;除了std::string和std::string_view&#xff0c;还有其他一些与字符串相关的类型&#xff0c;它们各自针对不同的场景进行了优化。以下是一些常见的字符串类型及其使用方式和适用场景的汇总。 紧接上篇 c字符串常用语法特性查询示例文档&#xff08;一&#…...

10.14 Function Calling与Tool Calling终极解析:大模型工具调用的5大核心差异与实战优化

Function Calling vs Tool Calling:大模型工具调用机制深度解析 关键词:Function Calling 原理, Tool Calling 实现, @tool 装饰器, ToolMessage 机制, 工具调用优化 1. 核心概念对比分析 #mermaid-svg-uDxSPB1CoQrHDxrT {font-family:"trebuchet ms",verdana,ari…...

opencascade如何保存选中的面到本地

环境&#xff1a;occ 7.6 需求场景&#xff1a;用户点击了一个TopoDS_Shape&#xff0c;还选中了其中一个面&#xff0c;这时候他点了保存。用户下次打开模型文件时&#xff0c;我们的viewer窗口要恢复上次的选中状态。 核心问题&#xff1a;如何把用户选中的面保存&#xff0c…...

CSS 单位详解:px、rem、em、vw/vh 的区别与使用场景

CSS 单位详解&#xff1a;px、rem、em、vw/vh 的区别与使用场景 在 CSS 中&#xff0c;各种单位有不同的特性和适用场景&#xff0c;理解它们的区别对实现响应式布局至关重要。 1. 绝对单位 px 特点&#xff1a; 像素&#xff08;Pixel&#xff09;是绝对长度单位1px 对应屏…...

YOLO模型predict(预测/推理)的参数设置

上一章描述了预测初体验,基本可以使用现有的yolo模型进行预测/推理。本次我们了解一下这个过程中的参数的作用。 1.参数示例 conf=0.68 :表示模型识别这个东西是车的概率为68% 。一般默认的情况下,概率小于25%的就不显示了。 1)调整一下python的代码的参数如下,可以预测图…...

MATLAB中NLP工具箱支持聚类算法

文章目录 前言一、层次聚类&#xff08;Hierarchical Clustering&#xff09;二、DBSCAN&#xff08;基于密度的空间聚类&#xff09;三、高斯混合模型&#xff08;GMM&#xff09;四、谱聚类&#xff08;Spectral Clustering&#xff09;五、模糊 C 均值&#xff08;Fuzzy C-M…...

甘特图工具怎么选?免费/付费项目管理工具对比测评(2025最新版)

2025年甘特图工具的全面指南 在项目管理领域&#xff0c;甘特图作为最直观的任务规划和进度追踪工具&#xff0c;已成为团队协作和项目执行的核心手段。随着数字化技术的快速发展&#xff0c;2025年的甘特图工具市场呈现出前所未有的多元化和智能化趋势。从开源软件到云端协作…...

Google设置app-ads.txt

问题&#xff1a; 应用上架后admob后台显示应用广告投放量受限&#xff0c;需要设置app-ads.txt才行。 如何解决&#xff1a; 官方教程: 看了下感觉不难&#xff0c;创建一个txt&#xff0c;将第二条的代码复制进行就得到app-ads.txt了。 然后就是要把这个txt放到哪才可以…...

Swift 二分查找实战:精准定位第一个“Bug版本”(LeetCode 278)

文章目录 摘要描述示例 题解答案&#xff08;Swift&#xff09;题解代码分析示例测试及结果输出结果&#xff1a; 时间复杂度分析空间复杂度分析总结 摘要 在版本迭代频繁的项目开发中&#xff0c;定位引入 bug 的第一个版本是一项高频任务。LeetCode 第278题“第一个错误的版…...

《AI革命重塑未来五年:医疗诊断精准度翻倍、自动驾驶事故锐减90%,全球科技版图加速变革》

1. 显著突破领域&#xff1a;AI 引发医疗与自动驾驶的范式变革 医疗领域的突破&#xff1a; AI正深刻改变医学研发和临床诊疗模式。在新药研发现代生物学中&#xff0c;DeepMind公司推出的 AlphaFold AI 模型在蛋白质折叠预测上取得了重大突破&#xff0c;被视为解决了困扰科学…...

【盈达科技】AICC™系统:重新定义生成式AI时代的内容竞争力

盈达科技AICC™系统&#xff1a;重新定义生成式AI时代的内容竞争力 ——全球首款AI免疫化内容中台的技术革命与商业实践 一、技术破局&#xff1a;AICC™系统如何重构AI内容生态 1. 技术架构&#xff1a;四大引擎构建闭环护城河 盈达科技AICC™&#xff08;AI-Immunized Con…...

芯驰科技与安波福联合举办技术研讨会,深化智能汽车领域合作交流

5月15日&#xff0c;芯驰科技与全球移动出行技术解决方案供应商安波福&#xff08;Aptiv&#xff09;在上海联合举办以“芯智融合&#xff0c;共赢未来”为主题的技术研讨会。会上&#xff0c;双方聚焦智能座舱与智能车控的发展趋势&#xff0c;展开深入交流与探讨&#xff0c;…...

开发 前端搭建npm v11.4.0 is known not to run on Node.js v14.18.1.

错误nodejs 和npm 版本不一致 ERROR: npm v11.4.0 is known not to run on Node.js v14.18.1. This version of npm supports the following node versions: ^20.17.0 || >22.9.0. You can find the latest version at https://nodejs.org/. ERROR: D:\softTool\node-v14…...

关于systemverilog中在task中使用force语句的注意事项

先看下面的代码 module top(data);logic clk; inout data; logic temp; logic sampale_data; logic [7:0] data_rec;task send_data(input [7:0] da);begin(posedge clk);#1;force datada[7];$display(data);(posedge clk);#1;force datada[6]; $display(data); (posed…...

国产 iPaaS 与国外 iPaaS 产品相比如何?以谷云科技为例

iPaaS&#xff08;Integration Platform as a Service&#xff09;作为企业集成的关键技术&#xff0c;受到了广泛关注。国产 iPaaS 产品与国外 iPaaS 产品存在诸多差异&#xff0c;以下将从多个方面进行分析探讨。 一、技术架构与创新 国外 iPaaS 产品往往技术架构成熟且先进…...

低功耗:XILINX FPGA如何优化功耗?

优化Xilinx FPGA及其外围电路的功耗需要从硬件设计、软件配置和系统级优化三个层面综合考虑。以下是具体的优化策略&#xff0c;涵盖硬件和软件方面&#xff1a; 一、硬件层面的功耗优化 选择低功耗FPGA型号 选择Xilinx低功耗系列芯片&#xff0c;如7系列中的Artix-7&#xff…...

从纸质契约到智能契约:AI如何改写信任规则与商业效率?​——从智能合约到监管科技,一场颠覆传统商业逻辑的技术革命

一、传统合同的“低效困境”&#xff1a;耗时、昂贵、风险失控 近年来&#xff0c;全球商业环境加速向数字化转型&#xff0c;但合同管理却成为企业效率的“阿喀琉斯之踵”。据国际商会&#xff08;International Chamber of Commerce&#xff09;数据显示&#xff0c;全球企业…...

在金融发展领域,嵌入式主板有什么优点?

在金融发展领域&#xff0c;嵌入式主板能够有力推动金融行业的智能化与高效化进程。主板的强大计算能力可以保障业务高效运行。例如在银行的高频交易场景下&#xff0c;其强大计算能力可确保系统在高负荷下依然保持流畅稳定&#xff0c;快速响应用户需求&#xff0c;大大提升金…...

打卡Day30

导入官方库的三种手段 方法一&#xff1a;直接导入整个模块 import math print(math.sqrt(16)) # 输出: 4.0方法二&#xff1a;从模块中导入特定函数或类 from datetime import datetime now datetime.now() print(now) # 输出当前日期和时间方法三&#xff1a;使用别名简…...

AI量化交易是什么?它是如何重塑金融世界的?

第一章&#xff1a;证券交易的进化之路 1.1 从喊价到代码&#xff1a;交易方式的革命性转变 在电子交易普及之前&#xff0c;证券交易依赖于交易所内的公开喊价系统。交易员通过手势、喊话甚至身体语言传递买卖信息&#xff0c;这种模式虽然直观&#xff0c;但效率低下且容易…...

AIGC与数字金融:人工智能金融创新的新纪元

AIGC与数字金融&#xff1a;人工智能金融创新的新纪元 引言 人工智能生成内容&#xff08;AIGC&#xff09;在数字金融领域发挥着关键作用&#xff0c;从金融内容生成到智能风控&#xff0c;从个性化服务到投资决策&#xff0c;AIGC正在重塑金融的方式和效果。本文将深入探讨A…...

芯片生态链深度解析(四):芯片制造篇——纳米尺度上的精密艺术

开篇&#xff1a;芯片制造——现代工业的"皇冠明珠" 在芯片生态链的版图中&#xff0c;芯片制造是连接设计与封测的核心枢纽&#xff0c;堪称现代工业的“皇冠明珠”。如果说芯片设计是人类对微观世界的构想&#xff0c;那么制造便是将这种构想转化为现实的终极工程…...

黄金批次在流程和离散行业的概念解析

流程行业 概念 流程行业中: “黄金批次”:通常指生产过程中质量最优、性能最稳定、符合甚至超越所有关键指标的特定批次产品。这类批次在流程行业中具有标杆意义,常用于质量控制、工艺优化和客户交付。 核心特征 在流程行业中,“黄金批次”的核心特征包括: 1、质量一…...

Transformer实战——循环神经网络详解

Transformer实战——循环神经网络详解 0. 前言1. 基本循环神经网络单元1.1 循环神经网络工作原理1.2 时间反向传播1.3 梯度消失和梯度爆炸问题 2. RNN 单元变体2.1 长短期记忆2.2 门控循环单元2.3 Peephole LSTM 3. RNN 变体3.1 双向 RNN3.2 状态 RNN 4. RNN 拓扑结构小结 0. 前…...

基于Qt的app开发第九天

写在前面 笔者的课设截止时间已经越来越近了&#xff0c;还有不少地方的功能没有完成&#xff0c;所以重构一事还是放到做完整个项目、学完设计模式再考虑。目前就是继续往屎山堆屎。 需求分析 笔者的学长要做多线程&#xff0c;传数据的时候涉及到互斥锁之类的内容&#xff0…...

Baklib内容中台驱动资源管理创新

内容中台驱动智能整合 现代企业数字化进程中&#xff0c;内容中台通过结构化数据治理与智能算法协同&#xff0c;有效解决跨系统内容孤岛问题。以Baklib为例&#xff0c;其核心功能通过多语言支持与API接口集成能力&#xff0c;实现营销素材、产品文档等异构资源的统一索引与动…...

项目记录:「五秒反应挑战」小游戏的开发全过程

我正在参加CodeBuddy「首席试玩官」内容创作大赛&#xff0c;本文所使用的 CodeBuddy 免费下载链接&#xff1a;腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 灵感来源与目标设定 最近我突然有个小想法&#xff1a;想做一个简洁但富有科技感的小型游戏&#xff0c;最好能…...

Git本地使用小Tips

要将本地仓库 d:\test 的更新推送到另一个本地仓库 e:\test&#xff0c;可以使用 Git 的远程仓库功能。以下是具体步骤&#xff1a; ​​在 e:\test 中添加 d:\test 作为远程仓库​​ 在 e:\test 目录中打开 Git Bash 或命令行&#xff0c;执行以下命令&#xff1a; git remo…...

【AWS入门】AWS身份验证和访问管理(IAM)

【AWS入门】AWS身份验证和访问管理&#xff08;IAM&#xff09; [AWS Essentials] AWS Identity and Access Management (IAM) By JacksonML 众所周知&#xff0c;AWS亚马逊云科技位列全球云计算服务第一位&#xff0c;并且持续为广大客户提供安全、稳定的各类云产品和服务。…...

【NLP】36. 从指令微调到人类偏好:构建更有用的大语言模型

从指令微调到人类偏好&#xff1a;构建更有用的大语言模型 大语言模型&#xff08;LLMs&#xff09;已经成为现代自然语言处理系统的核心&#xff0c;但单纯依赖传统语言建模目标&#xff0c;往往难以满足实际应用的“人类意图”。从 Instruction Tuning&#xff08;指令微调&…...

蓝桥杯1447 砝码称重

问题描述 你有一架天平和 N 个砝码&#xff0c;这 N 个砝码重量依次是 W1,W2,⋅⋅⋅,WN​。 请你计算一共可以称出多少种不同的重量&#xff1f; 注意砝码可以放在天平两边。 输入格式 输入的第一行包含一个整数 N。 第二行包含 N 个整数&#xff1a;W1,W2,W3,⋅⋅⋅,WN​…...

每日c/c++题 备战蓝桥杯(洛谷P4715 【深基16.例1】淘汰赛 题解)

洛谷P4715 【深基16.例1】淘汰赛 题解 题目大意 有 (2^n) 名选手进行淘汰赛&#xff0c;每场比赛两人对决&#xff0c;能力值高者胜出&#xff1b;若能力值相同&#xff0c;则编号较小者胜出。最终决出冠军&#xff0c;要求输出亚军的编号。 解题思路 关键观察&#xff1a;…...

基于深度学习的电力负荷预测研究

一、深度学习模型框架 在当今数字化时代&#xff0c;基于深度学习的电力负荷预测研究正成为保障电力系统稳定、高效运行的关键领域。其模型构建是一个复杂而精妙的过程&#xff0c;涉及多学科知识与前沿技术的融合应用。首先&#xff0c;要明确电力负荷预测的目标&#xff0c;…...

没有 Mac,我如何用 Appuploader 完成 iOS App 上架

没有 Mac&#xff0c;我如何用 Appuploader 完成 iOS App 上架&#xff5c;一个跨平台开发者的实战笔记 在做移动开发这些年里&#xff0c;唯一让我频繁想砸电脑的时刻&#xff0c;大概就是每次要把 iOS App 上传到 App Store。 作为一个主要在 Windows 和 Linux 开发的程序员…...

卓力达手撕垫片:精密制造的创新解决方案与多领域应用

目录 引言 一、手撕垫片的技术特性与核心功能 二、手撕垫片的多领域应用 三、南通卓力达手撕垫片的核心优势 四、未来技术展望 结语 引言 手撕垫片作为一种创新型精密金属部件&#xff0c;凭借其“可分层剥离、快速安装”的特性&#xff0c;在工业制造、电子通信、航空航天等…...

[[春秋云境] Privilege仿真场景

文章目录 靶标介绍&#xff1a;知识点卷影拷贝(VSS) 外网任意文件读取Jenkins管理员后台rdp远程登录Gitlab apiToken 内网搭建代理 Oracle RCESeRestorePrivilege提权mimikatzspn卷影拷贝提取SAM 参考文章 靶标介绍&#xff1a; 在这个靶场中&#xff0c;您将扮演一名资深黑客…...

【工具推荐】--Git详解

本文讲诉&#xff0c;git命令环境的安装和git命令的介绍 Git 是一个非常流行的分布式版本控制系统&#xff0c;它帮助开发者管理和跟踪项目中的代码变化。通俗地说&#xff0c;可以认为 Git 就像是一个代码的时间机器&#xff0c;它记录了项目从开始到结束的每一次代码变动。 …...

在linux里上传本地项目到github中

首先先安装git&#xff0c;安装完git后&#xff0c;输入如下操作指令&#xff1a; 输入自己的用户名和邮箱&#xff08;为注册GITHUB账号时的用户名和邮箱&#xff09;&#xff1a; git config --global user.name "111"git config --global user.email "121…...