云打印之菜鸟打印组件交互协议

菜鸟打印组件交互协议相关介绍如下:

1、打印组件下载地址

https://page.cainiao.com/waybill/cloud_printing/home.html

2、socket连接商品

如果是http的话,端口是13529

socket = new WebSocket('ws://localhost:13528');

如果是https的话,端口是13529

socket = new WebSocket('wss://localhost:13529');

3、简介

  • 云打印客户端是以独立进程和打印机交互(非作为浏览器插件进行打印)。
  • 浏览器或其他客户端需要通过 WebSocket,协议与云打印客户端进行通信,支持javascript,java,c/c++,python等常用的语言(建议使用对应开发与语言支持的 Websockt 库)。
  • 若 ISV 的 ERP 系统是B/S结构,建议使用如下版本浏览器:
  • chrome 45及以上(建议使用chrome的最新版本);
  • 相关浏览器的极速模式

4、请求协议头格式说明

请求协议头示例如下:

{"cmd": "command","requestID": "unique requestID","version": "1.0"
}

字段说明:

字段名

类型

说明

是否必须

cmd

string

请求的命令名称

requestID

string

请求的ID,用于唯一标识每个请求,每个客户端自己保证生成唯一ID,如UUID

version

string

协议当前版本,当前为“1.0”

响应协议头:

{"cmd": "command","requestID": "unique requestID"
}

字段说明: 

字段名

类型

说明

cmd

string

请求的命令名称

requestID

string

发送请求中的ID,原封不动返回,使客户端能识别出哪个请求对应的响应

5、print = 发送打印/预览数据协议

发送打印/预览数据协议(0.x版本)

注:因为打印机质量乘次不齐,建议 1 个 task 使用 一个 document,可以有效避免重打问题;

预览流程0.x:

打印流程(预览流程1.x):

请求协议格式(密文数据,针对菜鸟电子面单)如下:

{"cmd": "print","requestID": "123458976","version": "1.0","task": {"taskID": "7293666","preview": false,"printer": "","previewType": "pdf","firstDocumentNumber": 10,"totalDocumentCount": 100,"documents": [{"documentID": "0123456789","contents": [{"encryptedData":"AES:rU904rj6UH2oqfSUb43+Z+XlOkZaULeerkScS5xbmfjZC78uvsMTa3g6l33hRAz/srsk0TObjJaJI5n4tAPV1uv7szIPQGPDhwD6MK+zvTVIfuQCMC8p+cUB5S4FmqDhNE45LRVAlaoaI5YK8QmWK1WorhwnPxOFH4Ws/ApobtzDLDJaW6uu1AMEdAejEhRTWL3B1fRhhcDxc3gX+DZF9jJUB++fb9JZqmocWRu0Fvi/b1BokQx7Xt/N+FpJVRI0//NNUQ9b/W4tqGFIbf2IM/Ez1S5hBru5gKGdFzs99ZgCKqtWa0DnOzrZDXroU1mhurtlulE8QbipInu63fkIwn3h9ZSK0sMyV5Jrk5x3MIJDHeW9pc/Tw4TnKTAU134jl+GbbpYysa0+jBARWRjombeKIFSVfp/zgp15jClClUU1Nz4alTi22LimY2qteQRG6G/rCHiYxPoBRdrtqZZxNSdnKG5yjSdtA2CEL1DJNg1QkFVSSsOuqcHLdrKl6oMR+aUN6wM3GQikmKSU/CH4hWCCXxFaJXvBYoSxZ63GrM/d+l6D4+9+rCxHJoEVsa2E1TMHLUOnN6CweSM+45lcBK19bbCUJDyky6nb1NbxrZGYhmfkrNzE2GN+Cz4iTAgxJlQxd1gVvS4v5nB7qNfb0Uhy9NTopdumxOS7NXFFg3RFdBfAJ0nLGnxECUvUihBC3pwsLGimrUnIF4174m6J6Ga6cQE+Pp1LXgtKf5zWJdWHkm2vQhazcAsQC8JJZFb1ESp1vIAvpy0d0YmGrLLzxWNciHlOa7vguFCVF3UbTFe8r1Mxyym9rqNrZDXWRtBija9yeliMERVFuOTRjlc0PVAzveexQmuD4ESTzMZPtbO0jos1EITKhHcV35Na7E4I7bEe3L2u5yuFuzDA5cc8OA8v761+xOI70bGXUwvFO2kCCiUFEzI9ksLIDTtydBTA94lf4MYH6m0ziRmAhAgcwm5QJFd2G4JzpFIK4+dLuEZamrYUcnHmWzDIg+HYIXh6g3S2maFU7dUtwYoerptOTiVg8FxRlUTx30NDTgjm7ll8vEJXHj7yd/gAO3Vm9P54OSMv8w+pzX3gtCkvthrkjlToT1jMRNJyuJAeSBf5jruzYLS68inlSE/ehT10zhaiBvaCqojZZ2Ux0JQGhbR/nQ==","signature":"19d6f7759487e556ddcdd3d499af087080403277b7deed1a951cc3d9a93c42a7e22ccba94ff609976c5d3ceb069b641f541bc9906098438d362cae002dfd823a8654b2b4f655e96317d7f60eef1372bb983a4e3174cc8d321668c49068071eaea873071ed683dd24810e51afc0bc925b7a2445fdbc2034cdffb12cb4719ca6b7","templateURL":"http://cloudprint.cainiao.com/template/standard/101/123","ver":"waybill_print_secret_version_1"},{"data": {"value": "测试字段值需要配合自定义区变量名"},"templateURL": "http://cloudprint.cainiao.com/template/customArea/440439"}]}]}
}

请求协议格式(明文数据)如下:

{"cmd": "print","requestID": "123458976","version": "1.0","task": {"taskID": "7293666","preview": false,"printer": "","previewType": "pdf","firstDocumentNumber": 10,"totalDocumentCount": 100,"documents": [{"documentID": "0123456789","contents": [{"data": {"nick": "张三"},"templateURL": "http://cloudprint.cainiao.com/template/standard/278250/1"},{"data": {"value": "测试字段值需要配合自定义区变量名"},"templateURL": "http://cloudprint.cainiao.com/template/customArea/440439"}]}]}
}

 字段说明:

字段名

类型

说明

是否必须

taskID

string

打印机任务ID,每个打印任务会分配不同的且唯一的ID,在0.x中,默认不允许taskID重复,若重复则直接返回错误

在1.5.0中,当task结构中的idempotent设置为true时,不允许taskID重复,默认允许重复

idempotent

bool

1.5.0版本及以后支持

与taskID搭配使用,当值为true时,taskID不允许重复,默认为false

notifyType

array

打印通知类型:“render”, “print”

[“render”] : 仅渲染响应 notify

[“print”] : 仅出纸响应 notify

?“render”, “print” : 渲染完成会响应 notify && 出纸完成后会响应 notify

[] : 不允许

注:如果notifyType没有指定,默认为[“render”, “print”]

??在1.x版本中废弃此字段,总是会进行通知

preview

bool

是否预览.true为预览,false为打印

printType

string

可选dirctPrint或templatePrint

默认为templatePrint,当设置为dirctPrint时,templateURL可以放入PDF的链接进行PDF打印

previewType

string

属性取值“pdf” or “image” 预览模式,是以pdf还是image方式预览,二选一,此属性不是必选,默认以pdf预览。

firstDocumentNumber

int

task 起始 document 序号

totalDocumentCount

int

task document 总数

printer

string

打印机名,如果为空,会使用默认打印机

templateURL

string

模板文件url

signature

string

模板与数据的签名

documents

array

文档数组,每个数据表示一页

documentID

string

文档的唯一ID,对于菜鸟标准面单来讲,就是面单号;如果是自定义模板,需要保证唯一

data

Json Object

模板需要的打印数据

菜鸟打印组件响应

协议格式如下:{"cmd":"print","requestID":"123458976","taskID":"1","status":"success", //如果是打印,表示打印任务提交成功,如果是预览,表示预览PDF文件生成成功"previewURL":"http://127.0.0.1/previewxxx.pdf", //如果是预览,会返回这个属性,表示预览PDF文件的URL地址,如果是打印命令,不返回此属性
//如果是预览并且预览模式是previewType:image,会返回这个属性,表示预览图片的URL地址,如果是打印命令,不返回此属性"previewImage": [http://127.0.0.1/preview1.jpg,http://127.0.0.1/preview2.jpg,http://127.0.0.1/preview3.jpg],
//1.x后的菜鸟打印组件版本
"urls"["url1","url2"]
}

字段名

类型

说明

taskID

string

打印机任务ID,每个打印任务会分配不同的且唯一的ID

status

string

如果是打印,表示打印任务提交成功,如果是预览,表示预览PDF文件生成成功

previewURL

string

可预览的PDF文件URL路径

previewImage

string[]

预览image的URL路径,是一个字符串数组

urls

string[]

1.x后的菜鸟打印组件版本会在预览时返回

注:
* 如果是打印命令,只是表示将打印任务提交到任务队列,会快速返回。
* 如果是预览命令,且版本为0.x则需要将预览文件生成,才会返回,需要一段等待时间。如果是1.x版本,则会立即返回一条消息表示任务已提交到任务队列,随后当预览文件生成后,会再次返回一个消息并携带文件地址

最佳实践

由于网络协议本身的不可靠性,建议接入时按照以下规范进行,否则可能出现漏打、重复打等情况

  • 发送指令前先检查websock链接的可用性,若不可用则重连
  • 发送打印指令后等待【任务已提交】的响应,此时可以告知用户任务已提交打印
  • 保持发送的链接存活(不主动关闭),持续监听notifyPrintResult消息,当接收到失败或者成功后才关闭链接
  • 当接收到成功后,应修改业务系统中打印任务的状态并提示用户某任务已完成
  • 当接收到失败后,应修改业务系统中打印任务的状态并提示用户某任务已失败
  • 调整任务的taskID和idempotent配置以符合业务预期

6、notifyPrintResult = 打印通知

此消息总是由菜鸟打印组件向调用方返回

通知协议格式如下:

{"cmd":"notifyPrintResult","printer":"中通打印机A","taskID":"1","taskStatus":"printed","printStatus":[{"documentID”:”9890000112011”,"status":"success","msg":"if failed,some tips, if success ,nothing”,"detail":"错误信息的补充描述"}]
}

字段解释:

字段名

类型

说明

documentID

string

文档的唯一ID,对于菜鸟标准面单来讲,就是面单号;如果是自定义模板,需要保证唯一

taskStatus

string

任务状态:

failed : 失败;

rendered: 渲染完成

printed : 出纸完成

?注:当打印出纸之后才会发送通知并且只通知一次

status

string

任务状态:success成功;failed 失败,canceled 取消 (当一个任务中的一个文档打印失败,任务中其他的文档打印状态为“canceled”状态)

msg

string

如果任务状态为成功或挂起为空,如果任务状态为失败,则为失败原因概要。

detail

string

错误信息的补充描述

printer

string

负责打印的打印机名

taskID

string

任务ID,每个打印任务会分配不同的且唯一的ID

注:判断是否打印成功请根据“cmd=notifyPrintResult”、“taskStatus的状态”组合判断

7、 getPrinters = 获取打印机列表

请求协议格式如下:

{"cmd": "getPrinters","requestID": "123458976","version": "1.0"
}

响应协议格式如下:

{"cmd": "getPrinters","requestID": "123458976","defaultPrinter": "XX快递打印机","printers": [{"name": "XX快递打印机"},{"name": "YY物流打印机"}]
}

字段名

类型

说明

defaultPrinter

string

默认打印机

name

string

打印机的名字

 8、getPrinterConfig = 获取打印机配置

请求协议格式如下:

{"cmd":"getPrinterConfig","printer":"菜鸟打印机","version":"1.0","requestID":"123456789"
}

响应协议格式如下:

{"cmd": "getPrinterConfig","requestID": "123456789","status": "success/failed","msg": "如果出错,错误原因","printer": {"name": "打印机名称","needTopLogo": false,"needBottomLogo": false,"horizontalOffset": 1,"verticalOffset": 2,"forceNoPageMargins": true,"autoPageSize": false,"orientation": 0,"autoOrientation": false,"paperSize": {"width": 100,"height": 180}}
}

字段名

类型

说明

status

string

标示命令成功或失败,取值“success”或者“failed”

msg

string

如果出错,错误原因

printer.name

string

打印机名称

printer.needTopLogo

bool

是否需要模板上联的快递logo

true为需要

false为不需要

printer.needBottomLogo

bool

是否需要模板下联的快递logo

true为需要

false为不需要

printer.horizontalOffset

float

水平偏移量

printer.verticalOffset

float

垂直偏移量

printer.forceNoPageMargins

bool

强制设置页面无空边

true为强制设置页面无空边

false为由打印机驱动决定

printer.paperSize.width

int

打印机纸张的宽度,单位是毫米

printer.paperSize.height

int

打印机纸张的高度,单位是毫米

printer. autoPageSize

bool

true:自适应纸张大小

false:不自适应

printer. orientation

int

0:纵向 1: 横向

printer. autoOrientation

bool

true:按照 orientation 适应纸张方向

false:不自适应

9、setPrinterConfig = 设置打印机配置

请求协议格式如下:

{"cmd": "setPrinterConfig","requestID": "123458976","version": "1.0","printer": {"name": "菜鸟打印机","needTopLogo": true,"needBottomLogo": false,"horizontalOffset": 0.5,"verticalOffset": 0.7,"forceNoPageMargins": true,"autoPageSize": false,"orientation": 0,"autoOrientation": false,"paperSize": {"width": 100,"height": 180}}
}

注:参数说明参考 获取打印机配置(getPrinterConfig)

响应协议格式如下:

{"cmd":"setPrinterConfig","requestID":"123458976","status":"success","msg":"if failed ,some tips, if success,nothing"
}

字段名

类型

说明

status

string

消息处理结果。success:成功;failed:失败

msg

string

如果成功,则为空;如果失败,则为失败原因

注:如果要保持某个配置不变,应省略对应的配置字段。

10、getTaskStatus = 获取任务打印任务状态

请求协议格式如下:

{"cmd":"getTaskStatus","requestID":"123458976","version":"1.0","taskID":["12311","12312"]
}

字段名

类型

说明

是否必须

taskID

json数组

打印机任务ID列表

 响应协议格式如下:

{"cmd":"getTaskStatus","requestID":"123458976","printStatus":[{"taskID":"12312", "detailStatus":[{"documentID":"9890000112011","status":"success","msg":"if failed ,some tips, if success or pending nothing","printer":"中通打印机A"}]}]
}

字段名

类型

说明

taskID

string

打印机任务ID,每个打印任务会分配不同的且唯一的ID

documentID

string

文档的唯一ID,对于菜鸟标准面单来讲,就是面单号;如果是自定义模板,需要保证唯一

status

string

任务状态:success成功;failed失败;pending,提交到打印机打印队列

msg

string

如果任务状态为成功或挂起为空,如果任务状态为失败,则为失败原因。

printer

string

负责打印的打印机名

11、getGlobalConfig = 获取全局配置 

请求协议格式如下:

{"cmd":"getGlobalConfig","requestID":"12345678901","version":"1.0"
}

响应协议格式如下:

{"cmd":"getGlobalConfig","requestID":"12345678901","status":"success","msg":"return nothing when success, return some tips when failed","notifyOnTaskFailure":true,//忽略字体无法显示的问题"ignoreFontCanNotDisplay":true
}

字段解释:

字段名

类型

说明

status

string

表示命令成功或失败,取值“success”或者“failed”

msg

string

如果出错,错误原因

notifyOnTaskFailure

bool

打印任务失败时是否需要通知(弹出对话框提醒用户打印失败原因并默认暂停当前打印机的打印),true为需要,false为不需要

ignoreFontCanNotDisplay

bool

true表示忽略字体无法显示的问题

false则在字体无法显示时会弹窗报错

12、setGlobalConfig = 设置全局配置 

请求协议格式如下:

{"cmd":"setGlobalConfig","requestID":"12345678901","version":"1.0","notifyOnTaskFailure":true,//忽略字体无法显示的问题"ignoreFontCanNotDisplay":true
}

响应协议格式如下:

{"cmd":"setGlobalConfig","requestID":"12345678901","status":"success","msg":"return nothing when success, return some tips when failed"
}

字段解释:

字段名

类型

说明

status

string

表示命令成功或失败,取值“success”或者“failed”

msg

string

如果出错,错误原因

notifyOnTaskFailure

bool

打印任务失败时是否需要通知(弹出对话框提醒用户打印失败原因并默认暂停当前打印机的打印),true为需要,false为不需要

13、getAgentInfo = 获取客户端版本信息

请求协议格式如下:

{"cmd":"getAgentInfo","requestID":"12345678901","version":"1.0"
}

响应协议格式如下:

{"cmd":"getAgentInfo","requestID":"12345678901","status":"success","msg":"return nothing when success, return some tips when failed","version":"0.2.8.3" 
}

字段解释:

字段名

类型

说明

status

string

表示命令成功或失败,取值“success”或者“failed”

msg

string

如果出错,错误原因

version

string

版本号

注意事项

  • Websocket 建议使用长连接,不要每次发送交互请求去创建一个对象。
  • 在同打印组件交互过程中的json报文,如果文本中包含了特殊字符,比如常见的回车,引号等,需要对特殊字符做转义,详细请参考: JSON 。

14、JavaScript使用示例

function doConnect()
{socket = new WebSocket('ws://localhost:13528');//如果是https的话,端口是13529//socket = new WebSocket('wss://localhost:13529');// 打开Socketsocket.onopen = function(event){// 监听消息socket.onmessage = function(event){console.log('Client received a message',event);};// 监听Socket的关闭socket.onclose = function(event){console.log('Client notified socket has closed',event);};};
}
/**** * 获取请求的UUID,指定长度和进制,如 * getUUID(8, 2)   //"01001010" 8 character (base=2)* getUUID(8, 10) // "47473046" 8 character ID (base=10)* getUUID(8, 16) // "098F4D35"。 8 character ID (base=16)*   */
function getUUID(len, radix) {var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split('');var uuid = [], i;radix = radix || chars.length; if (len) {for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix];} else {var r;uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-';uuid[14] = '4';for (i = 0; i < 36; i++) {if (!uuid[i]) {r = 0 | Math.random()*16;uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];}}}return uuid.join('');
}
/**** 构造request对象*/
function getRequestObject(cmd){var request  = new Object();request.requestID=getUUID(8, 16);request.version="1.0";request.cmd=cmd;return request;
}
/**** 获取自定义区数据以及模板URL* waybillNO 电子面单号*/
function getCustomAreaData(var waybillNO){//获取waybill对应的自定义区的JSON object,此处的ajaxGet函数是伪代码var jsonObject = ajaxGet(waybillNO);var ret = new Object();ret.templateURL=jsonObject.content.templateURL;ret.data=jsonObject.content.data;return ret;
}
/**** 获取电子面单Json 数据* waybillNO 电子面单号*/
function getWaybillJson(var waybillNO){//获取waybill对应的json object,此处的ajaxGet函数是伪代码var jsonObject = ajaxGet(waybillNO);return jsonObject;
}
/*** 请求打印机列表demo* */
var request  = getRequestObject("getPrinters");
webSocket.send(JSON.stringify(request));
/*** 弹窗模式配置打印机* */
var request  = getRequestObject("printerConfig");
webSocket.send(JSON.stringify(request));
/*** 打印电子面单* printer 指定要使用那台打印机* waybillArray 要打印的电子面单的数组*/
function doPrint(var printer,var waybillArray)
{var request = getRequestObject("print");    request.task = new Object();request.task.taskID = getUUID(8,10);request.task.preview = false;request.task.printer = printer;var documents = new Array();for(i=0;i<waybillArray.length;i++) {var doc = new Object();doc.documentID = waybillArray[i];var content = new Array();var waybillJson = getWaybillJson(waybillArray[i]);var customAreaData = getCustomAreaData(waybillArray[i]);content.push(waybillJson,customAreaData);doc.content = content;documents.push(doc);}request.task.documents=documents;socket.send(JSON.stringify(request));
}
/*** 响应请求demo* */
websocket.onmessage = function(event){   var response = eval(event.data);if (response.cmd == 'getPrinters') {getPrintersHandler(response);//处理打印机列表} else if (response.cmd == 'printerConfig') {printConfigHandler(response);} 
};

15、JAVA使用示例

java使用websocket需要引入第三方库 下载地址 。

<dependency><groupId>org.java-websocket</groupId><artifactId>Java-WebSocket</artifactId><version>1.3.0</version>
</dependency>

自己创建一个websocket管理类,需要继承自第三方类库的WebSocketClient:

import java.net.URI;
import java.net.URISyntaxException;import org.java_websocket.client.WebSocketClient;
import org.java_websocket.drafts.Draft;
import org.java_websocket.drafts.Draft_17;
import org.java_websocket.handshake.ServerHandshake;public class WebSocketClientManager extends WebSocketClient {static WebSocketClientManager webSocket = null;public static void main(String[] args) throws URISyntaxException {String uri = "ws://127.0.0.1:13528";webSocket = new WebSocketClientManager(new URI(uri), new Draft_17());//建立连接webSocket.connect();}public WebSocketClientManager(URI serverUri, Draft draft) {super(serverUri, draft);}@Overridepublic void onOpen(ServerHandshake serverHandshake) {//获取打印机列表String getPrinterListCmd = "{\"requestID\":\"12345678901234567890\",\"verson\":\"1.0\",\"cmd\":\"getPrinters\"}";webSocket.send(getPrinterListCmd);//发送打印任务String printCmd = "打印任务报文,内容过长此处不粘贴";webSocket.send(printCmd);}//WebSocket回调函数@Overridepublic void onMessage(String message) {//TODO 对打印服务返回的数据进行处理System.out.println(message);}@Overridepublic void onClose(int i, String s, boolean b) {}@Overridepublic void onError(Exception e) {}
}

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

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

相关文章

CPT203 Software Engineering 软件工程 Pt.3 系统建模(中英双语)

文章目录 5. System Modeling&#xff08;系统建模&#xff09;5.1 Context models&#xff08;上下文模型&#xff09;5.2 Interaction models&#xff08;交互模型&#xff09;5.2.1 Use case modeling&#xff08;用况建模&#xff09;5.2.2 Sequence diagram&#xff08;顺…

什么是Kafka的重平衡机制?

Kafka 的重平衛机制是指在消费者组中新增或删除消费者时&#xff0c;Kafka 集群会重新分配主题分区给各个消费者&#xff0c;以保证每个消费者消费的分区数量尽可能均衡。 重平衡机制的目的是实现消费者的负载均衡和高可用性&#xff0c;以确保每个消费者都能够按照预期的方式…

Nginx——反向代理(三/五)

目录 1.Nginx 反向代理1.1.Nginx 反向代理概述1.2.Nginx 反向代理的配置语法1.2.1.proxy_pass1.2.2.proxy_set_header1.2.3.proxy_redirect 1.3.Nginx 反向代理实战1.4.Nginx 的安全控制1.4.1.如何使用 SSL 对流量进行加密1.4.2.Nginx 添加 SSL 的支持1.4.3.Nginx 的 SSL 相关指…

VBA(Visual Basic for Applications)编程|excel|一系列网址或文件路径快速转换为可点击的超链接

很多时候&#xff0c;我们需要把导入的数据某一列转换成超链接&#xff0c;比如URL形式的列。 那么&#xff0c;大批量的情况下&#xff0c;无疑一个个手动点击是非常愚蠢的办法&#xff0c;这个时候我们就需要VBA编程来编写宏&#xff0c;通过编写宏来简化这些手动操作并不现…

【C++】图像模糊处理题目详解与实现

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述题目内容输入格式输出格式示例输入&#xff1a;输出&#xff1a; &#x1f4af;题目分析问题拆解 &#x1f4af;我的做法代码实现代码分析 &#x1f4af;老师的做法…

在 Ubuntu 22.04 上部署 AppArmor 应用安全教程

在这篇教程中&#xff0c;我们将为你设置 AppArmor 以增强应用程序的安全性。 AppArmor 是一个 Linux 安全模块&#xff0c;允许你限制各个程序的功能。通过强制执行强制访问控制策略&#xff0c;AppArmor 增强了你的应用程序和系统的安全性。 AppArmor 是一个易于使用的 Lin…

Nginx:动静分离

什么是动静分离? 动静分离 是指将网站中的静态资源(如图片、样式表、脚本等)和动态内容(如 PHP、Python、Node.js 等后端生成的内容)分开部署和处理。这样做的好处是可以利用不同的服务器或缓存策略来优化不同类型的资源。 动静分离的好处 提高性能:静态资源可以直接从…

每天40分玩转Django:Django插件开发

Django插件开发 一、插件开发概述表 阶段主要任务技术要点难度准备工作项目结构设计、环境配置项目布局、setup.py★★★☆☆开发实现功能开发、测试编写Django AppConfig、Signals★★★★☆文档编写API文档、使用说明Markdown、reStructuredText★★★☆☆发布部署PyPI打包…

什么是TDD测试驱动开发(Test Driven Development)?

什么是测试驱动开发&#xff1f; 软件开发团队通常会编写自动化测试套件来防止回归。这些测试通常是在编写应用程序功能代码之后编写的。我们将采用另一种方法&#xff1a;在实现应用程序代码之前编写测试。这称为测试驱动开发 (TDD)。 为什么要应用 TDD&#xff1f;通过在实…

后台管理系统用户退出登录方案实现

退出登录一直是一个通用的前端实现方案&#xff0c;对于退出登录而言&#xff0c;它的触发时机一般有两种&#xff1a; 1. 用户主动退出&#xff0c;即用户点击登录按钮之后退出&#xff1b; 2. 用户被动退出&#xff0c;Token过期或被 其他人"顶下来" 时退出&…

文献分享:BGE-M3——打通三种方式的嵌入模型

文章目录 1. \textbf{1. } 1. 背景与导论 1.1. \textbf{1.1. } 1.1. 研究背景 1.2. \textbf{1.2. } 1.2. 本文的研究 1.3. \textbf{1.3. } 1.3. 有关工作 2. M3-Embedding \textbf{2. M3-Embedding} 2. M3-Embedding 2.1. \textbf{2.1. } 2.1. 模型核心: 混合检索方式 2.1.1. \…

Hadoop•FinalShell连接VMware免密登录

听说这是目录哦 FinalShell连接VMware&#x1f324;️解决重连失效FinalShell的使用 免密登录⛈️能量站&#x1f61a; FinalShell连接VMware&#x1f324;️ 保持虚拟机的开机状态&#xff0c;打开FinalShell&#xff0c;如果虚拟机关机或者挂起&#xff0c;连接就会断开。 …

一个在ios当中采用ObjectC和opencv来显示图片的实例

前言 在ios中采用ObjectC编程利用opencv来显示一张图片&#xff0c;并简单绘图。听上去似乎不难&#xff0c;但是实际操作下来&#xff0c;却不是非常的容易的。本文较为详细的描述了这个过程&#xff0c;供后续参考。 一、创建ios工程 1.1、选择ios工程类型 1.2、选择接口模…

《Rust权威指南》学习笔记(五)

高级特性 1.在Rust中&#xff0c;unsafe是一种允许绕过Rust的安全性保证的机制&#xff0c;用于执行一些Rust默认情况下不允许的操作。unsafe存在的原因是&#xff1a;unsafe 允许执行某些可能被 Rust 的安全性检查阻止的操作&#xff0c;从而可以进行性能优化&#xff0c;如手…

【顶刊TPAMI 2025】多头编码(MHE)之极限分类 Part 3:算法实现

目录 1 三种多头编码&#xff08;MHE&#xff09;实现1.1 多头乘积&#xff08;MHP&#xff09;1.2 多头级联&#xff08;MHC&#xff09;1.3 多头采样&#xff08;MHS&#xff09;1.4 标签分解策略 论文&#xff1a;Multi-Head Encoding for Extreme Label Classification 作者…

docker中使用Dockerfile设置Volume挂载点

关于在docker中如何使用Volume&#xff0c;可以参考文章&#xff1a; docker中使用Volume完成数据共享-CSDN博客 如果想在生成docker镜像的时候设置好挂载点&#xff0c;而不是在运行镜像生成容器时生成。 下面以自建一个tomcat镜像为例&#xff0c;演示如何在生成镜像时设置…

springboot548二手物品交易boot代码(论文+源码)_kaic

摘 要 如今社会上各行各业&#xff0c;都喜欢用自己行业的专属软件工作&#xff0c;互联网发展到这个时候&#xff0c;人们已经发现离不开了互联网。新技术的产生&#xff0c;往往能解决一些老技术的弊端问题。因为传统二手物品交易信息管理难度大&#xff0c;容错率低&#x…

通往O1开源之路

“Scaling of Search and Learning: A Roadmap to Reproduce o1 from Reinforcement Learning Perspective”由复旦大学和上海人工智能实验室的研究者撰写。该论文从强化学习视角出发&#xff0c;深入分析了实现类似OpenAI o1模型性能的路线图&#xff0c;聚焦于策略初始化、奖…

SD下载、安装、使用、卸载-Stable Diffusion整合包v4.10发布!

目录 前言概述 SD安装1、安装软件2、启动3、配置4、运行5、测试 导入SD模型【决定画风】常用模型下载安装模型 SD卸载SD文生图提示词提示词使用技巧提示词的高级使用技巧强调关键词 前言 我向来不喜欢搞一些没有用的概念&#xff0c;所以直接整理可能用到的东西。 sd简单的说…

Mac iTerm2集成DeepSeek AI

1. 去deepseek官网申请api key&#xff0c;DeepSeek 2. 安装iTerm2 AI Plugin插件&#xff0c;https://iterm2.com/ai-plugin.html&#xff0c;插件解压后直接放到和iTerms相同的位置&#xff0c;默认就在/Applications 下 3. 配置iTerm2 4. 重启iTerm2,使用快捷键呼出AI对话…