佳能镜头EOS系统EF协议逆向工程(三)解码算法

news2024/9/27 17:37:38

目录

数据结构

解码算法

解码效果


这篇文章基于上两篇文章继续,

佳能镜头EOS系统EF协议逆向工程(一)转接环电路设计_佳能ef自动对焦协议_岬淢箫声的博客-CSDN博客本文属于专栏——工业相机。此专栏首先提供我人工翻译的法语文档部分,然后通过STM32F103C8T6控制佳能镜头,最后协同上位机或者NVIDIA Xavier实现自动对焦。还有一个用处不大的River文档,它知道如何让相机和镜头通信,也许对当前的摄影实践几乎没有帮助。尽管如此,一些应用程序可能需要独立订购物镜的主要功能。撇开工业世界及其特殊机器不谈,在另一个品牌的图像采集系统上安装佳能光学元件是不可能的,而且是有罪的,无论是出于经济原因还是纯粹的技术原因,获得的组合提供了其他不可用的功能。https://blog.csdn.net/caoshiying/article/details/127609884?spm=1001.2014.3001.5502佳能镜头EOS系统EF协议逆向工程(二)逻辑分析仪测试_岬淢箫声的博客-CSDN博客本章描述了用于解密EF协议函数的思想和分析,如果其读数是可选的,则所获得结果的摘要将在专用章节中进行汇编,这仍然是正确理解EF协议函数的必要来源。随着函数的测试和逐步解码,它们的描述和列表中使用的注释将在过程中变得越来越精确,因为以前的分析或推理错误不会被强制纠正。所使用的逻辑分析仪是一个小型号的低成本8输入TTL USBEE AX Pro,所使用的软件是制造商的标准套件,可免费下载。软件不允许编辑或删除部分结果,某些读数或时钟意外激活,因为通电会导致SPI字解码不同步。https://blog.csdn.net/caoshiying/article/details/129057004?spm=1001.2014.3001.5502

数据结构

逻辑分析仪没有特别要求,某宝上的大部分逻辑分析仪可以用。数据格式要求很简单,举便如下:

; CSV, generated by libsigrok4DSL 0.2.0 on Fri Jul 29 10:17:48 2022
; Channels (3/16)
; Sample rate: 10 MHz
; Sample count: 50.896 M Samples
Time(s), CLK, DLC, DCL
0,1,1,1
0.0498275,1,1,0
0.0498285,1,1,1
0.0498697,1,1,0
0.049876,1,1,1
0.0499793,1,1,0
0.0499986,0,1,0
0.050005,1,1,0

分号开头表示注释,第一列是时间,这个时间是相对开始捕获的时间,用单词elapse表示列标题更合适。这款逻辑分析仪软件导出数据的表头就是这么写的,无所谓了。第二列是CLK信号,CLK是时钟的简写,搞硬件的同学是不是很熟悉呢?第三列是DLC信号。DLC是Data Lens to Camera的首字母缩写。第四列是DCL,DCL是Data Camera to Lens首字母缩写。所有信号用1表示高电平,0表示低电平。佳能相机的电平为3.3V。

我使用Qt写的解码工具,CMake工程代码如下:

cmake_minimum_required(VERSION 3.20)
project(TAMRON VERSION 1.0)

find_package(Qt5 COMPONENTS Widgets REQUIRED PATHS $ENV{Qt515_DIR})
link_libraries(Qt5::Widgets)
add_link_options(/SUBSYSTEM:CONSOLE)
file(GLOB TAMRON_SRCS *.cpp *.h *.ui *.qrc *.rc)
add_executable(${PROJECT_NAME} ${TAMRON_SRCS})
target_compile_definitions(${PROJECT_NAME} PRIVATE $<IF:$<CONFIG:DEBUG>,CWDEBUG,CWNDEBUG>)

Qt515_DIR环境变量是必须的。

解码算法

算法是关键是解决ACK干扰。核心思路是寻找U型特征的连续信号。源代码只有一个main.cpp,代码如下:

#include <QApplication>
#include <QMainWindow>
#include <QFileDialog>
#include <QFile>
#include <QMessageBox>
#include <QTextStream>
#include <QDebug>
#include <QFileInfo>
#include <QMetaEnum>

typedef struct _spi_signal_t
{
    uint32_t line_no;//行号
    double elapse;//消耗的时间,6位小数,单位为秒,存储化为us
    bool clk;//时钟是否高位
    bool dcl;//主机信号是否高位
    bool dlc;//从机信号是否高位
} spi_signal_t;

typedef struct _spi_data_t
{
    uint32_t order;//序号
    uint32_t start_line;//信号起始行号
    uint32_t end_line;//信号结束行号
    uint8_t dcl;//主机命令
    uint8_t dlc;//从机返回
    int frame_no;//字节所属帧号
    bool bad;//是否发生丢失bit的情况
    double elapse;//消耗的时间
    double frequency;//实际通讯频率,存储为KHz
} spi_data_t;

//信号数据中的U型结构,全部为引用指针,不管理内存
typedef struct _u_shape_t
{
    spi_signal_t *high_left;//U槽左端
    spi_signal_t *low_left;//U槽底左边
    spi_signal_t *low_right;//U槽底右边
    spi_signal_t *high_right;//U槽右端
    spi_signal_t *forward;//遍历前进到达的位置
    bool ok;//此U型结构是否能用
    int index_cursor;//信号列表遍历时当前的位置
} u_shape_t;

typedef QSharedPointer<spi_signal_t> pspi_signal_t;
typedef QSharedPointer<spi_data_t> pspi_data_t;
typedef QSharedPointer<u_shape_t> pu_shape_t;
#define ZERR_CAPTION u8"系统错误"
#define ZOK_CAPTION u8"系统提示"

//CSV文件文件转换为信号数据
QList<pspi_signal_t> resolve_csv(const QString &zcsv_path);
//信号数据转换为真实的主机与从机之间交换的信息
QList<pspi_data_t> resolve_signal(const QList<pspi_signal_t> &ps);
//保存信息
void save_spi_data(const QString &zcsv_path, const QList<pspi_data_t> &ds);
//寻找信号中的U形,返回下一个U形右端索引+1,返回值index_cursor与istart_cursor相等表示结束,ok为false表示出错
pu_shape_t find_u_shape(const QList<pspi_signal_t> &ps, int istart_cursor, int ilen);
//保存程序日志
void save_log(QtMsgType type, const QMessageLogContext &cxt, const QString &zlog);
//解析帧
void resolve_frame(QList<pspi_data_t> &ds);

//入口函数
int main(int argc, char **argv)
{
    qInstallMessageHandler(save_log);
    QApplication app(argc, argv);

    auto zcsv_path = QFileDialog::getOpenFileName(
                         nullptr,
                         u8"选择一个CSV文件",
                         "D:/tamron"
                     );
    if (zcsv_path.isEmpty())
        return 1;
    auto ps = resolve_csv(zcsv_path);
    if (ps.isEmpty())
        return 2;
    auto ds = resolve_signal(ps);
    if (ds.isEmpty())
        return 3;
    resolve_frame(ds);
    save_spi_data(zcsv_path, ds);
    return 0;
}

//CSV文件文件转换为信号数据
QList<pspi_signal_t> resolve_csv(const QString &zcsv_path)
{
    QFile f(zcsv_path);
    QList<pspi_signal_t> ps;
    if (!f.open(QFile::ReadOnly | QFile::Text))
    {
        QMessageBox::warning(nullptr, ZERR_CAPTION, f.errorString());
        return ps;
    }
    QTextStream ts(&f);
    ts.skipWhiteSpace();
    //DSView导出数据有5行注释和1行标头,应当跳过。
    for (short i = 0; i < 4; i++)
    {
        auto zline = ts.readLine();
        if (ts.atEnd() || !zline.startsWith(";"))
        {
            QMessageBox::warning(nullptr, ZERR_CAPTION, u8"无效的CSV文件。");
            break;
        }
    }
    ts.readLine();
    if (ts.atEnd())
        return ps;
    uint32_t iline = 6;
    while (!ts.atEnd())
    {
        QString zline = ts.readLine();
        if (zline.isEmpty())
            continue;
        QStringList cols = zline.split(u8",");
        if (cols.length() < 4)
        {
            QMessageBox::critical(nullptr, ZERR_CAPTION, QString(u8"第%1行无效数据").arg(iline));
            return ps;
        }
        spi_signal_t r;
        r.line_no = iline++;
        if (iline >= UINT32_MAX)
        {
            QMessageBox::warning(nullptr,
                                 ZERR_CAPTION,
                                 u8"input data amount exceed system capacity."
                                );
            break;
        }
        r.elapse = cols[0].toDouble() * 1000000;
        r.clk = cols[1].toInt() != 0;
        r.dlc = cols[2].toInt() != 0;
        r.dcl = cols[3].toInt() != 0;
        pspi_signal_t p = QSharedPointer<spi_signal_t>::create(r);
        ps.append(p);
    }
    qInfo() << u8"合计" << ps.length() << u8"个信号\n";
    return ps;
}

//寻找信号中的U形,返回下一个U形右端索引+1,返回值index_cursor与istart_cursor相等表示结束,ok为false表示出错
pu_shape_t find_u_shape(const QList<pspi_signal_t> &ps, int istart_cursor, int ilen)
{
    pu_shape_t pu(new u_shape_t
    {
        0,
    });
    u_shape_t *u = pu.data();
    u->index_cursor = istart_cursor;
    //1.寻找U槽
    //1.1.寻找下降沿,定位U槽左端
    while (u->index_cursor < ilen && ps[u->index_cursor]->clk)
        u->index_cursor++;
    u->forward = ps[u->index_cursor].data();
    if (u->index_cursor - 1 < 0)
    {
        u->ok = false;
        u->index_cursor++;
        return pu;
    }
    if (u->index_cursor + 1 >= ilen)
    {
        qWarning() << "data end on finding u->high_left and u->low_left";
        u->ok = false;
        return pu;
    }
    u->high_left = ps[u->index_cursor - 1].data();
    u->low_left = ps[u->index_cursor].data();
    u->forward = u->low_left;
#ifdef DEBUG
    qInfo() << "u begin: " << u->high_left->line_no;
#endif
    u->index_cursor++;
    if (u->index_cursor >= ilen)
    {
        qWarning() << "data end on finding u->low_right";
        u->ok = false;
        return pu;
    }
    //1.2.寻找上升沿,定位U槽底有多长
    while (u->index_cursor < ilen && !ps[u->index_cursor]->clk)
        u->index_cursor++;
    u->low_right = ps[u->index_cursor - 1].data();
    u->forward = u->low_right;
    if (u->index_cursor >= ilen)
    {
        qWarning() << "data end on finding u->high_right";
        u->ok = false;
        return pu;
    }
    //1.3.寻找下降沿,定位U槽右端
    //while (u->index_cursor < ilen && ps[u->index_cursor]->clk)
    //    u->index_cursor++;
    u->high_right = ps[u->index_cursor].data();
    u->forward = u->high_right;
    //1.4.如果U槽的时间跨度大于28us则不是数据传输,暂不处理佳能中的拱门
    double fuspan1 = u->high_right->elapse - u->low_right->elapse;
    double fuspan2 = u->low_right->elapse - u->low_left->elapse;
    if (fuspan1 > 28 || fuspan1 < 0 || fuspan2 > 28 || fuspan2 < 0)
    {
        qWarning() << "elapse time out: " << fuspan2 << ", " << fuspan1;
        u->ok = false;
    }
    else
        u->ok = true;
#ifdef DEBUG
    if (u->ok)
        qInfo() << "u end: " << u->forward->line_no;
#endif
    return pu;
}

//信号数据转换为真实的主机与从机之间交换的信息
QList<pspi_data_t> resolve_signal(const QList<pspi_signal_t> &ps)
{
    int icursor = 0;//遍历ps列表的索引
    int ilen = ps.length(); //总长度减2,双指针遍历
    int iorder = 1;
    QList<pspi_data_t> ds;
    if (ilen < 17)
    {
        QMessageBox::critical(nullptr, ZERR_CAPTION, u8"数据量太少。");
        return ds;
    }
    while (icursor < ilen - 17)
    {
        int ifor_cursor = icursor;
        //发现问题则回到U槽右端
        int iu_cursor = 0;
        pspi_data_t byte(new spi_data_t);
        pu_shape_t pu;
        byte->order = iorder;
        byte->frame_no = 0;
        byte->dcl = 0;
        byte->dlc = 0;
        byte->bad = false;
        byte->start_line = 0;
        byte->end_line = 0;
        byte->elapse = 0;
        byte->frequency = 0;
        qInfo() << "byte " << iorder << " begin: " << ps[icursor]->line_no;
        for (short i = 0; i < 8; i++)
        {
            pu = find_u_shape(ps, ifor_cursor, ilen);
            ifor_cursor = pu->index_cursor;
            if (iu_cursor == 0)
                iu_cursor = ifor_cursor;
            if (pu->ok)
            {
#ifdef DEBUG
                qInfo() << "byte " << iorder << " bit " << i << " at " << pu->low_left->line_no;
#endif
                if (byte->start_line == 0)
                {
                    byte->start_line = pu->low_left->line_no;
                    byte->elapse = pu->low_left->elapse;
                }
                if (pu->high_right->dcl)
                    byte->dcl |= 1 << (7 - i);
                if (pu->high_right->dlc)
                    byte->dlc |= 1 << (7 - i);
            }
            else
            {
                byte->bad = true;
                break;
            }
        }
        if (byte->bad)
        {
            icursor = iu_cursor;
            qWarning() << "byte " << iorder << " break: " << pu->forward->line_no;
        }
        else
        {
            icursor = ifor_cursor;
            byte->end_line = pu->forward->line_no;
            byte->elapse =  pu->forward->elapse - byte->elapse;
            byte->frequency = 8000 / byte->elapse;
            qInfo() << "byte " << iorder << " end: " << pu->high_right->line_no;
            icursor++;
            //跳过第9个下降沿
            while (ps[icursor]->clk && icursor < ilen)
                icursor++;
            while (!ps[icursor]->clk && icursor < ilen)
                icursor++;
            iorder++;
            //80KHz下每个bit用时约为13us,1个byte不超过120us
            if (byte->elapse > 120)
                byte->bad = true;
            ds.append(byte);
        }
    }
    return ds;
}

//保存信息
void save_spi_data(const QString &zcsv_path, const QList<pspi_data_t> &ds)
{
    //避免多次执行QChar的构造函数和析构函数
    static const QChar cfill('0');
    QFileInfo fi(zcsv_path);
    QString zresolve_path = fi.dir().absoluteFilePath(fi.baseName() + "resolve.csv");
    QFile f(zresolve_path);
    if (!f.open(QFile::ReadWrite | QFile::Text | QFile::Truncate))
    {
        QMessageBox::warning(nullptr, ZERR_CAPTION, f.errorString());
        return;
    }
    QTextStream ts(&f);
    ts << u8" command,sequence,   start,     end,   dcl,   dlc,elapse(us),frequency(KHz),bad\n";
    for (const pspi_data_t &d : ds)
    {
        // char c = ' ';
        // if (d->dlc >= 0x20 && d->dlc <= 0x7E && d->dcl == 0)
        //     c = (char)d->dlc;
        // if (c == ',')
        //     c = ' ';
        QString zline = QString("%1,%2,%8,%9,    %3,    %4,%5,%6,%7\n")
                        .arg(d->frame_no, 8)
                        .arg(d->order, 8)
                        .arg(d->dcl, 2, 16, cfill)
                        .arg(d->dlc, 2, 16, cfill)
                        .arg(d->elapse, 10, 'f', 0)
                        .arg(d->frequency, 14, 'f', 0)
                        .arg(d->bad, 3)
                        .arg(d->start_line, 8)
                        .arg(d->end_line, 8);
        // .arg(c, 4);
        zline = zline.toUpper();
        ts << zline;
    }
    ts.flush();
    f.close();
    QMessageBox::information(nullptr, ZOK_CAPTION, u8"数据转换完成。");
    qInfo() << u8"数据转换完成。";
}

void save_log(QtMsgType type, const QMessageLogContext &cxt, const QString &zlog)
{
    static QFile f;
    static QTextStream output(stdout);
    static QTextStream ts;
    if (!f.isOpen())
    {
        QDir d(QApplication::applicationDirPath());
        f.setFileName(d.absoluteFilePath("main.log"));
        if (!f.open(QFile::Append | QFile::Text | QFile::ReadWrite))
        {
            QMessageBox::warning(nullptr, ZERR_CAPTION, u8"无法打开日志文件。");
            return;
        }
        ts.setDevice(&f);
        ts.setAutoDetectUnicode(true);
    }
    QString ztype;
    switch (type)
    {
    case QtDebugMsg:
        ztype = u8"DEBG";
        break;
    case QtWarningMsg:
        ztype = u8"WARN";
        break;
    case QtCriticalMsg:
        ztype = u8"CRIT";
        break;
    case QtFatalMsg:
        ztype = u8"FATA";
        break;
    case QtInfoMsg:
        ztype = u8"INFO";
        break;
    default:
        Q_ASSERT(false);
        break;
    }
    ts << ztype << ", " << zlog.toUtf8() << "\n";
    ts.flush();
    output << ztype << " " << zlog.toUtf8() << "\n";
    output.flush();
}

//解析帧
void resolve_frame(QList<pspi_data_t> &ds)
{
    if (ds.length() < 2)
        return;
    int ilen = ds.length();
    int iframe = 0;
    for (int i = 0; i < ilen; i++)
    {
        if (ds[i]->dcl > 0)
            iframe++;
        ds[i]->frame_no = iframe;
    }
}

解码效果

如下图所示

 解码结果中通信频率、0A与AA应答信号、0x06与0x05的转动信号与实际匹配,说明解码成功。如果想要更多的通信规律,请与我私聊。下一篇讲解常见的指令。

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

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

相关文章

Python解题 - CSDN周赛第29期 - 争抢糖豆

本期问哥是志在必得&#xff0c;这本算法书我已经觊觎许久&#xff0c;而之前两次因为种种原因未能如愿。因此&#xff0c;问哥这几天花了不少时间&#xff0c;把所有之前在每日一练做过的题目重新梳理了一遍。苦心人&#xff0c;天不负&#xff0c;感谢官方大大&#xff01; 第…

ChatGPT 人工智能革命从实验室走入公众生活

11 月底&#xff0c;人工智能研究实验室OpenAI 发布了 ChatGPT 聊天机器人首个测试版本&#xff0c;这是一款基于人工智能的新型聊天机器人&#xff0c;可以与人类进行对话&#xff0c;经过测试后&#xff0c;新款机器人便踏上了社交网站之旅&#xff0c;尤其是在推特平台上&am…

机器学习+西瓜书笔记第2章【贝叶斯分类器】

机器学习笔记第2章【贝叶斯分类器】一、贝叶斯决策论1.相关知识补充2.生成模型与判别模型贝叶斯公式&#xff1a; 实际上&#xff0c;分母为全概率公式&#xff0c;分子为联合概率。在机器学习中&#xff0c;更常见的形式为 贝叶斯公式的作用在于将P(B|A)的估计转化为估计P(A…

做一个短视频应用,如何选择服务器配置?

大家好我是明杰&#xff0c;最近听说了“两微一抖”这个词。很容易联想到,“两微”指的是微信和微博,“一抖”指的是抖音,它描述的是今年开始互联网行业呈现的一种新的变化。抖音奇迹般地杀出重围,与微博、微信一起造就了流量市场“三权分立”的现象。抖音能与微信、微博齐名,与…

vue全家桶(四)前端工程化

vue全家桶&#xff08;四&#xff09;前端工程化1.模块化的相关规范1.1模块化概述1.2模块化的分类A.浏览器端的模块化B.服务器端的模块化C.ES6模块化1.2.1 Node.js中通过bable体验ES6模块化1.2.2 ES6模块化的基本语法1.2.2.1 默认导出与默认导入1.2.2.2 按需导出与按需导入1.2.…

vue - vue项目中解决 IOS + H5 滑动边界橡皮筋弹性效果

问题: 最近遇到一个问题&#xff0c;我们在企业微信中的 H5 项目中需要用到table表格&#xff08;支持懒加载 上划加载数据&#xff09;。但是他们在锁头、锁列的情况下&#xff0c;依旧会出现边界橡皮筋效果。就会显示的很奇怪。 什么是ios橡皮筋效果&#xff1a; 我们知道元素…

华为MateBook E Go电脑使用U盘怎么安装Win10系统?

华为MateBook E Go电脑使用U盘怎么安装Win10系统&#xff1f;有用户购买这款电脑之后&#xff0c;发现系统默认安装的电脑系统是Win11版本的。但是自己不习惯使用这个系统&#xff0c;所以想要去将系统重新安装到Win10来使用。那么要怎么去进行系统重装呢&#xff1f;一起来看看…

微服务门神-Gateway与Sentinel的集成

目录 引言 概述 集成Sentinel 限流维度 网关集成 Route维度 API分组 精准匹配 前缀匹配 正则匹配 自定义限流返回格式 转视频版 引言 书接上篇&#xff1a;微服务门神-Gateway过滤器Filter&#xff0c;讲完了解Gateway过滤器之后&#xff0c;接下来看下Gateway与…

免费常用IP归属地查询API

引言 因毕设需要&#xff0c;需要使用到根据IP地址查询归属地 经过百度查询&#xff0c;发现如下几个api可以尝试&#xff0c;本人决定使用最后一个api 免费常用IP归属地查询API ip-api.com 可切换显示语言 http://ip-api.com/json/117.136.12.79?langzh-CN {"status…

wodat:一款针对Windows Oracle数据库的渗透测试工具

关于wodat wodat是一款功能强大的针对Windows Oracle数据库的渗透测试工具&#xff0c;该工具基于C# .Net Framework开发&#xff0c;能够帮助广大研究人员对Windows平台下的Oracle数据库执行按摩全渗透测试任务。 注意&#xff1a;请在被授权执行安全测试的情况下使用该工具…

[Apache Hudi] 流转批的场景实践

文章目录1.EventTime计算原理2.案例使用2.1 Maven pom 依赖2.2 设置EventTime2.3 Flink API2.4 Flink SQL2.5 读取EventTime在某些业务场景下&#xff0c;我们需要一个标志来衡量hudi数据写入的进度&#xff0c;比如&#xff1a;Flink 实时向 Hudi 表写入数据&#xff0c;然后使…

vue2版本《后台管理模式》(中)

文章目录前言一、创建一个文件夹 utils 里面新增一个 setToken.js 文件(设置token验证&#xff09;二 、创建一个api文件夹 新增 service.js &#xff08;axios拦截器&#xff09;三、在api文件夹里 新增一个 api.js 来接收数据&#xff08;把api封装哪里需要某项数据直接引入就…

运维服务商低成本提升服务质量解决方案

在信息化高速发展的今天&#xff0c;网络建设的重要性不言而喻&#xff0c;更多客户选择将运维服务外包或托管给运维服务商&#xff0c;市场需求愈大竞争压力愈大&#xff0c;想要脱颖而出势必要优化自身提高服务质量&#xff0c;最好是低成本、大提升&#xff0c;nVisual助力渠…

饕餮 NFT 作品集来袭!

饕餮 NFT 作品集包含 Chili Game 创作的体验《饕餮》第一章中的角色。可以在 The Sandbox 农历新年活动期间&#xff08;01/18/23 至 02/28/23&#xff09;体验。 饕餮的故事植根于中国古代神话&#xff0c;主要灵感来自《山海经》&#xff0c;一个关于捉妖人「青蛙侠」的故事。…

ASEMI中低压MOS管18N20参数,18N20封装,18N20尺寸

编辑-Z ASEMI中低压MOS管18N20参数&#xff1a; 型号&#xff1a;18N20 漏极-源极电压&#xff08;VDS&#xff09;&#xff1a;200V 栅源电压&#xff08;VGS&#xff09;&#xff1a;30V 漏极电流&#xff08;ID&#xff09;&#xff1a;18A 功耗&#xff08;PD&#x…

神经网络基础部件-卷积层详解

前言 在全连接层构成的多层感知机网络中&#xff0c;我们要通过将图像数据展平成一维向量来送入模型&#xff0c;但这会忽略了每个图像的空间结构信息。理想的策略应该是要利用相近像素之间的相互关联性&#xff0c;将图像数据二维矩阵送给模型中学习。 卷积神经网络(convolu…

教育舆情监测方案有哪些,TOOM讲解教育舆情的应对与处理?

教育舆情方案是针对教育领域的舆情事件或问题而制定的应对方案。其主要目的是通过有效的信息收集、分析、处理和传播&#xff0c;帮助教育机构或相关组织及时掌握和应对公众舆论的发展趋势&#xff0c;维护良好的舆情形象和声誉&#xff0c;教育舆情监测方案有哪些&#xff0c;…

黑马Java后端项目实战--在线聊天交友

【课程简介】 越来越多的系统都有消息推送的功能&#xff0c;如聊天室、邮件推送、系统消息推送等&#xff1b; 要实现消息推送就需要服务端在数据有变化时主动推送消息给客户端&#xff0c;本次课程将带大家使用websocket实现消息推送。 【主讲内容】 1.方法&#xff1a;如…

ANR系列(二)——ANR监听方案之WatchDog

前言 ANR的监控在Android6.0之前可以通过监听文件data/anr/trace读取trace信息来分析&#xff0c;但从6.0之后就被禁止了。随着Android的发展&#xff0c;手机里的ANR越来越多&#xff0c;对ANR的监控方案也就五花八门。 WatchDog方案 WatchDog是个开源的框架&#xff0c;是…

大漠插件最新中文易语言模块7.2302

模块名称:大漠插件中文模块最新通用7.2302模块简介:大漠插件中文模块最新通用7.2302模块特色:原翻译:花老板完善命令备注:易生易世本人花费一个月时间才将命令完善了插件的备注说明.且用且珍惜去掉了大漠插件定制版类.因为没用.模块特色:什么是中文模块?大漠插件模块是由大漠类…