一、系统概述

本程序使用VC++ MFC对话框实现PC与单片机的串口通信,支持数据发送、接收、显示和存储功能,适用于工业控制、数据采集等场景。

二、实现原理

2.1 通信架构

graph LR
    A[PC端VC++程序] --串口--> B[51单片机]
    B --串口--> A
    A --> C[用户界面]
    C --> A

2.2 数据流

  1. PC发送控制命令到单片机
  2. 单片机执行命令并返回数据
  3. PC接收数据并显示在界面
  4. 用户可保存数据或发送新命令

三、完整实现代码

3.1 资源文件定义 (Resource.h)

#define IDD_SERIAL_COMM_DIALOG       102
#define IDC_MSCOMM1                  1001
#define IDC_COMBO_COM_PORT           1002
#define IDC_COMBO_BAUD_RATE          1003
#define IDC_BUTTON_OPEN              1004
#define IDC_BUTTON_CLOSE             1005
#define IDC_EDIT_SEND                1006
#define IDC_BUTTON_SEND              1007
#define IDC_EDIT_RECEIVE             1008
#define IDC_CHECK_HEX_SEND           1009
#define IDC_CHECK_HEX_RECEIVE         1010
#define IDC_BUTTON_CLEAR             1011
#define IDC_BUTTON_SAVE              1012
#define IDC_STATIC_STATUS            1013
#define IDC_COMBO_DATA_BITS          1014
#define IDC_COMBO_PARITY             1015
#define IDC_COMBO_STOP_BITS          1016

3.2 对话框类头文件 (SerialCommDlg.h)

#if !defined(AFX_SERIALCOMMDLG_H__)
#define AFX_SERIALCOMMDLG_H__

#include <afxwin.h>
#include <afxdisp.h>
#include <afxcmn.h>
#include <mscomm.h>

class CSerialCommDlg : public CDialog {
public:
    CSerialCommDlg(CWnd* pParent = NULL);
    enum { IDD = IDD_SERIAL_COMM_DIALOG };
    
protected:
    virtual void DoDataExchange(CDataExchange* pDX);
    virtual BOOL OnInitDialog();
    
    // 控件变量
    CMSComm m_mscom;
    CComboBox m_comboComPort;
    CComboBox m_comboBaudRate;
    CComboBox m_comboDataBits;
    CComboBox m_comboParity;
    CComboBox m_comboStopBits;
    CEdit m_editSend;
    CEdit m_editReceive;
    CButton m_checkHexSend;
    CButton m_checkHexReceive;
    CStatic m_staticStatus;
    
    // 消息处理
    afx_msg void OnButtonOpen();
    afx_msg void OnButtonClose();
    afx_msg void OnButtonSend();
    afx_msg void OnButtonClear();
    afx_msg void OnButtonSave();
    afx_msg void OnCommMscomm();
    afx_msg void OnClose();
    
    // 辅助函数
    void UpdateStatus(LPCTSTR msg);
    void ProcessReceivedData();
    CString FormatHexData(const BYTE* data, int len);
    void AddToReceiveBox(LPCTSTR str);
    
    DECLARE_MESSAGE_MAP()
    DECLARE_EVENTSINK_MAP()
};

#endif

3.3 对话框实现文件 (SerialCommDlg.cpp)

#include "stdafx.h"
#include "SerialComm.h"
#include "SerialCommDlg.h"
#include <afxmt.h>
#include <fstream>

// 事件映射
BEGIN_EVENTSINK_MAP(CSerialCommDlg, CDialog)
    ON_EVENT(CSerialCommDlg, IDC_MSCOMM1, 1, OnCommMscomm, VTS_NONE)
END_EVENTSINK_MAP()

// 消息映射
BEGIN_MESSAGE_MAP(CSerialCommDlg, CDialog)
    ON_BN_CLICKED(IDC_BUTTON_OPEN, OnButtonOpen)
    ON_BN_CLICKED(IDC_BUTTON_CLOSE, OnButtonClose)
    ON_BN_CLICKED(IDC_BUTTON_SEND, OnButtonSend)
    ON_BN_CLICKED(IDC_BUTTON_CLEAR, OnButtonClear)
    ON_BN_CLICKED(IDC_BUTTON_SAVE, OnButtonSave)
    ON_WM_CLOSE()
END_MESSAGE_MAP()

// 构造函数
CSerialCommDlg::CSerialCommDlg(CWnd* pParent) : CDialog(CSerialCommDlg::IDD, pParent) {
    m_mscom.m_bAutoSize = TRUE;
}

// 数据交换
void CSerialCommDlg::DoDataExchange(CDataExchange* pDX) {
    CDialog::DoDataExchange(pDX);
    DDX_Control(pDX, IDC_MSCOMM1, m_mscom);
    DDX_Control(pDX, IDC_COMBO_COM_PORT, m_comboComPort);
    DDX_Control(pDX, IDC_COMBO_BAUD_RATE, m_comboBaudRate);
    DDX_Control(pDX, IDC_COMBO_DATA_BITS, m_comboDataBits);
    DDX_Control(pDX, IDC_COMBO_PARITY, m_comboParity);
    DDX_Control(pDX, IDC_COMBO_STOP_BITS, m_comboStopBits);
    DDX_Control(pDX, IDC_EDIT_SEND, m_editSend);
    DDX_Control(pDX, IDC_EDIT_RECEIVE, m_editReceive);
    DDX_Control(pDX, IDC_CHECK_HEX_SEND, m_checkHexSend);
    DDX_Control(pDX, IDC_CHECK_HEX_RECEIVE, m_checkHexReceive);
    DDX_Control(pDX, IDC_STATIC_STATUS, m_staticStatus);
}

// 初始化对话框
BOOL CSerialCommDlg::OnInitDialog() {
    CDialog::OnInitDialog();
    
    // 初始化串口列表
    for (int i = 1; i <= 16; i++) {
        CString str;
        str.Format(_T("COM%d"), i);
        m_comboComPort.AddString(str);
    }
    m_comboComPort.SetCurSel(0);
    
    // 初始化波特率列表
    CString baudRates[] = {_T("1200"), _T("2400"), _T("4800"), 
                          _T("9600"), _T("19200"), _T("38400"),
                          _T("57600"), _T("115200")};
    for (int i = 0; i < 8; i++) {
        m_comboBaudRate.AddString(baudRates[i]);
    }
    m_comboBaudRate.SetCurSel(3); // 默认9600
    
    // 初始化数据位列表
    CString dataBits[] = {_T("5"), _T("6"), _T("7"), _T("8")};
    for (int i = 0; i < 4; i++) {
        m_comboDataBits.AddString(dataBits[i]);
    }
    m_comboDataBits.SetCurSel(3); // 默认8位
    
    // 初始化校验位列表
    CString parity[] = {_T("无"), _T("奇校验"), _T("偶校验")};
    for (int i = 0; i < 3; i++) {
        m_comboParity.AddString(parity[i]);
    }
    m_comboParity.SetCurSel(0); // 默认无校验
    
    // 初始化停止位列表
    CString stopBits[] = {_T("1"), _T("1.5"), _T("2")};
    for (int i = 0; i < 3; i++) {
        m_comboStopBits.AddString(stopBits[i]);
    }
    m_comboStopBits.SetCurSel(0); // 默认1位
    
    // 初始化MSComm控件
    if (!m_mscom.Create(NULL, WS_VISIBLE | WS_CHILD, CRect(0,0,0,0), this, IDC_MSCOMM1)) {
        AfxMessageBox(_T("无法创建MSComm控件!"));
        return FALSE;
    }
    
    // 设置默认参数
    m_mscom.SetCommPort(1);       // 默认COM1
    m_mscom.SetSettings(_T("9600,n,8,1")); // 默认参数
    m_mscom.SetInputMode(1);      // 二进制模式
    m_mscom.SetRThreshold(1);     // 每接收1个字符触发事件
    m_mscom.SetSThreshold(0);     // 不触发发送事件
    m_mscom.SetPortOpen(FALSE);   // 初始关闭
    
    UpdateStatus(_T("就绪"));
    
    return TRUE;
}

// 打开串口
void CSerialCommDlg::OnButtonOpen() {
    CString strCom, strBaud, strDataBits, strParity, strStopBits;
    m_comboComPort.GetWindowText(strCom);
    m_comboBaudRate.GetWindowText(strBaud);
    m_comboDataBits.GetWindowText(strDataBits);
    m_comboParity.GetWindowText(strParity);
    m_comboStopBits.GetWindowText(strStopBits);
    
    int nCom = _ttoi(strCom.Mid(3));
    
    // 转换校验位
    CString parityCode;
    if (strParity == _T("无")) parityCode = _T("n");
    else if (strParity == _T("奇校验")) parityCode = _T("o");
    else if (strParity == _T("偶校验")) parityCode = _T("e");
    else parityCode = _T("n");
    
    CString strSettings;
    strSettings.Format(_T("%s,%s,%s,%s"), strBaud, parityCode, strDataBits, strStopBits);
    
    // 设置串口参数
    if (m_mscom.GetPortOpen()) {
        m_mscom.SetPortOpen(FALSE);
    }
    
    m_mscom.SetCommPort(nCom);
    m_mscom.SetSettings(strSettings);
    
    // 打开串口
    if (!m_mscom.GetPortOpen()) {
        if (m_mscom.SetPortOpen(TRUE)) {
            UpdateStatus(_T("串口已打开: ") + strCom + _T(" ") + strSettings);
            GetDlgItem(IDC_BUTTON_OPEN)->EnableWindow(FALSE);
            GetDlgItem(IDC_BUTTON_CLOSE)->EnableWindow(TRUE);
        } else {
            AfxMessageBox(_T("无法打开串口!"));
        }
    }
}

// 关闭串口
void CSerialCommDlg::OnButtonClose() {
    if (m_mscom.GetPortOpen()) {
        m_mscom.SetPortOpen(FALSE);
        UpdateStatus(_T("串口已关闭"));
        GetDlgItem(IDC_BUTTON_OPEN)->EnableWindow(TRUE);
        GetDlgItem(IDC_BUTTON_CLOSE)->EnableWindow(FALSE);
    }
}

// 发送数据
void CSerialCommDlg::OnButtonSend() {
    if (!m_mscom.GetPortOpen()) {
        AfxMessageBox(_T("串口未打开!"));
        return;
    }
    
    CString strSend;
    m_editSend.GetWindowText(strSend);
    if (strSend.IsEmpty()) {
        return;
    }
    
    // 十六进制发送处理
    if (m_checkHexSend.GetCheck()) {
        CString strHex = strSend;
        strHex.Remove(' ');
        strHex.Remove('-');
        
        if (strHex.GetLength() % 2 != 0) {
            AfxMessageBox(_T("HEX数据长度必须为偶数!"));
            return;
        }
        
        int len = strHex.GetLength() / 2;
        BYTE* data = new BYTE[len];
        ZeroMemory(data, len);
        
        for (int i = 0; i < len; i++) {
            CString byteStr = strHex.Mid(i*2, 2);
            data[i] = (BYTE)strtol(byteStr, NULL, 16);
        }
        
        COleVariant var((BYTE*)data, len);
        m_mscom.SetOutput(var);
        delete[] data;
        
        AddToReceiveBox(_T("[发送] ") + FormatHexData(data, len));
    } 
    // 文本发送
    else {
        COleVariant var(strSend);
        m_mscom.SetOutput(var);
        AddToReceiveBox(_T("[发送] ") + strSend);
    }
    
    UpdateStatus(_T("数据已发送"));
}

// 清空接收区
void CSerialCommDlg::OnButtonClear() {
    m_editReceive.SetWindowText(_T(""));
}

// 保存数据
void CSerialCommDlg::OnButtonSave() {
    CString strData;
    m_editReceive.GetWindowText(strData);
    if (strData.IsEmpty()) {
        AfxMessageBox(_T("接收区为空!"));
        return;
    }
    
    CFileDialog dlg(FALSE, _T("txt"), _T("serial_data.txt"), 
                   OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
                   _T("文本文件 (*.txt)|*.txt|所有文件 (*.*)|*.*||"));
    
    if (dlg.DoModal() == IDOK) {
        CStdioFile file;
        if (file.Open(dlg.GetPathName(), CFile::modeCreate | CFile::modeWrite)) {
            file.WriteString(strData);
            file.Close();
            UpdateStatus(_T("数据已保存: ") + dlg.GetFileName());
        } else {
            AfxMessageBox(_T("保存文件失败!"));
        }
    }
}

// 关闭窗口
void CSerialCommDlg::OnClose() {
    if (m_mscom.GetPortOpen()) {
        m_mscom.SetPortOpen(FALSE);
    }
    CDialog::OnClose();
}

// 串口事件处理
void CSerialCommDlg::OnCommMscomm() {
    VARIANT variant_inp;
    COleSafeArray safearray_inp;
    LONG len, k;
    BYTE rxdata[2048]; // 接收缓冲区
    
    switch (m_mscom.GetCommEvent()) {
    case 2: // 接收事件
        variant_inp = m_mscom.GetInput();
        safearray_inp = variant_inp;
        len = safearray_inp.GetOneDimSize();
        
        // 数据复制到缓冲区
        for (k = 0; k < len; k++) {
            safearray_inp.GetElement(&k, rxdata + k);
        }
        
        // 处理接收到的数据
        ProcessReceivedData(rxdata, len);
        break;
        
    case 4: // 发送事件
        UpdateStatus(_T("数据发送完成"));
        break;
        
    case 5: // 错误事件
        UpdateStatus(_T("通信错误!"));
        break;
        
    default:
        break;
    }
}

// 处理接收数据
void CSerialCommDlg::ProcessReceivedData(BYTE* data, int len) {
    CString strDisplay;
    
    // 十六进制显示
    if (m_checkHexReceive.GetCheck()) {
        strDisplay = FormatHexData(data, len);
    } 
    // 文本显示
    else {
        // 转换为CString
        data[len] = '\0';
        strDisplay = CString((char*)data);
    }
    
    AddToReceiveBox(_T("[接收] ") + strDisplay);
    UpdateStatus(_T("收到数据"));
}

// 格式化十六进制数据
CString CSerialCommDlg::FormatHexData(const BYTE* data, int len) {
    CString strHex;
    for (int i = 0; i < len; i++) {
        CString byteStr;
        byteStr.Format(_T("%02X "), data[i]);
        strHex += byteStr;
        
        if ((i+1) % 16 == 0) {
            strHex += _T("\r\n");
        }
    }
    return strHex;
}

// 添加到接收框
void CSerialCommDlg::AddToReceiveBox(LPCTSTR str) {
    CString strOld;
    m_editReceive.GetWindowText(strOld);
    if (!strOld.IsEmpty()) {
        strOld += _T("\r\n");
    }
    m_editReceive.SetWindowText(strOld + str);
    
    // 滚动到最后一行
    int nLength = m_editReceive.GetWindowTextLength();
    m_editReceive.SetSel(nLength, nLength);
    m_editReceive.ReplaceSel(_T(""));
}

// 更新状态
void CSerialCommDlg::UpdateStatus(LPCTSTR msg) {
    m_staticStatus.SetWindowText(msg);
}

3.4 应用程序类 (SerialCommApp.cpp)

#include "stdafx.h"
#include "SerialComm.h"
#include "SerialCommDlg.h"

BEGIN_MESSAGE_MAP(CSerialCommApp, CWinApp)
    ON_COMMAND(ID_HELP, CWinApp::OnHelp)
END_MESSAGE_MAP()

CSerialCommApp::CSerialCommApp() {
    m_dwRestartManagerSupportFlags = AFX_RESTART_MANAGER_SUPPORT_RESTART;
}

CSerialCommApp theApp;

BOOL CSerialCommApp::InitInstance() {
    INITCOMMONCONTROLSEX InitCtrls;
    InitCtrls.dwSize = sizeof(InitCtrls);
    InitCtrls.dwICC = ICC_WIN95_CLASSES;
    InitCommonControlsEx(&InitCtrls);
    
    CWinApp::InitInstance();
    
    CSerialCommDlg dlg;
    m_pMainWnd = &dlg;
    dlg.DoModal();
    
    return FALSE;
}

四、单片机端示例代码 (C51)

4.1 串口初始化

#include <reg52.h>

#define FOSC 11059200L      // 晶振频率
#define BAUD 9600           // 波特率

void UART_Init() {
    SCON = 0x50;            // 8位数据,可变波特率
    TMOD |= 0x20;           // 定时器1工作方式2
    TH1 = TL1 = -(FOSC/12/32/BAUD); // 设置波特率
    TR1 = 1;                // 启动定时器1
    EA = 1;                 // 开总中断
    ES = 1;                 // 开串口中断
}

void UART_SendByte(unsigned char dat) {
    SBUF = dat;
    while(!TI);
    TI = 0;
}

void UART_SendString(char *s) {
    while(*s) {
        UART_SendByte(*s++);
    }
}

4.2 主程序逻辑

unsigned char receivedData[32];
unsigned char receiveCount = 0;

void main() {
    UART_Init();
    while(1) {
        // 主循环可以添加其他任务
    }
}

// 串口中断服务函数
void UART_ISR() interrupt 4 {
    if(RI) {
        RI = 0; // 清除接收中断标志
        
        unsigned char dat = SBUF;
        
        // 简单协议:收到0x01返回温度数据
        if(dat == 0x01) {
            // 模拟温度数据:25.5℃
            UART_SendByte(0x25); // 整数部分
            UART_SendByte(0x50); // 小数部分
        }
        // 收到0x02返回状态信息
        else if(dat == 0x02) {
            UART_SendString("System OK\r\n");
        }
        // 其他数据回显
        else {
            UART_SendByte(dat);
        }
    }
}

参考代码 利用串口和单片机进行通讯,VC用对话框实现 www.youwenfan.com/contentsfa123975.html

五、使用说明

5.1 系统要求

  • Windows XP/7/10/11
  • Visual C++ 6.0 或更高版本
  • MSComm控件 (mscomm32.ocx)

5.2 安装步骤

  1. 注册MSComm控件:

    regsvr32 mscomm32.ocx
  2. 在VC++项目中:

    • 添加mscomm.h和mscomm.cpp到项目
    • 在stdafx.h中添加:

      #include <mscomm.h>
      #pragma comment(lib, "mscomm.lib")

5.3 操作流程

  1. 选择串口号和通信参数(波特率、数据位、校验位、停止位)
  2. 点击"打开串口"按钮
  3. 在发送区输入数据(文本或HEX格式)
  4. 点击"发送"按钮
  5. 接收区显示返回数据
  6. 使用"清空"、"保存"按钮管理数据

5.4 通信协议示例

方向数据说明
PC→MCU0x01请求温度数据
MCU→PC0x25 0x50温度数据(25.5℃)
PC→MCU0x02请求状态信息
MCU→PC"System OK"状态信息
PC→MCU其他回显数据

六、常见问题解决

6.1 无法打开串口

  • 检查串口号是否正确
  • 确认没有其他程序占用串口
  • 检查串口线连接是否正常
  • 尝试降低波特率

6.2 数据乱码

  • 检查波特率是否匹配
  • 确认数据位、校验位、停止位设置一致
  • 检查是否有干扰源
  • 尝试使用较低的波特率

6.3 接收数据不完整

  • 增加接收缓冲区大小
  • 检查RThreshold设置(建议1-10)
  • 使用轮询方式补充接收
  • 检查硬件流控制设置

七、扩展功能

7.1 添加数据校验

// 在发送数据前添加校验和
BYTE checksum = 0;
for (int i = 0; i < len; i++) {
    checksum ^= data[i];
}
// 将校验和附加到数据末尾

7.2 添加自动发送功能

// 在OnInitDialog中添加定时器
SetTimer(1, 1000, NULL); // 1秒定时

// 添加定时器处理
void CSerialCommDlg::OnTimer(UINT_PTR nIDEvent) {
    if (nIDEvent == 1 && m_mscom.GetPortOpen()) {
        // 自动发送请求
        BYTE cmd = 0x01;
        COleVariant var(cmd);
        m_mscom.SetOutput(var);
    }
    CDialog::OnTimer(nIDEvent);
}

7.3 添加数据绘图功能

// 添加绘图控件
#include <ChartCtrl.h> // 使用第三方图表控件

// 在接收数据中更新图表
CChartCtrl m_chart;
m_chart.AddPoint(x, y); // 添加新数据点
m_chart.Invalidate();

八、总结

本程序使用VC++ MFC对话框实现了PC与单片机的串口通信,具有以下特点:

  1. 完整的通信功能

    • 支持多种串口参数配置
    • 支持文本和十六进制双模式
    • 支持数据发送、接收、显示和存储
  2. 用户友好界面

    • 直观的参数配置
    • 清晰的接收显示区
    • 实时的状态反馈
  3. 稳定的通信机制

    • 基于MSComm控件的中断接收
    • 完善的错误处理
    • 数据校验机制
  4. 良好的扩展性

    • 可添加自定义通信协议
    • 可集成数据分析和可视化
    • 可扩展为多设备通信平台

Lab4AI大模型实验室是面向AI开发者、科研党与学习者打造的一站式AI实践平台,深度绑定高性能弹性算力,支持模型复现、训练、推理全流程,以按需计费、低价高效破解高端算力紧缺与成本高昂难题;同步Arxiv前沿论文并提供翻译、导读、分析服务,支持各类大模型一键复现与数据集微调,对接孵化资源助力科研成果转化;同时搭载多样化AI在线课程,实现理论学习与代码实操同步推进,全方位覆盖AI研发、科研创新与技能学习全场景需求。

大模型实验室官网链接: https://www.lab4ai.cn/arxiv?utm_source=sf_daily_paper

作者单位:北京大学计算机学院

研究背景

  1. 场景图是计算机视觉中建模物体及其语义、空间关系的结构化表示,广泛应用于机器人、自主导航、具身智能等3D环境理解任务。
  2. 现有多视图场景图方法(如MSG)在欧氏空间中通过对比学习与注意力关联学习嵌入,虽能实现较好的位置识别精度,但欧氏几何无法显式建模场景中场所-物体的层级蕴含关系,导致学习到的表示结构一致性不足。
  3. 真实场景具备天然层级结构:场所语义蕴含物体,物体间也存在层级语义关系,而欧氏嵌入难以高效表示这类层级与蕴含关系,常需更高维度或导致结构组织劣化。
  4. 双曲空间具备指数级容量增长特性,天然适配层级与蕴含关系建模,已在视觉表征、视觉-语言对齐等任务中验证有效性,但尚未被用于多视图场景图的层级结构学习。

研究目的

  1. 解决欧氏空间场景图嵌入无法有效捕获场所-物体层级蕴含关系的核心问题,提升场景图的结构一致性与质量。
  2. 提出适配场景图建模的双曲表示学习框架,在保持位置检索性能的同时,显著优化场景图层级结构指标。
  3. 设计显式的层级约束损失,强化场所与物体间的蕴含关系建模,兼容现有MSG构建流程。

本文核心贡献

image

  1. 提出双曲场景图(HSG)框架,首次将双曲几何引入多视图场景图学习,利用双曲空间天然特性编码场景层级关系。
  2. 设计双曲空间下的蕴含损失(Entailment Loss),显式约束场所-物体的层级蕴含结构,提升表示的结构一致性。
  3. 通过大量定量与定性实验验证,HSG在保持高位置检索精度的同时,大幅超越欧氏基线方法,在图级指标上实现显著提升。
  4. 提供兼容现有MSG pipeline的双曲嵌入映射方案,可无缝集成到多视图场景图构建流程中,具备良好扩展性。

研究方法

image

  1. 双曲空间基础:采用洛伦兹双曲面模型(Lorentz hyperboloid model),将欧氏嵌入通过指数映射映射到双曲空间,原点表示最抽象概念,越具体实体距离原点越远。
  2. 模型架构:沿用MSG整体架构,将L2归一化超球嵌入与余弦相似度替换为洛伦兹双曲面嵌入与负洛伦兹距离,新增蕴含损失。
  3. 核心技术

    • 双曲对比学习:基于负洛伦兹距离重构InfoNCE损失,分别优化场所级与物体级对比目标。
    • 蕴含损失:采用双曲蕴含锥,定义场所嵌入为锥中心,约束物体嵌入位于锥内,控制层级约束强度。
    • 总损失:总损失=场所对比损失+物体对比损失+λ×蕴含损失。
  4. 实验设置

    • 骨干网络:DINOv2-Base(最优),辅以ConvNeXt、ViT、ResNet等对比。
    • 数据集:ARKitScenes,4492个训练场景、200个测试场景。
    • 训练配置:AdamW优化器,学习率2e-6,曲率初始值80(可学习),损失权重比1:1:20。
    • 评价指标:Recall@1、PP IoU(场所-场所交并比)、PO IoU(场所-物体交并比)、Graph IoU(图交并比)。

研究结果

  1. 核心性能:HSG的Recall@1达98.39%,与最优欧氏基线相当;PP IoU为33.17%,Graph IoU为33.51%,超越最优AoMSG变体25.37%,提升8.14%。
  2. 维度与骨干影响:投影头维度1024时性能最优;DINOv2-Base作为骨干效果最佳,自监督预训练对双曲场景图学习至关重要。
  3. 消融实验

    • 移除蕴含损失:图级指标小幅下降,验证蕴含损失对层级结构的优化作用。
    • 固定曲率c=1:PP IoU骤降,层级结构失效,说明可学习曲率的必要性。
    • 替换为欧氏损失:图级指标大幅退化,证明双曲表示的核心价值。
  4. 定性结果:HSG的场所嵌入更靠近双曲原点(更抽象),物体嵌入远离原点,呈现清晰层级分布;欧氏基线无明显层级结构。
  5. 超参数鲁棒性:对蕴含锥孔径阈值、InfoNCE温度参数(最优0.1)具备良好鲁棒性。

总结与展望

  1. 研究总结:HSG通过双曲几何学习场景图嵌入,有效捕获场所-物体层级蕴含关系,在保持检索性能的同时大幅提升场景图结构质量,验证了双曲表示在结构化视觉推理中的有效性。
  2. 局限性

    • 双曲投影对投影器维度敏感,过大或过小均影响性能。
    • 性能高度依赖底层编码器质量。
    • 曲率优化方式较为简单,极端曲率易引发数值不稳定。
  3. 未来展望

    • 采用自适应或多阶段曲率优化,提升几何表达能力与训练稳定性。
    • 集成更强基础模型(如DINOv3)与开放词汇检测器(如GroundingDINO),提升泛化性。
    • 结合多模态线索与下游任务联合优化,拓展双曲场景图的应用场景。
    • 探索时序场景图框架,适配长期、大规模环境理解。

laminin521(LN521)在hPSC人多能干细胞培养成本、扩增效率与操作简化对比研究

本文摘要:人多能干细胞培养长期面临成本高、操作繁琐、稳定性差三大痛点。本文基于实验数据,系统解析laminin521(LN521)重组层粘连蛋白作为包被基质的核心优势,从单位面积成本、细胞扩增效率、单位细胞成本、培养便利性四个维度,与截短层粘连蛋白、EHS提取物、玻连蛋白进行对比,证明LN521可显著提升扩增效率、降低整体培养成本、简化操作流程,是hPSC基础研究与临床前培养的高性价比优选方案。


一、hPSC 培养的典型痛点与基质选择重要性

在人多能干细胞(hPSC)日常培养中,科研人员普遍面临三大核心问题:

  1. 成本高:基质、培养基消耗大,长期实验开销显著
  2. 周期长:细胞增殖慢,传代频繁,实验周期不可控
  3. 操作繁琐:需每日换液、手动剔除分化细胞,重复性差

包被基质直接决定细胞黏附、增殖、稳定性与整体成本,选择适配的基质是提升hPSC培养效率的关键(点击查看细胞外基质技术详情)。


二、Biolaminin 521(LN521)核心特点

laminin521是全长重组人源层粘连蛋白,为hPSC生理环境天然存在的细胞外基质,具备多重实验与产业化优势:

2.1 临床级合规稳定

无动物源、无异种成分,批次一致性高,满足标准化与可重复性要求。

2.2 高质量干细胞培养

支持均一、遗传稳定的 hPSC 生长,无需手动去除分化细胞。

2.3 高生理相关性

激活细胞天然信号通路,促进基因均匀表达,提升分化效率与成熟度。

2.4 易用灵活体系

兼容多种培养基;周末无需换液;支持单细胞传代,无需ROCKi;接种密度范围宽,易上手。


三、实验数据对比:LN521vs其他包被基质

本文数据基于 HS181(人胚胎干细胞)、iPSC3(诱导多能干细胞)两株细胞系,对比底物包括:

  • LN521(全长层粘连蛋白)
  • 截短型 LN 片段
  • EHS 肿瘤提取物
  • 玻连蛋白(Vitronectin)

3.1 单位面积培养成本更低

LN521使用浓度仅0.5 μg/cm²,基质消耗量少,单次传代单位面积总成本(基质+培养基)显著低于同类产品。

3.2 细胞扩增效率更高

  • 单次传代增殖倍数更高
  • 生长速率更快,在hESC与iPSC均表现一致优势
  • 相同时间内可获得更高总细胞产量

3.3 单位细胞成本显著更低

因增殖快、产量高,LN521的单次传代单位细胞成本远低于截短LN、EHS、玻连蛋白,整体性价比突出。

3.4 实验更省心:简化操作流程

  • 无需每日换液,**周末可停换液**
  • 单细胞传代,**不用 ROCKi**
  • 无需频繁挑取分化克隆,降低人为误差
  • 接种密度宽松,工艺容错率高

四、laminin521成本及hPSC培养效率数据展示

4.1 单位面积成本分析

hPSC培养的经济可行性首先体现在单位面积的培养成本上。实验数据显示(数据图1),在单次传代过程中,LN521 的单位面积成本(基质与培养基成本合计)低于同类但为截短/片段型层粘连蛋白。较低的基质用量使得单次细胞培养的耗材支出更加可控,为长期培养项目提供了经济基础。

不同包被基质下综合培养基传代的单位面积hPSC培养成本汇算
数据图1. LN521和其他包被基质结合培养基在单位面积下平均的单次传代hPSC培养成本汇算(包被基质(深色)和培养基(浅色斜线)的配制成本均根据市售产品定价计算。实验采用LN521涂层浓度为0.5微克/平方厘米,其他基质材料(截短型片段层粘连蛋白、EHS肿瘤提取物及玻连蛋白Vitronectin)均按制造商说明书使用。培养基用量为每平方厘米0.2毫升,并每日更换。传代次数取两个细胞系(HS181和iPSC3)的平均5-6次传代数据作为基准。)

4.2 细胞扩增效率

细胞产量与生长速度是评估培养体系的核心指标。在 HS181(人胚胎干细胞系)与 iPSC3(诱导多能干细胞系)的对比实验中(数据图2-4),Biolaminin全长LN521 相比于其他类型包被基质,如截短/片段型层粘连蛋白、EHS肿瘤提取物以及玻连蛋白Vitronectin,均获得了更高的细胞产量与更快的生长速度。

相较于其他包被基质BioLamina品牌LN521的每代次hPSC具更高增殖次数
数据图2. LN521和其他包被基质培养HS181和IPSC3单次传代hPSC扩增倍数统计(LN521相较于其他包被基质在不同细胞系中均表现较高的扩增倍数。HS181(人胚胎干细胞)和iPSC3(诱导多能干细胞)细胞系的单次传代中增殖次数基于5-6次传代的平均值计算,接种密度为50,000个细胞/平方厘米。)

相较于其他包被基质BioLamina品牌LN521对HS181细胞系培养具较高扩增速率
数据图3. LN521和其他包被基质培养HS181时细胞的生长速率统计(LN521相较于其他包被基质在HS181细胞系显示出更高的增殖速率。随时间的倍增累积计算为每次传代(5-6次传代)的HS181倍增(log2)的总和。)

相较于其他包被基质BioLamina品牌LN521对生长较慢的IPSC3培养也支持较高扩增速率
数据图4. LN521和其他包被基质培养HS181时细胞的生长速率统计(LN521相较于其他包被基质在生长较慢的IPS C3细胞系显示出更高的增殖速率。)

这种稳定的扩增表现意味着:

  • 相同时间内可获得更多细胞
  • 对生长特性不同的细胞系具有较好的适应性

4.3 单位细胞成本

将成本与产量结合分析,单位细胞成本更能反映培养体系的经济性。由于 LN521 培养体系下 hPSC 生长更快、周期内产量更高,其单次传代的单位细胞成本远低于其他基质(数据图5)。

相较于其他包被基质BioLamina品牌LN521对hPSC培养具每单位细胞省时与省钱成本优势
数据图5. 不同包被基质结合培养基在单次传代的单位细胞hPSC的培养成本汇算(LN521相较于其他包被基质的单次传代周期中基质(深色)和培养基(浅色斜线)单位细胞的成本较低。)

值得注意的是:

  • 结合细胞产量速率分析,基质与培养基的成本分配更均衡
  • 使用全长LN521无需通过过度增加培养基用量来增加产量
  • 接种密度为50,000细胞/cm²,无需特殊调整即可达到良好效果

以上hPSC培养良好表现,主要得益于全长的Biolaminin521(LN521)为hPSC天然环境存在的细胞外基质,在生物相关性较高的环境中,有助于细胞适应体外培养体系,获得较高细胞扩增倍数,获得较高的总细胞产量,进而分摊培养过程中基质和培养基的总体支出。


五、使用laminin521进行hPSC培养的常见问题FAQ

Q1:LN521 包被的使用浓度是多少?

A:标准使用浓度为0.5 μg/cm²,按厂家推荐即可,用量低、性价比高。

Q2:用LN521培养hPSC真的可以周末不换液吗?

A:可以。该体系支持周末无需换液,大幅降低操作负担,且不影响细胞状态。

Q3:使用LN521需要添加ROCKi吗?

A:不需要。支持单细胞直接传代,可省去 ROCKi,降低成本与操作步骤。

Q4:LN521 相比截短型层粘连蛋白优势在哪里?

A:LN521是全长层粘连蛋白,生理信号更完整,细胞黏附/增殖/稳定性更优,成本更低。

Q5:适用于哪些干细胞培养?

A:适用于hESC、iPSC等人多能干细胞的扩增、维持、分化及临床前研究。


六、总结

Biolaminin521(LN521)作为全长重组人源层粘连蛋白包被基质,在hPSC培养中实现了低成本、高扩增、易操作、高稳定的均衡优势,您可点击此处了解曼博生物提供的laminin521产品详情

它不仅能明显降低单位细胞培养成本,还能简化实验流程、提升结果可靠性,是人多能干细胞基础研究、药物筛选、临床前开发的理想选择,真正做到省时、省力、省钱。

在设备登录活动中,计算机账户与用户账户同样重要。如果用户的 Active Directory(AD)计算机账户被删除,该用户将无法登录其设备继续工作。这不仅会增加 IT 团队恢复账户的时间成本,还会导致员工无法正常工作的生产力损失。因此,找出是谁删除了计算机账户,对于管理员分析删除原因、防止类似问题再次发生具有重要意义。下面将介绍具体方法。

使用 PowerShell 查找删除计算机账户的用户步骤

在域控制器(Domain Controller,DC)上执行以下操作:

点击“开始”,搜索 Windows PowerShell,右键选择“以管理员身份运行”。

在控制台中输入以下脚本:

Get-EventLog -LogName Security | Where-Object {$_.EventID -eq 4743} | Select-Object -Property *

按下 Enter 键执行命令。

该脚本会显示已删除的计算机账户相关事件。在输出结果中,定位到 Message > Subject > Account Name,即可查看执行删除操作的用户名称及其安全标识符(SID)。

注意: 如果你是在工作站上执行查询,则需要使用以下脚本:

Get-EventLog -LogName Security -ComputerName <DC name> | Where-Object {$_.EventID -eq 4743} | Select-Object -Property *

image.png

其中 <DC name> 表示你要查询的域控制器名称。

image.png
通过这种方式,可以基于安全日志筛选出计算机账户被删除的事件,从而定位具体执行操作的用户。

需要注意的是,虽然通过原生审计功能可以查询对象删除事件,但当环境中存在大量计算机账户时,需要持续跟踪和分析每一次事件,这种方式会变得非常低效且难以维护。

使用 ManageEngine ADAudit Plus 查找删除计算机账户的用户

相比手动查询日志,使用专业审计工具可以大幅提升效率。通过 ManageEngine ADAudit Plus,管理员可以更快速、直观地获取删除操作的详细信息。

操作步骤如下:

打开 ADAudit Plus 控制台,并使用管理员账号登录。

导航至: Active Directory >计算机管理>最近删除的计算机,导航到报表。

管理员还可以根据不同条件对报告进行筛选,例如:

计算机名称

操作用户创建、删除或修改时间

目标计算机名称
image.png

通过这些筛选条件,可以有针对性地监控关键计算机账户,快速定位删除行为。

使用 ADAudit Plus 相比原生审计的优势

与 Active Directory 原生审计功能相比,ADAudit Plus 提供了更强大、更高效的审计能力。

首先,ADAudit Plus 能够对所有 Active Directory 变更进行持续审计和报告,确保形成完整、可靠的审计轨迹。管理员无需记住每种操作对应的事件 ID,也不需要手动筛选日志,大大简化了运维工作。

其次,平台支持基于机器学习的异常行为检测,可以针对异常操作设置实时告警,并自动触发响应机制,从而帮助企业及时发现和应对内部威胁。

此外,ADAudit Plus 还内置多种合规性报告模板,可帮助组织满足包括 SOX、HIPAA、GLBA、PCI-DSS、FISMA 和 GDPR 在内的多项法规要求,减少审计压力,提高合规效率。

感觉以后的软件发展方向可能不像现在这样了,人们要查询数据的时候,通过网页上的筛选框去搜,我觉得通过 AI 来实现问数才是更方便的一种形式,比如直接和 AI 说,这会儿系统在线人数是多少,然后他就去查询数据库,然后在对话框里面把人数返回给我。再比如我要问我今天的待办事项有多少,他就把事件列表返回给我。和阿里的千问那样很像,直接说我要点外卖,然后就把外卖订单发给我。

我想讨论的是,这个技术路线应该怎么样才比较合理?我现在想到的两种方案:

1 、让大模型理解我的业务系统以及 sql 数据库结构,然后让模型根据问题自己生成 sql 语句,再去执行的话。这中间有个问题,就是它生成的 sql 并不正确,如果是语法错误还好解决,让他再修复一下,但是如果它都理解错了,这个问题就大了。

2 、提供常问的一些固定问题,通过 function call 的方式来实现,这种方式肯定出错的概率就很低,但是吧,用户的问题又受限制,最终模型能够回答多少问题,完全取决于我们给他提供多少接口函数。

还请大家解答一下,关于 AI 问数,有啥很好的参考吗

早上想找个文件,才发现文件打不开了,其中有几个文件夹的文件图标都多了个黄色盒子,点击没有任何反应。可以排除电脑中毒(电脑有些文件夹的文件是正常的),在网上搜了好久,貌似是 Onedrive ,但是不确定。请大佬帮助指点一下怎么处理。

感谢感谢感谢鞠躬

【前言】

在生成式 AI 技术的推动下,企业级办公应用正经历着深刻的范式重构。作为业务数据处理与分析的基石,传统的电子表格如何跨越复杂的交互壁垒,真正迈入“自然语言驱动”的智能时代?面对高达数千个接口的表格 API 与大语言模型固有的“幻觉”挑战,开发者又该如何构建安全、稳定的 AI Agent 工程落地架构?

在近日落幕的“2026赋能开发者大会”产品技术分论坛上,葡萄城 SpreadJS / GcExcel 产品经理张明以《智慧表格:拥抱Web端对话式办公》为主题,全方位揭秘了 SpreadJS 结合 AI 的底层工程化实践,并深度剖析了行业内各大主流智能表格的技术演进路线。

一、 破局与重构:AI 赋予电子表格的新生命

传统电子表格在企业级业务应用中,长期面临着来自终端用户和前端开发者的双重痛点:

  • 终端用户困境:陡峭的学习曲线。传统的 Excel 交互繁杂且不够直观。对于非专业数据人员而言,诸如 VLOOKUP、INDEX、MATCH 等复杂函数和公式的学习成本极高,高级数据分析与报表美化更是存在难以跨越的技术门槛。
    在这里插入图片描述
  • 前端开发者困境:工程化落地的鸿沟。面对 SpreadJS 庞大的 API 库(包含数千个接口),直接结合大语言模型极易触发“模型幻觉”。此外,如何维持表格实例与 AI 之间的状态同步,也是工程落地中的核心难题。
    在这里插入图片描述

为了打破“人找工具”的壁垒,实现从传统操作向“自然语言驱动”的转变,葡萄城 2026 客户峰会-表格分论坛展示了 SpreadJS 结合 AI 后的三大核心业务场景(Demo):

1.具备业务 Know-how 的数据理解:AI 能够智能分析导入的报表结构。例如在处理资产负债表时,AI 不仅能自动生成汇总计算公式,还能执行如 Validations = 资产总计 - (负债合计 + 所有者权益合计) 的专业财务校验逻辑,确保底层计算逻辑的一致性与准确性。
在这里插入图片描述

2.自然语言驱动的高级美化:用户仅需输入简单的格式指令,AI 即可自动应用复杂的商务风格设计规范。系统能够自动执行包括设置主标题为海军蓝粗体、年份表头中蓝底色、明细数据白浅蓝斑马纹、层级缩进以及千位分隔符等一系列繁琐操作,瞬间完成报表的专业化重构。
在这里插入图片描述

3.一键可视化及数据洞察:根据当前选区的数据,AI 可快速提炼出资产规模、流动性、资本结构等关键业务洞察,并自动生成结构清晰的图表(如“2025 年财务健康程度分析”视图),辅助管理层进行快速决策。
在这里插入图片描述

这些能力的落地,使得智能表格能够广泛应用于智能财务风控、自动化审计以及管理层数据驾驶舱等垂直场景。

二、 稳若磐石:工程化架构深度揭秘

要将 AI 可靠地接入极其复杂的表格系统中,底层的架构设计至关重要。本次分论坛公开了 SpreadJS AI Agent 的六层核心架构,并详细拆解了其中的三大技术创新点:

1.全局架构总览

系统自上而下划分为六层:表现层 (Presentation) 负责 UI 与用户交互;状态层 (State) 解决多端同步;业务逻辑层 (Business Logic) 处理对话与分发;服务与 AI 层 (Service / AI) 进行路由与幻觉治理;工具层 (Tool) 定义具体操作;数据与底层操作层 (Data) 负责 API 执行与外部通信。
在这里插入图片描述

2.核心技术点解析

  • 打造专属的工具库 (Tool Registry)
  • 鉴于 SpreadJS 功能跨度极大,且大模型存在上下文限制(一次性暴露过多工具极易导致“认知过载”),系统放弃了单一通用的工具设计。取而代之的是原子化执行策略:将复杂操作拆解为 read_rangeswrite_dataset_cell 等基础原子工具,再结合网关工具和特定行业工具,以此保障极高的执行成功率和精细化的安全合规管控。
    在这里插入图片描述

在这里插入图片描述

  • 基于 SpreadContext 的全局状态同步

为了解决 AI 对话框、Spread 设计器界面、后台工具执行流这三方访问同一工作簿时的冲突问题,引入了 SpreadContext 作为全局状态中心。它安全、高效地管理着 Spread 工作簿实例,确保数据读写的绝对一致性。
在这里插入图片描述

  • 渐进式 API 披露 (ModuleTracker)
  • 这是治理大模型“工具选择幻觉”的核心机制。ModuleTracker 依托于基于有限状态机 (FSM) 的动态路由系统:

    • 默认模式:仅向 LLM 暴露约 30 个最常用的基础数据读写工具和网关工具。
    • 网关触发:当用户下达特定指令(如“创建图表”),LLM 调用 manage_chart 网关工具,触发状态机切换。
    • 进入模块:系统进入专属的 Chart 模块,此时才向 LLM 暴露 add_chart 等深度操作 API。
    • 任务闭环:操作完成后,自动调用 exit_module 退出并重置回默认状态,彻底隔绝无关上下文的干扰。

在这里插入图片描述

在这里插入图片描述

三、 行业视野:对话式智能表格的三大主流实现路线

立足于当前的行业生态,实现对话式智能表格主要有三条主流技术路线,各自具备不同的技术特征与适用场景:

路线模式代表案例原理简述核心优势局限性
Tool-based (后端解析模式)RampLLM 仅输出结构化 JSON,后端进行参数验证 (Schema Validation) 后执行具体操作。具备极高的系统安全性,执行结果高度可预测,且审计链路清晰。灵活性较低,受限于预设的工具集合,难以组合出不可预知的复杂操作。
Code Gen (前端代码生成模式)Shortcut将全量 API 文档作为上下文输入,动态生成 JS 源码块,并通过前端浏览器原生引擎执行。极致灵活,无预设代码限速,可原生支持复杂图表与深度数据验证。安全管控挑战极大,极度依赖顶级大模型的海量 Token 消耗与自我纠错能力。
Python Sandbox (后端沙箱模式)Sourcetable利用云端隔离沙箱运行 Python/Pandas 代码,随后通过 WebSockets 将 DataFrames 渲染结果传输至前端。完美契合数据科学工作流,能够突破 Token 限制处理 PB 级海量数据。系统架构沉重复杂,AI 对前端表格 API 的控制较弱,难以实现单元格级别的精细化格式还原。

(注:业内如扣子(Coze)、Genspark、Skywork 等平台也在积极探索 AI 与办公场景的结合,但底层核心逻辑多跳脱不出上述三种范式。)

在这里插入图片描述

四、 殊途同归:为什么 SpreadJS 是无法替代的底座?

无论企业选择上述哪种工程化架构来构建智能表格系统,最终都需要一个极其强大的前端电子表格组件作为承载底座。SpreadJS 之所以成为通向成功的最优解,原因在于以下三项核心能力:

  1. 高度兼容任意 AI 架构:无论是基于后端的 Tool Calling 验证执行流、前端动态生成的代码执行沙箱,还是前文提到的渐进式状态机,SpreadJS 开放且灵活的 API 体系均能提供完美支撑。
  2. 全面覆盖复杂业务场景的 API 深度:企业级需求远不止于简单的“数据填充”。面对 AI 生成的复杂逻辑(如透视表 PivotTable、条件格式 ConditionFormat、高级图表 Charts 等),SpreadJS 具备无损的、企业级的渲染与还原能力。
  3. 卓越的运行态控制与快照提取:SpreadJS 提供的原生 Headless 能力以及结构化数据导出功能,是支撑 AI 获取精准上下文 (Context) 并进行多模态理解的底层基石。
    在这里插入图片描述
    为了进一步降低企业接入门槛,赋能开发者群体,SpreadJS AI Agent Framework 现已正式开源。开发者可通过访问 Gitee 官方仓库 (gitee.com/grapecity/spreadjs-ai-agent) 获取完整源码,深入体验这套融合了结构化 Tool Calling、模块化状态机与受保护沙箱的先进框架体系,共同迈向对话式办公的新纪元。
    在这里插入图片描述

核心功能模块:

商品展示:展示护航代练套餐,如红物资、吞天包等,提供详细的服务描述和价格信息。

快速下单:简化下单流程,支持一键提交,实现“秒上号”核心卖点。

订单管理:用户可查看订单进度,商家可接单处理订单,支持订单状态跟踪、支付、售后等功能。

基础用户体系:支持手机号登录、微信授权登录等多种登录方式,提供个人中心页面,展示用户信息、订单记录等。

抢单与派单双模式:支持打手主动抢单(提升积极性)或客服手动派单(确保高价值订单分配),提高派单效率和打手日均接单量。

运营赋能:从搭建到盈利的闭环支持

快速部署:提供详细图文教程与环境配置指南(Nginx+PHP+MySQL5.6),技术小白也可在1小时内完成服务器搭建与小程序上线。

多俱乐部管理:支持无限创建独立俱乐部站点,总后台可统筹各俱乐部数据,子后台实现本地化运营(如定制打手招募规则、玩家会员体系),适合团队跨区复制盈利模式。

盈利模式多元:通过订单抽佣、打手入驻费、会员增值服务(如优先抢单、专属客服)、广告位出租等方式实现变现,源码内置财务结算模块,自动统计收益与提现记录。

在新零售浪潮席卷全球的今天,传统商超正加速向数字化、智能化转型。作为连接商品、消费者与后台管理的核心枢纽,一套高效、稳定、易用的收银系统已成为现代商超运营不可或缺的基础设施。OctShop不仅是一款广受认可的开源B2B2C电商系统,其延伸打造的商超收银系统模块,正以软硬结合、前后端协同、线上线下融合的优势,为中小型超市、便利店、生鲜店及社区零售门店提供一体化智能收银解决方案。

图片

OctShop商超收银系统源码详细介绍: https://pc.opencodetiger.com

OctShop商超收银系统并非简单的支付终端,而是一个集商品管理、库存同步、会员营销、销售分析与多支付方式于一体的全链路零售操作系统。它深度集成于OctShop整体电商生态中,既可独立部署用于线下门店,也能与线上商城无缝打通,真正实现“一店双营”——线下收银、线上下单、库存共享、会员互通。系统采用轻量化设计,支持Windows、Linux及国产操作系统,兼容主流POS机、扫码枪、小票打印机、钱箱、电子秤等外设设备。商家只需普通电脑或平板搭配基础硬件,即可快速搭建专业收银台,大幅降低初期投入成本。同时,得益于OctShop开源架构,系统支持高度定制化,可根据不同业态(如生鲜、烟酒、日百)灵活配置商品分类、促销规则与结算流程。在核心功能上,OctShop收银系统突出“快、准、稳”三大特点:快:扫码即出商品信息,支持批量扫码、组合商品、快速退货,单笔交易平均处理时间低于5秒;准:实时同步云端商品库与价格策略,杜绝价签不符或库存超卖;支持按重量、按件数、按规格等多种计价方式,尤其适合生鲜场景;稳:本地缓存+云端备份双机制,即使网络中断仍可离线收银,恢复后自动同步数据,保障营业连续性。更值得称道的是其强大的会员与营销能力。顾客在收银时可即时注册会员、积分累积、使用优惠券或参与满减活动。系统自动记录消费行为,生成用户画像,为后续精准营销提供数据支撑。商家可通过后台一键发放电子优惠券、设置生日特权、开展储值返现等活动,有效提升复购率与客户黏性。

图片

在支付方面,OctShop收银系统全面支持微信、支付宝、银联云闪付、数字人民币、会员余额、现金、银行卡等多种支付方式,并可自定义组合支付(如“微信,积分”)。所有交易流水实时入账,对账清晰,财务人员可随时导出日报、月报,极大简化财务管理流程。此外,系统内置智能库存预警与采购建议功能。当某商品库存低于设定阈值时,自动提醒补货;结合历史销售数据,还能预测热销品趋势,辅助采购决策。对于拥有多门店的连锁商超,OctShop支持总部统一管理商品、价格与促销策略,各门店独立运营但数据集中可视,实现“千店千面,一盘棋管”。安全性方面,OctShop收银系统遵循金融级数据加密标准,操作日志全程留痕,权限分级管控(如收银员仅能结账,店长可查看报表),有效防范内部操作风险。所有敏感信息均不存储于本地设备,确保用户隐私与商业数据安全。值得一提的是,作为开源项目,OctShop商超收银系统无授权费、无年服务费,企业可自主部署、自由修改,避免被SaaS厂商“锁定”。活跃的开发者社区和详尽的技术文档,也为后续维护与功能扩展提供了坚实保障。

图片

总之,OctShop商超收银系统不仅是收银工具,更是商超数字化转型的入口。它以开源为基、以体验为先、以效率为核心,帮助传统零售门店降本增效、提升服务、连接线上,真正迈向“智慧零售”新阶段。在竞争日益激烈的零售市场中,选择OctShop,就是选择一个开放、灵活、可持续进化的未来。

亚马逊云科技宣布 DevOps Agent 正式可用,这是一款由生成式 AI 驱动的智能助手,旨在帮助开发者和运维人员排查问题、分析部署,并在 AWS 环境中自动化执行运维任务。

该服务在 2025 年的 re:Invent 大会上预览发布,基于 Amazon Bedrock AgentCore 构建。DevOps Agent 通过学习应用关联关系并集成可观测性工具、运行手册、代码仓库和 CI/CD 管道来分析事件。该智能体可将遥测数据、代码和部署数据关联起来,自主完成问题分类与排查,加快故障解决,并从历史事件中识别规律、给出优化建议,助力防范后续故障。AWS 高级解决方案架构师 Madhu Balaji 在宣布正式可用时表示

SRE 在凌晨 2 点收到告警时往往需要手动整合多个来源的遥测数据,梳理跨服务依赖关系并进行问题假设,这一过程通常耗时数小时。随着系统复杂度的增加,对 AI 驱动的运维队友——SRE 智能体的需求变得日益迫切。

正式可用版本的主要改进包括:支持对 Azure 及本地环境中的应用进行排查、支持通过自定义智能体 Skill 扩展能力,同时新增自定义图表与报告功能。Balaji 补充道:

DevOps Agent 并非一个被动的问答工具,而是一个能自主行动的运维助手。当事件通过 CloudWatch 告警、PagerDuty 警报、Dynatrace 问题、ServiceNow 工单或通过 WebHook 配置的其他任意事件源触发时,该智能体无需人工干预即可立即启动排查工作。

在另一篇文章中,Janardhan Molumuri、Bill Fine、Joe Alioto 和 Tipu Qureshi 以一个无服务器 URL 短链应用为例,解释了如何利用智能体式 AI 通过 DevOps Agent 实现自主事件响应。他们写道:

借助 MCP 的可扩展性以及与 CloudWatch、Datadog、Dynatrace、New Relic、Splunk、Grafana、GitHub、GitLab 和 Azure DevOps 的内置集成,智能体可以从团队运维数据所在的任意位置获取监测信号。

来源:AWS 博客

亚马逊云科技表示,DevOps 团队通常会借助接入日志与监控系统的 AI 编码工具开展事件排查,但这类工具缺乏在大规模复杂生产环境中管理所需上下文信息和运维管控能力。Agentic Hamburg 联合创始人 Sebastian Korfmann 写道

早期数据表现亮眼:预览阶段的平均故障恢复时间(MTTR)最高降低 75%,根因分析准确率达到 94%,可与 Datadog、Grafana、Splunk、PagerDuty、ServiceNow 等平台集成。

The Duckbill Group 首席云经济学家 Corey Quinn 评论道

你花钱让 AI 去做凌晨两点值班工程师的活儿,只不过它事后不会在 Slack 上阴阳怪气地 @ 整个团队。平均故障恢复时间从小时级压缩到分钟级,而账单则从按分钟计费变成了按小时计费。

Reddit 上的一个热门讨论帖中,不少开发者对其缺乏问责机制提出了质疑,用户 The_Flexing_Dude 问道:

这和上个月搞崩生产环境的是同一个吗?

随着该服务正式可用,不再提供免费使用,定价基于代理执行运维任务的累计时长,按秒计费。AWS Support 客户可根据上月支持服务支出,获得每月对应的 DevOps Agent 使用额度,可使用额度的百分比随支持级别而定。目前该服务已在六个区域上线,包括弗吉尼亚北部、爱尔兰和法兰克福。

在另一项公告中,亚马逊云科技宣布 Security Agent 按需渗透测试功能正式可用。这款 AI 驱动的智能体可持续分析应用设计、代码和运行时行为,自动执行按需渗透测试并识别可被利用的安全漏洞。

【声明:本文由 InfoQ 翻译,未经许可禁止转载。】

查看英文原文:https://www.infoq.com/news/2026/04/aws-devops-agent-ga/

现在主要能做这些:

  • 文生图
  • 图生图
  • 上传参考图继续改
  • 看案例后直接一键去生成类似方向
  • 新用户登录后可以免费体验 2 次

这两天自己用下来,感觉这一代最明显的还是图中文字、细节稳定性,还有整体画面完成度,拿来做海报、封面图、带字图片这些场景 会顺手很多。

体验地址:
https://gptimage2.top/zh/create

案例页:
https://gptimage2.top/zh/showcases

如果有兴趣的话欢迎帮我试试,也欢迎直接吐槽下哪里难用、还缺什么功能,我继续改。

有一个充不上电,然后需要手按下去才可以冲上,去官方店维修大概多少钱?

或者有其他的便宜维修渠道么?

准备把囤积的金庸武侠电视剧看完

  1. 射雕英雄传 - 黄日华版(已看完)
  2. 神雕侠侣 - 古天乐版 + 黄晓明版(已看完)
  3. 倚天屠龙记 - 苏有朋版(待看)
  4. 天龙八部 - 黄日华版 + 胡军版(待看)
  5. 笑傲江湖 - 霍建华版(已看完)
  6. 雪山飞狐 - 黄日华版(待看)
  7. 连城诀(已看完)

说什么还有宅基地
还有口粮田
问题是这宅基地和口粮田在农村根本不值钱
我们那农村很多宅基地房子下过大雨塌了也没人管
还有人均一亩的口粮田,种粮食抛去成本,白搭人工,一年只能赚几百块
还说农村老年人可以去打工赚钱
先不说年纪大了百病缠身
健康的农村老年人的打工渠道只有时薪 5 块钱的干绿化拔草的渠道

之前和对象吵了一架,大概就是「我自认为已经对你很好了你怎么还说不够」那种经典剧情。事后想了解下这种问题怎么办,发现了一本书《 The 5 Love Languages: The Secret to Love That Lasts 》(中文版常译《爱的五种语言》/《爱之语》)

然后借此理论搭了个网站叫 LoveBridge: https://lovebridge.love
中文直达: https://lovebridge.love/zh/

感觉测评出来还挺有道理的,有类似困惑的可以试一下,比如有的人会喜欢生活中的小礼物,有的人更喜欢拥抱等,每个人感受爱意的方式不一样。这个网站简单说就是 30 道题 5 分钟做完,不用注册不用邮箱。你做完会生成一条链接发给对方,对方答完才出双人对比那一页,告诉你你最常用哪种方式表达爱、对方最容易接收哪种、错位在哪、下一步可以怎么试一下。

今天我们这里(坐标广西)突然降温,但是发现 agent 查找的数据来源是:Open-Meteo ,与中国气象局、天气网的气温都相差非常的大。
所以想问问各位平时都是看哪个平台的天气预报,或者让 hermes agent 查看哪个平台的天气进行推送?

接上次帖子,年后回来之后就开始看新工作机会,我想换城市北边的工作,但是这个城市软件开发的岗位基本都在城市南边(我现在的工作也是在南边),所以北边基本没有几个岗位。

2 月份的时候刷到一家各项要求都挺符合的公司,去面试之后也通过了,但是聊下来发现有大坑,就拒绝了。

后面一个多月,再也没刷到北边有新的岗位,前段时间就换了一下思路,既然北边没有,那就改成看西边靠近北边的岗位,然后投了一家意向公司,也收到了面试邀请。

我面试的是 Flutter 开发( Android+硬件方向),初试的时候面试我的应该是做 Flutter 或者前端的,因为似乎他对原生 Android 不太了解,全程是拿着笔记本搜面试题问我,问的也是些老旧的 Android 八股文问题。总之问了很多技术问题。

过了几天 HR 告诉我初试通过了,和我约时间复试。

约的 14 点面试,但是去了之后 HR 告诉我面试官正在和经理开会,不好中途离席,让我等等。结果等了一个多小时,面试官才急匆匆的赶回来。我还以为是人事面试,原来是技术总监面试。技术总监基本没问几个问题,就看了下我的简历,简单问了些简历上的东西,全程不到十分钟就结束了。

几天后 HR 发消息给我说,这个岗位暂停招聘了。

这家公司我搜了下情况,目前正在 Pre-IPO ,而且是第二次提交招股书了,去年提交了一次没过,今天刚又提交了一次。

让 AI 分析了一下情况,AI 说这家公司目前财务状况岌岌可危,如果这次 IPO 再失败基本就宣布倒闭了,并且现在在大量招人(招聘软件上开放岗位有 2000+),大概率是为了 IPO “凑人头”,IPO 过后不管成功还是失败,被裁的概率都非常大。

本来我的打算是,如果面试通过了,只要薪资开的符合预期,其他方面不是太坑我都准备去了。

就算半年后( IPO 期限是半年)被裁也无所谓了,当然不被裁肯定是最好的,但是就算被裁也没事,正好可以刷新一下我简历上的薪资,下次跳槽谈薪底气能足点。(我现在的薪资水平远低于行业水平,每次面试谈薪都非常被动)。

结果我自己算盘打的挺响,人家还不要我呢,哈哈哈哈。

现在刷招聘软件,连西边也没有新岗位了。

我很纠结两件事:

  1. 一定要跳槽吗?
  2. 一定要去北边吗?

想跳槽的原因上面也提到了,因为我现在薪资实在太低了,工作起来完全没有动力,现公司也明确了,不会给调薪了,而且现在在这家公司待的也不是很爽。

想去北边的原因是我对象在北边事业编,房子也买在了北边,我们计划一两年之后结婚,到时候我肯定是越靠近北边越好。

简介

.NET 异步编程里,Task 大多数时候都是“自动完成”的。

比如:

  • async 方法执行完了,返回的 Task 自动完成;
  • HttpClient.GetAsync 底层 I/O 完成了,Task 自动完成;
  • Task.Run 里的委托跑完了,Task 自动完成。

但还有一类场景,不是“代码块执行完就结束”,而是:

  • 某个回调什么时候被触发,不确定;
  • 某个事件什么时候到来,不确定;
  • 某个外部信号什么时候准备好,不确定;
  • 你需要自己决定这个异步操作什么时候成功、失败或取消。

这时候,Task 就不能只靠“自动执行”来产生了,而需要一个“手动完成器”。

TaskCompletionSource<T> 就是干这个的。

一句话先给结论:

TaskCompletionSource<T> 的作用,不是启动任务,而是手动控制一个 Task<T> 什么时候完成。

这篇文章重点讲清楚几件事:

  • TaskCompletionSource<T> 到底是什么;
  • 它和 Task.Runasync/await 的边界在哪里;
  • 回调、事件为什么经常要靠它桥接成 Task
  • SetResultSetExceptionSetCanceled 到底意味着什么;
  • 为什么很多代码都应该优先用 TrySet*
  • RunContinuationsAsynchronously 为什么是实战里的关键选项;
  • 使用 TaskCompletionSource<T> 最容易踩的坑有哪些。

先拆几个最容易混淆的点

1. TaskCompletionSource<T> 不负责执行工作

很多人第一次看到它,会误以为它和 Task.Run 差不多,也是“创建一个异步任务”。

其实不是。

Task.Run 的重点是:

  • 把一个委托扔到线程池执行;
  • 由运行时去调度这段代码;
  • 最终把执行结果包装成一个 Task

TaskCompletionSource<T> 的重点是:

  • 先生成一个还没完成的 Task<T>
  • 什么时候完成,不由委托自动决定;
  • 而是由你在外部手动调用 SetResult / SetException / SetCanceled 来决定。

所以它更像:

  • 一个 Task 的生产者控制器;
  • 而不是一个工作执行器。

2. 它不等于“开线程”

TaskCompletionSource<T> 本身不会新开线程,也不会自动占用线程。

例如:

var tcs = new TaskCompletionSource<string>();

这行代码只是创建了一个“尚未完成的任务源”,并没有开始任何后台工作。

后面如果有别的线程、回调、事件、定时器或 I/O 完成信号去调用:

tcs.SetResult("ok");

等待这个 Task 的代码才会继续。

3. 它最适合做“桥接”

TaskCompletionSource<T> 最常见的价值,不是替代 async/await,而是补上 Task 世界和“非 Task 世界”之间的缺口。

比如:

  • 老式回调 API;
  • 事件通知模型;
  • 某些底层协议回包;
  • 自定义同步原语;
  • 一个操作的完成由多个条件共同决定。

这些场景天然不是 Task 形式,但业务代码又很希望能直接:

await SomethingAsync();

这时就需要 TaskCompletionSource<T> 来做桥接。

TaskCompletionSource<T> 到底是什么?

可以把它拆成两部分理解:

1. 它持有一个 Task<T>

通过 Task 属性,你可以拿到一个供外部等待的任务:

var tcs = new TaskCompletionSource<int>();
Task<int> task = tcs.Task;

调用方拿到的只是这个 Task<int>,并不知道也不应该知道背后的完成细节。

2. 它掌握这个 Task<T> 的完成权

你可以显式控制三种完成方式:

tcs.SetResult(123);
tcs.SetException(new InvalidOperationException("failed"));
tcs.SetCanceled();

也就是说,TaskCompletionSource<T> 的本质是:

让我自己成为这个 Task<T> 的完成者。

这套模型很像生产者和消费者:

  • 生产者:TaskCompletionSource<T>,负责发出完成信号;
  • 消费者:await tcs.Task 的代码,负责等待结果。

它和 Task.Run 到底有什么区别?

这是最常见、也最值得单独拉出来讲的点。

对比项Task.RunTaskCompletionSource<T>
核心职责调度代码执行手动控制 Task 完成
是否自带执行委托
是否通常依赖线程池不一定
适合场景CPU 密集型工作、包装同步阻塞代码回调桥接、事件桥接、自定义异步协调
完成时机委托跑完后自动完成Set* / TrySet* 手动决定

最简单的判断方式是:

  • 你要“让一段代码异步跑起来”,优先想 Task.Run
  • 你要“把一个外部信号转换成可 await 的任务”,优先想 TaskCompletionSource<T>

基础用法先跑通

先看一个最小示例:

public static async Task DemoAsync()
{
    var tcs = new TaskCompletionSource<int>();

    _ = Task.Run(async () =>
    {
        await Task.Delay(1000);
        tcs.SetResult(42);
    });

    int result = await tcs.Task;
    Console.WriteLine(result);
}

这段代码真正重要的不是 Task.Run,而是流程:

  1. 创建一个未完成的 TaskCompletionSource<int>
  2. tcs.Task 暴露给等待方;
  3. 将来某个时刻手动调用 SetResult(42)
  4. await tcs.Task 恢复执行,拿到结果。

这里的 Task.Run 只是为了模拟“未来某个时刻有外部信号到来”,真实项目里它更可能来自:

  • Socket 回调;
  • 消息队列回包;
  • UI 事件;
  • 定时器;
  • 某个订阅通知。

await tcs.Task 时到底发生了什么?

理解 TaskCompletionSource<T>,最好别只停留在“能用”。

看下面这段代码:

var tcs = new TaskCompletionSource<string>();

var task = WaitAsync();
tcs.SetResult("ok");

async Task WaitAsync()
{
    string value = await tcs.Task;
    Console.WriteLine(value);
}

执行流程可以概括成这样:

  1. 创建 TaskCompletionSource<string> 时,内部先有了一个未完成的 Task<string>
  2. await tcs.Task 发现任务还没完成,于是当前方法先挂起,并把“后续怎么恢复执行”注册到这个 Task 上。
  3. 之后某个时刻,外部代码调用 SetResult("ok")
  4. 这个 Task 被标记为成功完成,等待它的 continuation 开始恢复。
  5. await 后面的代码继续往下执行,拿到结果 "ok"

所以更准确地说:

  • await 做的是“注册后续逻辑并在未完成时先返回”;
  • SetResult 做的是“宣布任务已经完成,可以恢复等待方了”。

这也是为什么 TaskCompletionSource<T> 特别适合桥接回调和事件。

因为回调、事件这类模型,本质上都缺一个东西:

  • 一个能被 await 直接等待的完成信号。

TaskCompletionSource<T> 刚好把这个信号补出来了。

一张图看懂 TaskCompletionSource<T> 的工作流程

如果你想把上面的过程快速记成一张图,可以直接看下面这个时序图:

sequenceDiagram
    participant Caller as 调用方
    participant AsyncMethod as 异步方法
    participant TCS as TaskCompletionSource
    participant External as 外部回调/事件

    Caller->>AsyncMethod: 调用方法
    AsyncMethod->>TCS: 创建 TaskCompletionSource
    AsyncMethod-->>Caller: 返回 tcs.Task
    Caller->>TCS: await tcs.Task
    Note over Caller,TCS: Task 尚未完成,await 挂起并注册 continuation
    External->>TCS: 某个时刻触发 SetResult / SetException / SetCanceled
    Note over TCS: Task 状态变为完成
    TCS-->>Caller: 恢复 await 后续逻辑
    Caller->>Caller: 继续执行后面的代码

这张图最关键的信息只有两点:

  • await 的本质不是“卡住线程等结果”,而是“先挂起,等 Task 完成后再恢复”;
  • TaskCompletionSource<T> 的本质不是“执行异步工作”,而是“在合适的时机把这个 Task 变成已完成”。

三种完成方式分别意味着什么?

1. 成功完成:SetResult

var tcs = new TaskCompletionSource<string>();
tcs.SetResult("done");

string result = await tcs.Task;

此时:

  • Task 状态变成成功完成;
  • await 直接拿到返回值;
  • 后续 continuation 会被触发。

2. 异常完成:SetException

var tcs = new TaskCompletionSource<string>();
tcs.SetException(new InvalidOperationException("bad state"));

string result = await tcs.Task;

此时 await 会重新抛出异常。

也就是说,SetException 不是“记录一下错误”,而是明确告诉等待方:

这次异步操作失败了,应该按异常路径处理。

3. 取消完成:SetCanceled

var tcs = new TaskCompletionSource<string>();
tcs.SetCanceled();

string result = await tcs.Task;

此时 await 会抛出与取消相关的异常。

这条路径和异常路径看起来相似,但语义不一样:

  • SetException:操作失败了;
  • SetCanceled:操作没有继续执行下去,属于取消。

如果你手里有对应的 CancellationToken,也可以带上它:

tcs.SetCanceled(cancellationToken);

这样等待方能保留更完整的取消上下文。

为什么实战里更推荐 TrySet*

SetResultSetExceptionSetCanceled 都有一个共同前提:

  • 这个 Task 之前还没完成过。

如果已经有别的线程或别的回调先一步完成了它,再调 Set* 就会抛异常。

所以在这些场景里,通常更推荐:

  • 多个竞争路径都可能完成任务;
  • 超时、取消、正常结果可能同时抢完成权;
  • 事件或回调可能重复触发;
  • 并发环境下存在竞态。

示例:

var tcs = new TaskCompletionSource<string>();

_ = Task.Run(async () =>
{
    await Task.Delay(1000);
    tcs.TrySetResult("success");
});

_ = Task.Run(async () =>
{
    await Task.Delay(500);
    tcs.TrySetCanceled();
});

这里最终只有一个分支会成功完成任务,另一个分支会返回 false,但不会抛异常。

因此更务实的经验是:

  • 明确只有单一路径完成时,可以用 Set*
  • 只要存在竞争,优先用 TrySet*

最经典的场景:把回调 API 包成 Task

假设你有一个老式 API:

public void BeginLoadUser(Action<User> onSuccess, Action<Exception> onError)
{
    // 某个库内部完成后回调
}

如果直接使用,调用方通常会写成回调嵌套。

更现代的写法往往希望是:

User user = await LoadUserAsync();

这时就可以这样桥接:

public Task<User> LoadUserAsync()
{
    var tcs = new TaskCompletionSource<User>(
        TaskCreationOptions.RunContinuationsAsynchronously);

    BeginLoadUser(
        user => tcs.TrySetResult(user),
        ex => tcs.TrySetException(ex));

    return tcs.Task;
}

这个例子里,TaskCompletionSource<User> 做了两件事:

  • 把原本的回调模型转换成 Task<User>
  • 把成功和失败语义自然映射进 await 流程。

于是上层代码就能写成:

var user = await LoadUserAsync();

这也是 TaskCompletionSource<T> 最标准、最有价值的用法之一。

第二个高频场景:把事件变成可等待任务

比如你想“等待下一条消息到来”:

public Task<string> WaitNextMessageAsync(MessageClient client)
{
    var tcs = new TaskCompletionSource<string>(
        TaskCreationOptions.RunContinuationsAsynchronously);

    void OnMessage(object? sender, MessageEventArgs e)
    {
        client.MessageReceived -= OnMessage;
        tcs.TrySetResult(e.Text);
    }

    client.MessageReceived += OnMessage;

    return tcs.Task;
}

调用方就可以:

string message = await WaitNextMessageAsync(client);

但这种写法有一个很关键的细节:

  • 事件一旦完成,要及时解绑;
  • 否则可能造成重复触发、内存泄漏,甚至错误完成别的等待操作。

如果还需要支持异常和取消,就要把清理逻辑补完整。

再看一个更接近实战的版本:事件 + 取消

public Task<string> WaitNextMessageAsync(
    MessageClient client,
    CancellationToken cancellationToken = default)
{
    var tcs = new TaskCompletionSource<string>(
        TaskCreationOptions.RunContinuationsAsynchronously);

    EventHandler<MessageEventArgs>? handler = null;
    CancellationTokenRegistration registration = default;

    handler = (sender, e) =>
    {
        client.MessageReceived -= handler;
        registration.Dispose();
        tcs.TrySetResult(e.Text);
    };

    client.MessageReceived += handler;

    if (cancellationToken.CanBeCanceled)
    {
        registration = cancellationToken.Register(() =>
        {
            client.MessageReceived -= handler;
            tcs.TrySetCanceled(cancellationToken);
        });
    }

    return tcs.Task;
}

这里要注意三点:

  • 事件完成后要解绑;
  • 取消时也要解绑;
  • CancellationTokenRegistration 也应该及时释放。

否则代码“逻辑上能跑”,但长期运行会留下资源和行为问题。

一个更像生产代码的包装模板

很多时候,真正难的不是“怎么把回调转成 Task”,而是怎么把收尾逻辑放对位置。

下面这个模板更接近实际项目里的写法:

public Task<string> SendAndWaitAsync(
    Request request,
    CancellationToken cancellationToken = default)
{
    var tcs = new TaskCompletionSource<string>(
        TaskCreationOptions.RunContinuationsAsynchronously);

    EventHandler<ResponseEventArgs>? handler = null;
    CancellationTokenRegistration registration = default;

    void Cleanup()
    {
        _client.ResponseReceived -= handler;
        registration.Dispose();
    }

    handler = (sender, e) =>
    {
        if (e.RequestId != request.Id)
        {
            return;
        }

        Cleanup();
        tcs.TrySetResult(e.Payload);
    };

    _client.ResponseReceived += handler;

    if (cancellationToken.CanBeCanceled)
    {
        registration = cancellationToken.Register(() =>
        {
            Cleanup();
            tcs.TrySetCanceled(cancellationToken);
        });
    }

    try
    {
        _client.Send(request);
    }
    catch (Exception ex)
    {
        Cleanup();
        tcs.TrySetException(ex);
    }

    return tcs.Task;
}

这个模板里有几个关键点:

  • 先订阅,再发请求,避免响应回来得太快而错过事件;
  • 统一抽一个 Cleanup,避免成功、失败、取消三条路径清理不一致;
  • 回调里先过滤无关消息,再尝试完成 TCS
  • 发送请求本身如果同步抛错,也要把 Task 走到异常完成,而不是让等待方永远挂住。

TaskCompletionSource<T> 和超时控制怎么配合?

超时控制也是它的高频用法。

例如,我们想等待某个外部响应,但最多等 5 秒:

public async Task<string> WaitResponseAsync()
{
    var tcs = new TaskCompletionSource<string>(
        TaskCreationOptions.RunContinuationsAsynchronously);

    using var cts = new CancellationTokenSource(TimeSpan.FromSeconds(5));

    using var registration = cts.Token.Register(() =>
    {
        tcs.TrySetException(new TimeoutException("等待响应超时。"));
    });

    StartRequest(response =>
    {
        tcs.TrySetResult(response);
    });

    return await tcs.Task;
}

这里本质上是两个完成路径在竞争:

  • 正常回包;
  • 超时触发。

所以使用 TrySet* 才更稳妥。

不过要特别注意一个边界:

你让 TaskCompletionSource<T> 超时完成,并不等于底层真实操作一定被取消了。

这点非常重要。

比如:

  • 你只是让等待方不再继续等;
  • 但底层网络请求、设备操作、第三方 SDK 任务,可能还在继续跑。

所以“超时了”分成两个层面:

  • 等待逻辑超时;
  • 底层操作真的被取消。

如果你需要两者都成立,就必须把取消信号继续传到底层系统,而不能只完成一个 TCS

RunContinuationsAsynchronously 为什么这么重要?

这是 TaskCompletionSource<T> 最容易被忽视,也最容易在生产环境出问题的一点。

先看现象:

当你调用:

tcs.SetResult(value);

如果没有额外选项,等待这个任务的 continuation 有可能就在当前线程上同步执行。

这会带来几个风险:

  • 让触发完成的线程顺带执行一大段后续逻辑;
  • 回调链条彼此嵌套,增加栈深度;
  • 某些锁、串行队列或单线程上下文里,更容易形成卡死或延迟放大。

所以更稳妥的构造方式通常是:

var tcs = new TaskCompletionSource<string>(
    TaskCreationOptions.RunContinuationsAsynchronously);

这个选项的意义是:

  • 即使任务被完成了;
  • 后续 continuation 也尽量异步调度出去;
  • 而不是直接在当前 SetResult 的线程里内联执行。

这并不是说“任何时候都必须加”,但在绝大多数通用库、基础设施代码、并发协调代码里,它通常都是更安全的默认选择。

可以把它理解成:

不要让“完成任务的人”顺手把“等待任务后的整段业务逻辑”也一起跑掉。

一张图看懂 RunContinuationsAsynchronously 的区别

这个选项抽象上不复杂,但很多人第一次读文字说明时,很难立刻建立画面感。

可以直接看下面这组对比图。

未开启 RunContinuationsAsynchronously

sequenceDiagram
    participant Producer as 完成任务的线程
    participant TCS as TaskCompletionSource
    participant Awaiter as await 后续逻辑

    Producer->>TCS: SetResult()
    TCS-->>Awaiter: 可能直接内联执行 continuation
    Awaiter-->>Producer: 当前线程继续跑后续业务代码

开启 RunContinuationsAsynchronously

sequenceDiagram
    participant Producer as 完成任务的线程
    participant TCS as TaskCompletionSource
    participant Scheduler as 调度器 / 线程池
    participant Awaiter as await 后续逻辑

    Producer->>TCS: SetResult()
    TCS->>Scheduler: 投递 continuation
    Scheduler-->>Awaiter: 稍后异步执行 continuation

如果把这张图翻成更直白的话,就是:

  • 没开 RunContinuationsAsynchronously 时,SetResult() 的那个线程,可能顺手把 await 后面的代码也一起执行了;
  • 开了之后,完成任务和执行 continuation 这两件事会尽量拆开,后续逻辑改为异步调度。

这也是为什么在这些场景里,它通常更值得加上:

  • 锁内部完成任务;
  • 事件回调线程里完成任务;
  • 单线程上下文或串行执行器里完成任务;
  • 通用库、基础设施组件、并发协调组件。

一个典型坑:在锁里完成 TaskCompletionSource<T>

看一个简化例子:

private readonly object _lock = new();
private TaskCompletionSource<bool>? _waiter;

public Task WaitAsync()
{
    lock (_lock)
    {
        _waiter ??= new TaskCompletionSource<bool>();
        return _waiter.Task;
    }
}

public void Signal()
{
    lock (_lock)
    {
        _waiter?.SetResult(true);
        _waiter = null;
    }
}

这段代码的问题在于:

  • SetResult(true) 可能同步执行 continuation;
  • continuation 又可能回过头来访问同一个对象;
  • 于是锁竞争、重入、阻塞链条都会变复杂。

更稳妥的思路通常是:

  • 要么使用 RunContinuationsAsynchronously
  • 要么先把待完成对象拿到锁外,再执行完成动作。

例如:

private readonly object _lock = new();
private TaskCompletionSource<bool>? _waiter;

public Task WaitAsync()
{
    lock (_lock)
    {
        _waiter ??= new TaskCompletionSource<bool>(
            TaskCreationOptions.RunContinuationsAsynchronously);
        return _waiter.Task;
    }
}

public void Signal()
{
    TaskCompletionSource<bool>? waiter;

    lock (_lock)
    {
        waiter = _waiter;
        _waiter = null;
    }

    waiter?.TrySetResult(true);
}

这样会安全很多。

常见坑 1:把它当成“异步工作启动器”

错误方向通常长这样:

public Task DoWorkAsync()
{
    var tcs = new TaskCompletionSource<bool>();
    return tcs.Task;
}

如果后面根本没有任何地方去完成这个 tcs,那这个任务就会永远挂着。

所以使用 TaskCompletionSource<T> 时,一定要先问自己:

  • 谁来完成它?
  • 正常路径在哪里完成?
  • 异常路径在哪里完成?
  • 取消路径在哪里完成?
  • 是否存在永远不完成的分支?

如果这些问题答不上来,通常说明这里还不该上 TaskCompletionSource<T>

常见坑 2:忘记处理异常路径

很多包装代码只写了成功回调:

public Task<string> GetDataAsync()
{
    var tcs = new TaskCompletionSource<string>();

    BeginOperation(result =>
    {
        tcs.TrySetResult(result);
    });

    return tcs.Task;
}

如果底层 API 还有失败回调、错误事件或断开通知,而你没接进去,结果往往是:

  • 上层代码一直等;
  • 任务永远不完成;
  • 问题很难排查。

因此包装时必须把可能的结束路径补全:

  • 成功;
  • 失败;
  • 取消;
  • 超时;
  • 资源释放或连接关闭。

常见坑 3:事件桥接后忘记解绑

这在 UI、消息总线、长连接、订阅模型里非常常见。

如果你写了:

client.MessageReceived += handler;

但完成后没有:

client.MessageReceived -= handler;

后果可能包括:

  • 同一个等待器被重复触发;
  • 旧对象迟迟不能回收;
  • 多次调用方法后,订阅越来越多;
  • 某次消息错误地完成了别的请求。

所以事件桥接里,“解绑”不是锦上添花,而是正确性的一部分。

常见坑 4:错误理解“取消”

很多人会把这两件事混成一件事:

  • tcs.TrySetCanceled()
  • 真正取消底层操作。

实际上,前者只是告诉等待方:

  • 这个 Task 以取消语义结束了。

但底层操作如果没有感知 CancellationToken,它依然可能继续运行。

所以当你用 TaskCompletionSource<T> 做取消包装时,要明确自己做的是哪一层:

  • 只是取消等待;
  • 还是连底层工作一起取消。

如果只是前者,最好在注释或方法命名上把语义写清楚,避免误导调用方。

常见坑 5:在高并发下使用 Set* 导致额外异常

只要存在“谁先完成都行”的竞争关系,就不要轻易写:

tcs.SetResult(value);

因为只要别的路径先完成了,这里就会抛 InvalidOperationException

更通用、更稳妥的模式通常是:

if (tcs.TrySetResult(value))
{
    // 只有真正赢得完成权时,才做一次性的收尾逻辑
}

这样异常噪音更少,也更方便在竞态场景下做清理。

它和 async/await 是什么关系?

可以这样理解:

  • async/await 负责把异步流程写得像同步代码;
  • TaskCompletionSource<T> 负责把“原本不是 Task 的完成信号”变成 Task

两者不是替代关系,而是协作关系。

很多时候,真正的完整写法是:

public async Task<string> ReceiveWithTimeoutAsync(CancellationToken cancellationToken)
{
    var message = await WaitNextMessageAsync(_client, cancellationToken);
    return message.Trim();
}

其中:

  • WaitNextMessageAsync 内部靠 TaskCompletionSource<string> 桥接事件;
  • 外层业务方法继续用 async/await 组织流程。

所以更准确地说:

TaskCompletionSource<T> 是给 async/await 提供“可等待对象来源”的底层工具之一。

它和 ValueTaskSource 有什么关系?

如果你已经看到 ValueTaskIValueTaskSource 那一层,会发现两者有一点相似:

  • 都涉及“手动控制异步完成”;
  • 都不是直接执行工作;
  • 都是异步基础设施的一部分。

但定位不一样:

  • TaskCompletionSource<T>:给你一个手动完成的 Task<T>,易用、通用;
  • IValueTaskSource / ManualResetValueTaskSourceCore<T>:给高性能组件做更底层、更可复用的异步承载,复杂很多。

所以在绝大多数业务和普通框架代码里:

  • 能用 TaskCompletionSource<T> 解决的问题,通常没必要上 ValueTaskSource

什么场景特别适合用它?

如果你遇到下面这些问题,基本都可以优先想到 TaskCompletionSource<T>

  • 把回调风格 API 包成 Task
  • 把事件模型转成 await
  • 等待某个外部信号;
  • 把多条竞争路径合并成一个等待点;
  • 自己实现一个异步协调原语;
  • 给旧接口补上超时、取消、组合等待能力。

反过来说,如果你的需求只是:

  • 跑一段 CPU 计算;
  • 把同步代码临时丢到后台;

那就不该优先想到 TaskCompletionSource<T>,而更可能是:

  • Task.Run
  • 线程池;
  • 真正的异步 I/O API。

总结

TaskCompletionSource<T> 最重要的价值,不是“又一种创建任务的方法”,而是:

  • 让你手动控制 Task 的完成;
  • 让非 Task 世界的信号,能自然接入 async/await
  • 让成功、失败、取消都能被统一表示成标准异步语义。

实战里最该记住的几点是:

  • 它不是工作执行器,而是任务完成控制器;
  • 回调、事件、外部信号桥接,是它最核心的用途;
  • 只要存在竞争,优先用 TrySet*
  • 通用库和并发协调代码里,通常应该考虑 RunContinuationsAsynchronously
  • 超时或取消一个 TCS,不等于底层操作真的被取消了。

如果你已经理解了 Taskasync/await,那 TaskCompletionSource<T> 就是下一步必须掌握的关键拼图。

因为从这一层开始,你才真正拥有了:

“不是只会等待异步,而是能自己定义异步完成方式”的能力。