JCEF学习

JCEF重要概念

CEF

CEF,全称Chromium Embedded Framework ,它是基于Google Chromium的开源项目,它的目标是能够向第三方程序添加WEB浏览器功能,以及可以使用HTML、CSS和JS渲染界面。

CEF框架是由Marshall Greenblatt 在 2008 年创立的,由C/C++编写而成

而JCEF则是CEF的java接口,可以使用java调用CEF的接口。

JCEF的官方网站(chromiumembedded / java-cef / wiki / Home — Bitbucket)

其他的帮助文档:JCEF帮助文档

https://github.com/fanfeilong/cefutil/blob/master/doc/CEF%20General%20Usage-zh-cn.md

POI文档:POI中文帮助文档


Render进程

Cef3采用了多进程架构,Browser进程为主进程,负责窗口管理、界面绘制、网络交互等;Render进程负责页面渲染、V8引擎、Dom节点等。默认的进程模型中,会为每一个标签页创建一个新的Render进程。

Render进程

CEF 的渲染进程(Render Process)负责处理浏览器窗口内的内容渲染,包括 HTML、JavaScript、CSS 等。

例如,当用户在浏览器中访问一个网页时,CEF 会创建一个渲染进程来解析 HTML 代码,并根据 CSS 和 JavaScript 渲染出网页的内容。

frame render 进程

如果网页中包含了 frame 标签,CEF 会再创建一个 frame render 进程来渲染 frame 内的内容。

CEF 的 frame render 进程可以帮助开发者在应用内嵌入多个浏览器窗口,并且可以让每个浏览器窗口独立地加载和渲染网页内容,从而提升应用的性能和用户体验。


Browser进程

Browser被定义为主进程,负责窗口管理,界面绘制和网络交互Blink的渲染和Js的执行被放在一个独立的Render 进程中;除此之外,Render进程还负责Js Binding和对Dom节点的访问 默认的进程模型中,会为每个标签页创建一个新的Render进程

Browser和Render进程可以通过发送异步消息进行双向通信

一个从Browser进程发送到Render进程的消息将会在CefRenderProcessHandler::OnProcessMessageReceived()方法里被接收。一个从Render进程发送到Browser进程的消息将会在CefClient::OnProcessMessageReceived()方法里被接收。

JavaScript被集成在Render进程,但是需要频繁和Browser进程交互。 JavaScript API应该被设计成可使用闭包异步执行。


JCEF入门案例

初始化JCEF:在应用程序启动时,必须先初始化JCEF

import org.cef.CefApp;
import org.cef.CefClient;
import org.cef.CefSettings;
import org.cef.OS;
import org.cef.browser.CefBrowser;
import org.cef.handler.CefAppHandlerAdapter;import javax.swing.*;
import java.awt.*;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;public class TestFrame extends JFrame {private static final long serialVersionUID = -7410082787754606408L;public static void main(String[] args) {new TestFrame();}public TestFrame() {//是否Linux系统boolean useOSR = OS.isLinux();//是否透明boolean isTransparent = false;//添加Handler,在CEFAPP状态为终止时退出程序CefApp.addAppHandler(new CefAppHandlerAdapter(null) {@Overridepublic void stateHasChanged(org.cef.CefApp.CefAppState state) {//CefApp.CefAppState TERMINATED:CefApp已终止,无法再使用。您现在可以安全地关闭应用程序if (state == CefApp.CefAppState.TERMINATED) System.exit(0);}});//--------------------------初始化JCEF---------------------------CefSettings settings = new CefSettings();settings.windowless_rendering_enabled = useOSR;//获取CefApp实例CefApp cefApp = CefApp.getInstance(settings);//创建客户端实例CefClient cefClient = cefApp.createClient();//-----------------------------创建浏览器实例-------------------------------//创建一个新的浏览器实例,并打开指定的URLCefBrowser cefBrowser = cefClient.createBrowser("http://hao.fly63.com/", useOSR, isTransparent);//将浏览器UI添加到窗口中getContentPane().add(cefBrowser.getUIComponent(), BorderLayout.CENTER);pack();setTitle("JCEF打开极简导航");setSize(800, 600);setVisible(true);//添加一个窗口关闭监听事件addWindowListener(new WindowAdapter() {@Overridepublic void windowClosing(WindowEvent e) {CefApp.getInstance().dispose();dispose();}});}
}

CefSettings

CefSettings结构体允许定义全局的CEF配置,经常用到的配置项如下

single_process 设置为true时,Browser和Renderer使用一个进程。

locale 此设置项将传递给Blink。如果此项为空,将使用默认值“en-US”。

log_file 此项设置的文件夹和文件名将用于输出debug日志。如果此项为空,默认的日志文件名为debug.log,位于应用程序所在的目录

log_severity 此项设置日志级别。只有此等级、或者比此等级高的日志的才会被记录。此项可以通过命令行参数“log-severity”配置,可以设置的值为“verbose”,“info”,“warning”,“error”,“error-report”,“disable”。

emote_debugging_port 此项可以设置1024-65535之间的值,用于在指定端口开启远程调试。例如,如果设置的值为8080,远程调试的URL为http://localhost:8080。CEF或者Chrome浏览器能够调试CEF


JCEF重要API

CefSettings

CefSettings定义了Cef的全局配置信息,比如指定单进程模式、指定渲染子进程路径、设置localstorage路径、设置日志等级、Cef资源文件路径。其中对于项目最重要的字段是single_process、multi_threaded_message_loop、windowless_rendering_enabled,分别用于指定单进程模式、多线程渲染模式、离屏渲染模式。
经常用到的配置项如下:

  1. single_process 设置为true时,Browser和Render使用一个进程。此项也可以通过命令行参数”single-process”配置。
  2. browser_subprocess_path 设置用于启动子进程单独执行的路径。
  3. cache_path 设置存放缓存数据的位置。如果此项为空,某些功能使用内存缓存,多数功能使用临时的磁盘缓存。
  4. locale 此设置项将传递给Blink。如果此项为空,将使用默认值”en-US”。在Linux平台下此项被忽略,使用环境变量中的值,解析的依次顺序为:LANGUAE,LC_ALL,LC_MESSAGES和LANG。此项也可以通过命令行参数”lang”配置。
  5. log_file 此项设置文件路径用于输出debug日志,如果此项为空,默认的日志文件名为debug.log,位于应用程序所在的目录。此项也可以通过命令参数”log-file”配置。
  6. log_severity 此项设置日志级别,只有当前等级、或者更高等级的日志才会被记录。此项可以通过命令行参数”log-severity”配置,可以设置的值为”verbose”,”info”,”warning”,”error”,”error-report”,”disable”。
  7. resources_dir_path 此项设置资源文件夹的位置。如果此项为空,Windows平台下cef.pak、Linux平台下devtools_resourcs.pak、Mac OS X下的app bundle Resources目录必须位于组件目录。此项也可以通过命令行参数”resource-dir-path”配置。
  8. locales_dir_path 此项设置locale文件夹位置。如果此项为空,locale文件夹必须位于组件目录,在Mac OS X平台下此项被忽略,pak文件从app bundle Resources目录。此项也可以通过命令行参数”locales-dir-path”配置。
  9. remote_debugging_port 此项可以设置1024-65535之间的值,用于在指定端口开启远程调试。此项也可以通过命令行参数”remote-debugging-port”配置。
  10. command_line_args_disabled 用于禁用Cef进程命令行参数功能,只能通过CefSettings方式进行设置。

1


CefApp

CefApp接口提供了不同进程的可定制回调函数,每一个进程对应一个CefApp接口

 CefApp:公开用于管理全局 CEF 上下文的静态方法

1、addAppHandler(CefAppHandler appHandler)

//将一个AppHandler分配给CefApp,必须在初始化CefApp之前调用此方法
public static void addAppHandler(CefAppHandler appHandler) throws java.lang.IllegalStateException

2、createClient()

//创建一个新的客户端实例并将其返回给调用者
public CefClient createClient()

3、getInstance(

//CefApp已启动并正在运行。至少创建了一个CefClient,并且消息循环正在运行。您现在可以使用JCEF的所有类和方法。
public static final CefApp.CefAppState INITIALIZED//CefApp正在进行初始化过程。请等待初始化完成
public static final CefApp.CefAppState INITIALIZING//(CefApp是新创建的,但尚未初始化。到目前为止,还没有CefClient和CefBrowser被创建
public static final CefApp.CefAppState NEW

 settings)

//获取CefApp类的实例
public static CefApp getInstance(CefSettings settings) throws java.lang.UnsatisfiedLinkError

4、dispose()

//调用此方法将关闭所有客户端实例以及每个客户端拥有的所有浏览器实例。此后,消息循环终止并且CEF关闭
public final void dispose()

CefApp.CefAppState 枚举常量

总共有6个

//CefApp已启动并正在运行。至少创建了一个CefClient,并且消息循环正在运行。您现在可以使用JCEF的所有类和方法
public static final CefApp.CefAppState INITIALIZED//CefApp已终止,无法再使用。您现在可以安全地关闭应用程序
public static final CefApp.CefAppState TERMINATED

CefClient

Client that owns a browser and renderer.:拥有浏览器和渲染器的客户端

每一个CefBrowser对象会对应一个CefClient接口,用于处理浏览器页面的各种回调信息,包括了Browser的生命周期,右键菜单,对话框,状态通知显示,下载事件,拖曳事件,焦点事件,键盘事件,离屏渲染事件等。

public CefBrowser createBrowser(java.lang.String url,boolean isOffscreenRendered,boolean isTransparent)//返回所有浏览器实例的列表
protected java.lang.Object[] getAllBrowser()

CefClient: 提供了一些获取Handler的方法。

  • CefDisplayHandler: 回调类,用来处理与页面状态相关的事件,如页面加载情况的变化,地址栏变化,标题变化等。
  • CefLifeSpanHandler: 回调类,主要用来处理与浏览器生命周期相关的事件,和浏览器对象的创建、销毁以及弹出框的管理。
  • CefLoadHandler: 回调类,主要用来处理浏览器页面加载状态的变化,如页面加载开始,完成,出错等。

CefBrowser

Interface representing a browser. (代表浏览器的界面)

//获取与此浏览器关联的客户端
CefClient getClient()//要求关闭浏览器。 close表示是否强制关闭
void close(boolean force)//返回唯一的浏览器标识符
int getIdentifier()//在此框架中执行一串JavaScript代码
void executeJavaScript(java.lang.String code,java.lang.String url,int line)//返回浏览器窗口的框架
CefFrame getFocusedFrame()//返回浏览器窗口的顶层框架
CefFrame getMainFrame()//如果窗口是弹出窗口,则为true
boolean isPopup()//重新加载当前页面
void reload()//如果document已加载到浏览器中,则为true
boolean hasDocument()

关于Frame的

//返回当前存在的Frame个数
int getFrameCount()//返回所有现有的框架的标识符
java.util.Vector<java.lang.Long> getFrameIdentifiers()//返回所有现有的框架的名称
java.util.Vector<java.lang.String> getFrameNames()-----------------------------------------//返回具有指定标识符的Frame;如果未找到,则返回NULL
CefFrame getFrame(long identifier)//返回具有指定名称的框架,如果找不到,则返回NULL。
CefFrame getFrame(java.lang.String name)

CefFrame

//返回此框架的名称。如果框架具有分配的名称(例如,通过iframe的“名称”属性设置),则将返回该值。
//否则,将基于框架父级层次结构构造一个唯一的名称。
java.lang.String getName()//返回此框架的父级;如果这是主要(顶级)框架,则返回NULL
CefFrame getParent()//如果此框架是顶级,则为True,否则为false
boolean isMain()//发出当前加载到该框架中的 URL
java.lang.String getURL()//在此框架中执行一串JavaScript代码
void executeJavaScript(java.lang.String code,java.lang.String url,int line)code -要执行的代码
url -可以找到相关脚本的网址
line -用于错误报告的基线号

CefLifeSpanHandler

CefBrowser对象的生命周期事件的回调接口。

//创建浏览器对象后会立马触发这个回调
void onAfterCreated(CefBrowser browser)//当浏览器对象即将销毁时会触发这个回调
void onBeforeClose(CefBrowser browser)//在浏览器收到关闭请求时调用
boolean doClose(CefBrowser browser)//当单击了网页中会弹出新窗口的链接时,会触发这个回调,要取消创建弹出窗口,请返回true
boolean onBeforePopup(CefBrowser browser,CefFrame frame,java.lang.String target_url,java.lang.String target_frame_name)

1


CefQueryCallback

Interface representing a query callback. (表示查询回调的接口

//通知关联的 JavaScript onFailure 回调查询失败
void failure(int error_code,java.lang.String error_message)error_code  -错误代码传递给JavaScript
error_message -错误消息传递给JavaScript//通知关联的JavaScript onSuccess回调查询已成功完成
void success(java.lang.String response)
response - 传递给JavaScript的响应

CefMessageRouter

//使用默认配置创建一个新路由器
public static final CefMessageRouter create()//创建具有指定配置的新路由器
public static final CefMessageRouter create(CefMessageRouter.CefMessageRouterConfig config)//添加一个新的查询处理程序
public abstract boolean addHandler(CefMessageRouterHandler handler,boolean first)
handler-要添加的处理程序
first -如果为true,则将处理程序添加为第一个处理程序,否则将添加为最后一个处理程序//删除现有的查询处理程序,成功删除则返回为True。
public abstract boolean removeHandler(CefMessageRouterHandler handler)//如果不再使用CefMessageRouter实例,则必须调用
public abstract void dispose()

1


CefMessageRouterHandler

onQuery

  • 返回如果为True,则处理查询;
  • 返回如果为false,则将查询传播到其他注册的处理程序(如果有)。

如果没有处理程序从此方法返回true,则查询将被自动取消,并向JavaScript onFailure回调传递错误代码-1

//当浏览器收到JavaScript查询时调用
boolean onQuery(CefBrowser browser,CefFrame frame,long queryId,java.lang.String request,boolean persistent,CefQueryCallback callback)browser -相应的浏览器
frame -生成事件的框架。实例仅在此方法的范围内有效
queryId -查询的唯一ID
persistent -如果查询是持久的,则为true
callback -用于异步继续或取消查询的对象

1

//(取消待处理的JavaScript查询时调用
void onQueryCanceled(CefBrowser browser,CefFrame frame,long queryId)browser -相应的浏览器
frame -生成事件的框架。实例仅在此方法的范围内有效
queryId -查询的唯一ID

CefMessageRouterHandlerAdapter用于接收消息路由器事件的抽象适配器类。此类中的方法为空。此类的存在是为了方便创建处理程序对象


JCEF应用举例

执行JavaScript代码

CEF使用的V8 JavaScript 引擎用于内部JavaScript实现,每一个frame都有JS上下文(context),为JS代码执行提供范围和安全。CEF暴露了很多JS特性可以和客户端程序进行交互。

CefBrowser:一个普通的浏览器页面(HTML)
CefFrame:每一个页面都由至少一个frame组成,最顶层的为mainframe
context:JS执行环境,每个frame都有自己独立的context,CEF中使用V8 JavaScriptEngine解析和执行JS代码

执行JavaScript代码

cefBrowser.executeJavaScript("document.getElementById('myElement').innerHTML = 'Hello, JCEF!';", "",0);

CEF有专门的调用js方法的函数:executeJavaScript,它是一个属于CefFrame类的方法。所以我们想要调用js的方法,只需要获取到页面的frame,然后调用executeJavaScript就可以了。

//在此框架中执行一串JavaScript代码
void executeJavaScript(java.lang.String code,java.lang.String url,int line)

参数一是要执行的js语句;

参数二是可以找到问题脚本(如果有的话)的URL。渲染器可能会请求这个URL来向开发人员显示错误的来源;

第三个参数是用于错误报告的基线编号。


发送消息给网页

发送消息给网页:

CefProcessMessage message = CefProcessMessage.create("myMessage");
message.getArgumentList().setString(0, "Hello, JCEF!"); cefBrowser.sendProcessMessage(CefProcessId.BROWSER, message);


处理网页发送的消息

当JavaScript代码中调用cefQuery并传递"getJavaData"时,Java代码会返回一段数据。

// 在JavaScript中调用Java方法并获取返回值
cefQuery({request: "getJavaData",onSuccess: function (response) {console.log("Data from Java: " + response);},onFailure: function (errorCode, errorMessage) {console.error("Java query failed: " + errorCode + " - " + errorMessage);},
});

render进程和browser进程之间是通两端的消息路由传递消息的,

JCEF使得在Java应用程序中嵌入Chromium引擎成为可能,并且通过使用CefMessageRouter,您可以在Java和JavaScript之间实现双向的交互。

将这个JCEFMessageRouterHandler注册到CefMessageRouter中。这样,您就可以在JavaScript中使用cefQuery来与Java进行交互了

//配置一个查询路由,html页面可使用 window.java({}) 和 window.javaCancel({}) 来调用此方法
CefMessageRouter.CefMessageRouterConfig cmrc = new CefMessageRouter.CefMessageRouterConfig("java", "javaCancel");
//创建查询路由
CefMessageRouter router = CefMessageRouter.create(cmrc);
router.addHandler(new JCEFMessageRouterHandler(), true);
cefClient.addMessageRouter(router);

创建了一个CefMessageRouterHandlerAdapter的子类,重写了onQuery方法来处理来自JavaScript的查询。

import org.cef.browser.CefBrowser;
import org.cef.browser.CefFrame;
import org.cef.callback.CefQueryCallback;
import org.cef.handler.CefMessageRouterHandlerAdapter;public class JCEFMessageRouterHandler extends CefMessageRouterHandlerAdapter {//重写了onQuery方法来处理来自JavaScript的查询@Overridepublic boolean onQuery(CefBrowser browser, CefFrame frame, long query_id, String request, boolean persistent,CefQueryCallback callback) {if (request.equals("getJavaData")) {String javaData = "This data is from Java";callback.success(javaData);return true;}return false;}
}

文件选择对话框

举例: 用js去打开文件选择对话框

    //打开文件选择器public static void openFileChooser(CefBrowser browser, final CefQueryCallback callback) {CefRunFileDialogCallback dialogCallBack = new CefRunFileDialogCallback() {@Overridepublic void onFileDialogDismissed(int selectedAcceptFilter, Vector<String> filePaths) {if (filePaths.size() == 0) {return;}File selectedFile = new File(filePaths.get(0));if (selectedFile != null) {String selectedPath = selectedFile.getAbsolutePath();System.out.println(selectedPath);callback.success(selectedPath);}}};browser.runFileDialog(CefDialogHandler.FileDialogMode.FILE_DIALOG_OPEN, "文件选择器", null, null, 0, dialogCallBack);}

弹出窗口处理

CefLifeSpanHandlerAdapter:弹出窗口创建一个处理的Handler

//onBeforePopup 在接口中  CefLifeSpanHandler
public boolean onBeforePopup(CefBrowser browser,CefFrame frame,java.lang.String target_url,java.lang.String target_frame_name)browser - 弹出请求的来源
frame - 弹出请求的来源。实例仅在此方法的范围内有效
target_url -如果请求中未指定任何内容,则可以为空
target_frame_name -如果请求中未指定任何内容,则可以为空返回:如果为True,则取消创建弹出窗口;如果为false,则继续进行。)

当我们点击target值为_blank的链接时,JCEF默认以弹出窗口的形式打开新页面。创建一个实现CefLifeSpanHandlerAdapter的类,重写onBeforePopup方法:根据url创建一个CefBrowser对象。


开发者工具

public class DevToolsDialog extends JDialog {/*** */private static final long serialVersionUID = 4115973232808017898L;private final CefBrowser devTools_;public DevToolsDialog(Frame owner, String title, CefBrowser browser) {this(owner, title, browser, null);}public DevToolsDialog(Frame owner, String title, CefBrowser browser, Point inspectAt) {super(owner, title, false);setLayout(new BorderLayout());setSize(1000, 800);setLocation(owner.getLocation().x + 20, owner.getLocation().y + 20);devTools_ = browser.getDevTools(inspectAt);add(devTools_.getUIComponent());addComponentListener(new ComponentAdapter() {@Overridepublic void componentHidden(ComponentEvent e) {dispose();}});}@Overridepublic void dispose() {devTools_.close(true);super.dispose();}
}

怎么打开devtools呢

DevToolsDialog devToolsDlg = new DevToolsDialog(CefManager.getInstance().getFrame(), "开发者工具-"+browser.getMainFrame().getName(), browser);
devToolsDlg.setVisible(true);

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

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

相关文章

第17节-高质量简历写作求职通关-投递反馈

&#xff08;点击即可收听&#xff09; 投递跟进和感谢信 如果对一家公司特别心仪&#xff0c;但是投递简历后一直得不到回复怎么办&#xff1f; 面试之后觉得自己没有表现好怎么办&#xff1f; 面试完几天了&#xff0c;依然没有得到回应怎么办&#xff1f; 这个时候你需要写一…

八种Flink任务告警方式

目录 一、Flink应用分析 1.1 Flink任务生命周期 1.2 Flink应用告警视角分析 二、监控告警方案说明 2.1 监控消息队中间件消费者偏移量 2.2 通过调度系统监控Flink任务运行状态 2.3 引入开源服务的SDK工具实现 2.4 调用FlinkRestApi实现任务监控告警 2.5 定时去查询目标…

无人机在三维空间中的转动问题

前提 这篇博客是对最近一个有关无人机拍摄图像项目中所学到的新知识的一个总结&#xff0c;比较杂乱&#xff0c;没有固定的写作顺序。 无人机坐标系旋转问题 上图是无人机坐标系&#xff0c;绕x轴是翻滚(Roll)&#xff0c;绕y轴是俯仰(Pitch)&#xff0c;绕z轴是偏航(Yaw)。…

力扣日记1.27-【回溯算法篇】131. 分割回文串

力扣日记&#xff1a;【回溯算法篇】131. 分割回文串 日期&#xff1a;2023.1.27 参考&#xff1a;代码随想录、力扣 131. 分割回文串 题目描述 难度&#xff1a;中等 给你一个字符串 s&#xff0c;请你将 s 分割成一些子串&#xff0c;使每个子串都是 回文串 。返回 s 所有可…

D. Epic Transformation(堆+贪心)

思路&#xff1a;我们删的策略是从次数多的数开始删&#xff0c;每次取两种不同的数&#xff0c;每种删去一个&#xff0c;然后放回堆中。 代码&#xff1a; void solve(){int n;cin >> n;map<int,int>mp;for(int i 1;i < n;i ){int x;cin >> x;mp[x] …

Java笔记 --- 四、异常

四、异常 Java.lang.Throwable Error Exception&#xff08;异常&#xff09; 异常的作用 异常的处理方式 JVM默认的处理方式 捕获异常&#xff08;自己处理&#xff09; try里面没有出现异常&#xff0c;就不会运行catch里面的代码 如果出现多个异常&#xff0c;需要多个c…

【归并排序】【图论】【动态规划】【 深度游戏搜索】1569将子数组重新排序得到同一个二叉搜索树的方案数

本文涉及知识点 动态规划汇总 图论 深度游戏搜索 归并排序 组合 LeetCoce1569将子数组重新排序得到同一个二叉搜索树的方案数 给你一个数组 nums 表示 1 到 n 的一个排列。我们按照元素在 nums 中的顺序依次插入一个初始为空的二叉搜索树&#xff08;BST&#xff09;。请你统…

精选6款前端动画特效分享(附在线演示)

分享6款好玩的前端动画特效 其中有CSS动画、canvas动画、js小游戏等等 下方效果图可能不是特别的生动 那么你可以点击在线预览进行查看相应的动画特效 同时也是可以下载该资源的 CSS日食与太阳碰撞动画 一款基于CSS实现的日食动画特效 碰撞物体会从右侧旋转向太阳靠近重合而后…

程序员成被裁最多的职业,互联网成围城,“转码”神话破灭?

随着互联网蓬勃发展&#xff0c;“转码”一直被视为找不到工作时的灵丹妙药。所谓转码&#xff0c;就是转行成为程序员。专业太偏&#xff1f;没关系&#xff0c;可以转码。失业了&#xff1f;没关系&#xff0c;可以转码。不知道该做什么工作&#xff1f;那就转码吧。程序员薪…

idea提交代码到git或svn上时,怎么忽略.class、.iml文件和文件夹等不必要的文件

第一种方法 在Setings–> Editor --> File Types -->Ignore files and folders中添加需要忽略的文件和文件夹&#xff1a; .idea 忽略 .idea 的文件或者文件夹 *.iml 忽略后缀为iml的文件 target 忽略target 文件或目录以及目录下的所有文件注…

Linux学习之文件系统与动静态库

目录 一&#xff0c;文件的管理 什么是磁盘&#xff1f; 磁盘的逻辑抽象结构 格式化 inode 挂载 软硬链接 二&#xff0c;动静态库 什么是动静态库&#xff1f; 1.站在库的制作者角度 静态库&#xff1a; 制作一个静态库 2.站在静态库使用者的角度 动态库 作为制…

80.网游逆向分析与插件开发-背包的获取-自动化助手显示物品数据1

内容参考于&#xff1a;易道云信息技术研究院VIP课 上一个内容&#xff1a;升级Notice类获得背包基址-CSDN博客 码云地址&#xff08;ui显示角色数据 分支&#xff09;&#xff1a;https://gitee.com/dye_your_fingers/sro_-ex.git 码云版本号&#xff1a;3be017de38c50653b…

操作日志应记录编辑的前后内容变化

总体思路是增加一个注解类&#xff0c;将注解加到要进行记录变化的Java类属性上却可。 上代码&#xff1a; 1. 实现注解类&#xff1a; Target(ElementType.FIELD) Retention(RetentionPolicy.RUNTIME) public interface FieldName {String value();boolean isIgnoreNull()…

day34_js

今日内容 0 复习昨日 1 事件 1.1 事件介绍 1.2 事件绑定方式 1.3 不同事件的演示 2 DOM操作 2.1 概述 2.2 查找元素 2.3 元素内容的查找和设置 2.4 元素属性的查找和设置 2.5 元素CSS样式的查找和设置 2.6 创建元素 2.7 创建文本节点 2.8 追加元素 2.9 删除元素 3 案例练习 0 复…

TCP 异常断开连接【重点】

参考链接 https://xiaolincoding.com/network/3_tcp/tcp_down_and_crash.html https://xiaolincoding.com/network/3_tcp/tcp_unplug_the_network_cable.html#%E6%8B%94%E6%8E%89%E7%BD%91%E7%BA%BF%E5%90%8E-%E6%9C%89%E6%95%B0%E6%8D%AE%E4%BC%A0%E8%BE%93 关键词&#xff1a…

无际线复选框

效果演示 实现了一个网格布局&#xff0c;其中每个网格是一个复选框&#xff0c;可以选择是否显示。每个复选框都有一个漂浮的天花板&#xff0c;表示它是一个房间的天花板。每个房间的天花板都有一个不同的形状和颜色&#xff0c;分别对应不同的房间。整个页面的背景是一个由两…

echarts多个折线图共用X轴,实现tooltip合并和分离

echarts共享X轴案例&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>Document</…

DevSecOps核心流程基本组成分析

目录 一、DevSecOps核心流程基本组成 1.1 核心流程概述 1.2 DevSecOps 核心流程说明 1.2.1 核心流程图 1.2.2 流程说明 1.2.2.1 持续开发 1.2.2.2 持续构建 1.2.2.3 持续运维 1.2.2.4 持续监控 二、DevSecOps核心流程经典场景 2.1 Azure DevSecOps核心流程 2.1.1 核…

HCIA-HarmonyOS设备开发认证-3.内核基础

目录 前言目标一、进程与线程待续。。。 前言 对于任何一个操作系统而言&#xff0c;内核的运行机制与原理是最为关键的部分。本章内容从多角度了解HarmonyOS的内核运行机制&#xff0c;涵盖进程与线程的概念&#xff0c;内存管理机制&#xff0c;网络特性&#xff0c;文件系统…

HTTP连接池在Java中的应用:轻松应对网络拥堵

网络拥堵是现代生活中无法避免的问题&#xff0c;尤其是在我们这个“点点点”时代&#xff0c;网页加载速度直接影响到我们的心情。此时&#xff0c;我们需要一位“救世主”——HTTP连接池。今天&#xff0c;就让我们一起探讨一下&#xff0c;这位“救世主”如何在Java中大显神…