基于 JNI + Rust 实现一种高性能 Excel 导出方案(下篇)
衡量一个人是否幸福,不应看他有多少高兴的事,而应看他是否为小事烦扰。只有幸福的人,才会把无关痛痒的小事挂心上。那些真正经历巨大灾难和深重痛苦的人,根本无暇顾及这些小事的。因此人们往往在失去幸福之后,才会发现它们曾经存在。
在上篇我们实现了最基础的导出功能,本篇我们拓展样式 API,封装为通用 Jar 包,提升开发者的使用友好度,并且优化性能以及基本的测试。
基于 JNI + Rust 实现一种高性能 Excel 导出方案(上篇)
一、写入表头
1、实现逻辑
1、定义一个注解,作用与类字段上,用于标识字段对应的表头名称;
2、定义一个 Excel 导出 VO 类,为每个字段添加注解,如果没有注解,则默认采用字段名;
3、创建 WorkSheet 时,通过反射获取所有字段与注解,整合到一个 TreeMap,然后写入表头
2、Java 部分
新增注解类 ExcelColumn
/*** @version: V1.0* @author: 余衫马* @description: Excel导出属性* @data: 2024-11-28 10:26**/
@Retention(RetentionPolicy.RUNTIME) // 注解在运行时可用
@Target(ElementType.FIELD) // 注解作用于字段
public @interface ExcelColumn {/*** 列名*/String value();}
调整构造方法
调整构造方法的参数,通过参数指定是否导出表头。
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略.../*** 构造方法* 初始化一个 Excel 对象*/public MyExportResultHandler() {this(DEFAULT_SHEET_NAME, true, ((ServletRequestAttributes) Objects.requireNonNull(RequestContextHolder.getRequestAttributes())).getResponse());}/*** 构造方法* 初始化一个 Excel 对象** @param writeHeader 是否需要写入表头,默认 true*/public MyExportResultHandler(String sheetName, boolean writeHeader, HttpServletResponse response) {// 基本配置this.sheetName = StringUtils.isEmpty(sheetName) ? DEFAULT_SHEET_NAME : sheetName;this.writeHeader = writeHeader;this.startRowIndex = writeHeader ? 1 : 0;this.httpServletResponse = response;// 解析字段this.fieldMap = getExportFieldMap();// 创建 Excel 指针Object[] objects = this.fieldMap.keySet().toArray();this.handle = writeHeader ? createWorksheet(this.sheetName, objects, objects.length) : createWorksheet(this.sheetName);}// 省略...
}
控制数据顺序
封装方法 getExportFieldMap 将导出类字段转为 TreeMap,这用于控制表头顺序与写入数据的顺序。
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略.../*** 将导出类字段转为 TreeMap** @return TreeMap*/private TreeMap<String, String> getExportFieldMap() {// 创建 TreeMap 来存储字段名和注解值TreeMap<String, String> fieldMap = new TreeMap<>();// 获取 TestVo 类的所有字段Field[] fields = TestVo.class.getDeclaredFields();// 遍历所有字段for (Field field : fields) {// 检查字段是否有 ExcelColumn 注解if (field.isAnnotationPresent(ExcelColumn.class)) {// 获取注解ExcelColumn excelColumn = field.getAnnotation(ExcelColumn.class);// 将注解值和字段名存储到 TreeMap 中fieldMap.put(excelColumn.value(), field.getName());} else {// 如果没有注解,使用字段名作为键fieldMap.put(field.getName(), field.getName());}}// 打印 TreeMap 内容for (String key : fieldMap.keySet()) {System.out.println(key + " -> " + fieldMap.get(key));}return fieldMap;}// 省略...
}
写入数据逻辑
写入数据时,按照 TreeMap 顺序。
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略.../*** 获取类字段值* @param obj 类实例* @param fieldName 字段名* @return Object* @throws Exception*/public static Object getFieldValueUsingGetter(Object obj, String fieldName) throws Exception {// 获取类对象Class<?> clazz = obj.getClass();// 构造getter方法名String getterName = "get" + Character.toUpperCase(fieldName.charAt(0)) + fieldName.substring(1);// 获取getter方法Method getterMethod = clazz.getMethod(getterName);// 调用getter方法并返回值return getterMethod.invoke(obj);}@Overridepublic void handleResult(ResultContext<? extends TestVo> resultContext) {TestVo testVo = resultContext.getResultObject();int row = startRowIndex == 0 ? resultContext.getResultCount() - 1 : resultContext.getResultCount();int col = 0;try {// 处理逻辑:按照 TreeMap 顺序写入数据for (String key : fieldMap.keySet()) {String fieldName = fieldMap.get(key);writeToWorksheet(handle, row, col, getFieldValueUsingGetter(testVo, fieldName).toString());col = col + 1;}} catch (Exception e) {e.printStackTrace();}}// 省略...}
3、Rust部分
改动 createWorksheet 函数,新建参数解析。
#[no_mangle]
pub extern "system" fn Java_com_yushanma_crazyexcel_handler_MyExportResultHandler_createWorksheet(mut env: JNIEnv,_class: JClass,sheet_name: JObject,header_array: JObjectArray,header_length: jint,
) -> jlong {// worksheet 为空if sheet_name.is_null() {eprintln!("sheet name argument is null");return 0;}// 需要写表头但是为空if header_length > 0 && header_array.is_null() {eprintln!("excel header argument is null");return 0;}let name: String = match unsafe { env.get_string_unchecked(&sheet_name.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return 0;}};let workbook = Box::new(Workbook::new());let workbook_ptr = Box::into_raw(workbook);// SAFETY: We just created the raw pointer from a Box, so it's valid.let worksheet = unsafe { (*workbook_ptr).add_worksheet().set_name(name.as_str()).unwrap() };// 写入表头if header_length > 0 {let mut data: Vec<String> = Vec::with_capacity(header_length as usize);for i in 0..header_length {let e = env.get_object_array_element(&header_array, i).unwrap();data.push(unsafe { env.get_string_unchecked(&e.into()).unwrap().into() });}if let Err(e) = worksheet.write_row(0, 0, &data) {eprintln!("Failed to write to worksheet: {:?}", e);// 释放指针unsafe {let _ = Box::from_raw(workbook_ptr);}return 0;}}let handle = Box::new(WorksheetHandle {workbook: workbook_ptr,worksheet,});Box::into_raw(handle) as jlong
}
4、运行效果
给 TestVo 的后 5 个字段添加表头注解,
可以看到表头与数据的顺序保持一致。
二、配置样式
1、常见样式
宽度高度
// Set the column width for clarity.
worksheet.set_column_width(0, 22)?;// Set the row height in Excel character units.
worksheet.set_row_height(0, 30)?;
字体加粗
let bold_format = Format::new().set_bold();
// Write a string with the bold format defined above.
worksheet.write_with_format(1, 0, "World", &bold_format)?;
日期格式
let date_format = Format::new().set_num_format("yyyy-mm-dd");
// Write a date.
let date = ExcelDateTime::from_ymd(2024, 11, 30)?;
worksheet.write_with_format(6, 0, &date, &date_format)?;
小数格式
let decimal_format = Format::new().set_num_format("0.000");
// Write a number with formatting.
worksheet.write_with_format(4, 0, 3.00, &decimal_format)?;
颜色相关
// 字体颜色
let format = Format::new().set_font_color(Color::Red);
// let format = Format::new().set_font_color(Color::RGB(0x000000));
worksheet.write_string_with_format(0, 0, "Wheelbarrow", &format)?;// 背景颜色
let format1 = Format::new().set_background_color(Color::Green);
// let format1 = Format::new().set_font_color(Color::RGB(0x000000));
worksheet.write_string_with_format(0, 0, "Rust", &format1)?;
Url 超链接
// Write some links.
worksheet.write(7, 0, Url::new("https://www.rust-lang.org"))?;
worksheet.write(8, 0, Url::new("https://www.rust-lang.org").set_text("Rust"))?;
2、实现逻辑
- Java:添加注解属性,拓展对应字段的属性,同样通过反射机制将属性解析到 TreeMap<String,Object>
- Rust:解析是否有格式,没有则采用默认格式
3、实现表头样式
Java 部分
新增注解属性,
/*** @version: V1.0* @author: 余衫马* @description: Excel导出属性* @data: 2024-11-28 10:26**/
@Retention(RetentionPolicy.RUNTIME) // 注解在运行时可用
@Target(ElementType.FIELD) // 注解作用于字段
public @interface ExcelColumn {/*** 列名*/String value();/*** 宽度* @return int*/int columnWidth() default 0;/*** 高度* @return int*/int rowHeight() default 0;/*** 字体加粗* @return boolean*/boolean bold() default false;/*** 时间格式* @return String*/String dateFormat() default "yyyy-mm-dd";/*** 小数格式* @return String*/String decimalFormat() default "0.000";/*** 字体颜色* @return int*/int fontColor() default 0x000000;/*** 背景颜色* @return int*/int backgroundColor() default 0xFFFFFF;/*** 是否超链接* @return boolean*/boolean isLink() default false;
}
构造函数解析导出属性,
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略.../*** 解析字段类型** @param fieldType 字段* @return int*/private int parseFieldType(Class<?> fieldType) {// 确定字段类型if (fieldType == String.class) {// "String"return 0;} else if (fieldType == Date.class) {// "Date"return 1;} else if (fieldType == java.sql.Timestamp.class || fieldType == java.time.LocalDateTime.class) {// "DateTime"return 2;} else if (Number.class.isAssignableFrom(fieldType) || fieldType.isPrimitive()) {// "Number"return 3;} else {// "Unknown"return 99;}}/*** 将导出类字段转为 TreeMap** @return TreeMap*/public TreeMap<String, Object> getExportFieldMap() {// 创建 TreeMap 来存储字段名和注解值TreeMap<String, Object> fieldMap = new TreeMap<>();// 获取 TestVo 类的所有字段Field[] fields = TestVo.class.getDeclaredFields();// 遍历所有字段for (Field field : fields) {int fieldType = parseFieldType(field.getType());// 创建 ExcelColumnProperties 对象ExcelColumnProperties properties = new ExcelColumnProperties();properties.setFieldType(fieldType);properties.setFieldName(field.getName());// 检查字段是否有 ExcelColumn 注解if (field.isAnnotationPresent(ExcelColumn.class)) {// 获取注解ExcelColumn excelColumn = field.getAnnotation(ExcelColumn.class);// 设置属性properties.setColumnName(excelColumn.value());properties.setColumnWidth(excelColumn.columnWidth());properties.setRowHeight(excelColumn.rowHeight());properties.setBold(excelColumn.bold());properties.setDateFormat(excelColumn.dateFormat());properties.setDecimalFormat(excelColumn.decimalFormat());properties.setFontColor(excelColumn.fontColor());properties.setBackgroundColor(excelColumn.backgroundColor());properties.setIsLink(excelColumn.isLink());// 将注解值和字段名存储到 TreeMap 中fieldMap.put(excelColumn.value(), properties);} else {// 设置默认属性properties.setColumnName(field.getName());properties.setColumnWidth(0);properties.setRowHeight(0);properties.setBold(false);properties.setDateFormat(DEFAULT_DATE_FORMAT);properties.setDecimalFormat(DEFAULT_DECIMAL_FORMAT);properties.setFontColor(DEFAULT_FONT_COLOR_RGB);properties.setBackgroundColor(0xFFFFFF);properties.setIsLink(DEFAULT_IS_LINK);// 将字段名存储到 TreeMap 中fieldMap.put(field.getName(), properties);}}// 打印 TreeMap 内容for (String key : fieldMap.keySet()) {System.out.println(key + " -> " + fieldMap.get(key));}return fieldMap;}// 省略...
}
创建 worksheet 时使用默认表头样式,
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略...private final static Integer DEFAULT_COLUMN_WIDTH = 50;private final static Integer DEFAULT_ROW_HEIGHT = 26;private final static Boolean DEFAULT_BOLD = true;private final static Integer DEFAULT_FONT_COLOR_RGB = 0x000000;private final static Integer DEFAULT_BG_COLOR_RGB = 0x808080;private final static Boolean DEFAULT_IS_LINK = false;private final static String DEFAULT_DATE_FORMAT = "yyyy-mm-dd";private final static String DEFAULT_DECIMAL_FORMAT = "0.000";/*** 创建一个 Worksheet** @param sheetName worksheet name* @param headers 表头数组* @param length 表头列数* @param columnWidth 列宽度* @param rowHeight 行高度* @param bold 字体加粗* @param fontColorRgb 字体颜色* @param backgroundColorRgb 背景颜色* @return 指针 handle*/private native long createWorksheet(String sheetName, Object[] headers, int length,int columnWidth, int rowHeight, boolean bold,int fontColorRgb, int backgroundColorRgb);/*** 构造方法* 初始化一个 Excel 对象** @param writeHeader 是否需要写入表头,默认 true*/public MyExportResultHandler(String sheetName, boolean writeHeader, HttpServletResponse response) {// 基本配置this.sheetName = StringUtils.isEmpty(sheetName) ? DEFAULT_SHEET_NAME : sheetName;this.writeHeader = writeHeader;this.startRowIndex = writeHeader ? 1 : 0;this.httpServletResponse = response;// 解析字段this.fieldMap = getExportFieldMap();// 创建 Excel 指针Object[] objects = this.fieldMap.keySet().toArray();this.handle = writeHeader ?createWorksheet(this.sheetName, objects, objects.length,HEADER_COLUMN_WIDTH, HEADER_ROW_HEIGHT, HEADER_BOLD, HEADER_FONT_COLOR_RGB, HEADER_BG_COLOR_RGB): createWorksheet(this.sheetName);}// 省略...
}
Rust 部分
#[no_mangle]
pub extern "system" fn Java_com_yushanma_crazyexcel_handler_MyExportResultHandler_createWorksheet(mut env: JNIEnv,_class: JClass,sheet_name: JObject,header_array: JObjectArray,header_length: jint,column_width: jint,row_height: jint,bold: jboolean,font_color: jint,bg_color: jint,
) -> jlong {// worksheet 为空if sheet_name.is_null() {eprintln!("sheet name argument is null");return 0;}// 需要写表头但是为空if header_length > 0 && header_array.is_null() {eprintln!("excel header argument is null");return 0;}let name: String = match unsafe { env.get_string_unchecked(&sheet_name.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return 0;}};let workbook = Box::new(Workbook::new());let workbook_ptr = Box::into_raw(workbook);// SAFETY: We just created the raw pointer from a Box, so it's valid.let worksheet = unsafe {(*workbook_ptr).add_worksheet().set_name(name.as_str()).unwrap()};// 写入表头if header_length > 0 {// Set the row height in Excel character units.if row_height > 0 {worksheet.set_row_height(0, row_height).unwrap();}let mut data: Vec<String> = Vec::with_capacity(header_length as usize);for i in 0..header_length {// Set the column width for clarity.if column_width > 0 {worksheet.set_column_width(i as u16, column_width).unwrap();}let e = env.get_object_array_element(&header_array, i).unwrap();data.push(unsafe { env.get_string_unchecked(&e.into()).unwrap().into() });}// 样式let mut format = Format::new();// 字体颜色format = format.set_font_color(Color::RGB(font_color as u32));// 背景颜色format = format.set_background_color(Color::RGB(bg_color as u32));// 字体大小format = format.set_font_size(20);// 边框format = format.set_border(FormatBorder::Thin);// 字体加粗if bold == 1 {format = format.set_bold();}// 写入数据if let Err(e) = worksheet.write_row_with_format(0, 0, &data, &format) {eprintln!("Failed to write to worksheet: {:?}", e);// 释放指针unsafe {let _ = Box::from_raw(workbook_ptr);}return 0;}}let handle = Box::new(WorksheetHandle {workbook: workbook_ptr,worksheet,});Box::into_raw(handle) as jlong
}
4、实现内容样式
Java 部分
/*** @version: V1.0* @author: 余衫马* @description: Excel 导出处理器* @data: 2024-11-21 19:56**/
public class MyExportResultHandler implements ResultHandler<TestVo>, AutoCloseable {// 省略.../*** 把数据写入 Worksheet** @param handle 指针* @param row 行* @param col 列* @param text 内容* @param columnWidth 列宽度* @param rowHeight 行高度* @param bold 字体加粗* @param dateFormat 日期格式* @param decimalFormat 小数格式* @param fontColorRgb 字体颜色* @param backgroundColorRgb 背景颜色* @param isLink 是否超链接* @param fieldType 字段类型*/private native void writeToWorksheet(long handle, int row, int col, String text,int columnWidth, int rowHeight, boolean bold,String dateFormat, String decimalFormat,int fontColorRgb, int backgroundColorRgb, boolean isLink,int fieldType);@Overridepublic void handleResult(ResultContext<? extends TestVo> resultContext) {//invoke(resultContext.getResultObject());TestVo testVo = resultContext.getResultObject();int row = startRowIndex == 0 ? resultContext.getResultCount() - 1 : resultContext.getResultCount();int col = 0;try {// 处理逻辑:按照 TreeMap 顺序写入数据for (String key : fieldMap.keySet()) {ExcelColumnProperties properties = (ExcelColumnProperties) fieldMap.get(key);String fieldName = properties.getFieldName();writeToWorksheet(handle, row, col, getFieldValueUsingGetter(testVo, fieldName).toString(),properties.getColumnWidth(),properties.getRowHeight(),properties.getBold(),properties.getDateFormat(),properties.getDecimalFormat(),properties.getFontColor(),properties.getBackgroundColor(),properties.getIsLink(),);col = col + 1;}} catch (Exception e) {e.printStackTrace();}}// 省略...}
Rust 部分
#[no_mangle]
pub extern "system" fn Java_com_yushanma_crazyexcel_handler_MyExportResultHandler_writeToWorksheet(env: JNIEnv,_class: JClass,handle_ptr: jlong,row: u32,col: u16,text: JObject,column_width: jint,row_height: jint,bold: jboolean,date_format: JString,decimal_format: JString,font_color: jint,bg_color: jint,is_link: jboolean,field_type: jint,
) {if handle_ptr == 0 {eprintln!("Invalid handle pointer");return;}if text.is_null() {eprintln!("Text argument is null");return;}let handle = unsafe { &*(handle_ptr as *mut WorksheetHandle) };let worksheet: &mut Worksheet = unsafe { &mut *handle.worksheet };// 列宽度if column_width > 0 {worksheet.set_column_width(col, column_width).unwrap();}// 行高度if row_height > 0 {worksheet.set_row_height(row, row_height).unwrap();}// 样式let mut format = Format::new();// 字体颜色format = format.set_font_color(Color::RGB(font_color as u32));// 背景颜色format = format.set_background_color(Color::RGB(bg_color as u32));// 边框format = format.set_border(FormatBorder::Thin);// 字体加粗if bold == 1 {format = format.set_bold();}// 写入 URLif is_link == 1 {// let content: String = match env.get_string(&text.into()) {// Ok(java_str) => java_str.into(),// Err(e) => {// eprintln!("Couldn't get java string: {:?}", e);// return;// }// };let content: String = match unsafe { env.get_string_unchecked(&text.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};if let Err(e) = worksheet.write_with_format(row, col, Url::new(content), &format) {eprintln!("Failed to write to worksheet: {:?}", e);}return;}// 写入字符串if field_type == 0 {let content: String = match unsafe { env.get_string_unchecked(&text.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};if let Err(e) = worksheet.write_string_with_format(row, col, content, &format) {eprintln!("Failed to write to worksheet: {:?}", e);}return;}// 写入日期时间if field_type == 1 || field_type == 2 {// 日期格式if !date_format.is_null() {let date_format: String = match unsafe { env.get_string_unchecked(&date_format.into()) }{Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};format = format.set_num_format(date_format);}let content: String = match unsafe { env.get_string_unchecked(&text.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};let datetime = ExcelDateTime::parse_from_str(content.as_str()).unwrap();if let Err(e) = worksheet.write_datetime_with_format(row, col, datetime, &format) {eprintln!("Failed to write to worksheet: {:?}", e);}return;}// 写入数字if field_type == 3 {//小数格式if !decimal_format.is_null() {let decimal_format: String =match unsafe { env.get_string_unchecked(&decimal_format.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};format = format.set_num_format(decimal_format);}let content: String = match unsafe { env.get_string_unchecked(&text.into()) } {Ok(java_str) => java_str.into(),Err(e) => {eprintln!("Couldn't get java string: {:?}", e);return;}};let number = match content.parse::<f64>() {Ok(number) => number,Err(e) => {eprintln!("Failed to parse the string: {}", e);return;}};if let Err(e) = worksheet.write_number_with_format(row, col, number, &format) {eprintln!("Failed to write to worksheet: {:?}", e);}return;}
}
效果,
三、泛型抽象
/*** @version: V1.0* @author: 余衫马* @description: BaseExcel 导出处理器* @data: 2024-12-03 18:00**/
public abstract class BaseExportResultHandler<T> implements ResultHandler<T> {// 省略.../*** 构造方法* 初始化一个 Excel 对象** @param writeHeader 是否需要写入表头,默认 true*/public BaseExportResultHandler(String sheetName, boolean writeHeader, HttpServletResponse response) {// 基本配置this.sheetName = StringUtils.isEmpty(sheetName) ? DEFAULT_SHEET_NAME : sheetName;this.writeHeader = writeHeader;this.startRowIndex = writeHeader ? 1 : 0;this.httpServletResponse = response;// 解析字段this.fieldMap = getExportFieldMap();// 创建 Excel 指针Object[] objects = this.fieldMap.keySet().toArray();this.handle = writeHeader ?createWorksheet(this.sheetName, objects, objects.length,DEFAULT_COLUMN_WIDTH, DEFAULT_ROW_HEIGHT, DEFAULT_BOLD, DEFAULT_FONT_COLOR_RGB, DEFAULT_BG_COLOR_RGB): createWorksheet(this.sheetName);}/*** 通过 ResultHandler 流式查询数据,由子类自定义实现*/public abstract void fetchDataByStream();/*** 开始导出* @throws IOException*/public void startExportExcel() throws IOException {httpServletResponse.setContentType("application/vnd.ms-excel");httpServletResponse.setCharacterEncoding("utf-8");httpServletResponse.setHeader("Content-disposition", "attachment;filename=out.xlsx");OutputStream outputStream = httpServletResponse.getOutputStream();try {// 子类流式获取数据this.fetchDataByStream();// 写入响应流给客户端writeToStream(handle, outputStream);} finally {if (handle != 0) {freeWorksheetHandle(handle);handle = 0;}// 关闭流if (outputStream != null) {outputStream.flush();outputStream.close();}}}// 省略...}
ServiceImpl 调用,
/*** @version: V1.0* @author: 余衫马* @description: 测试 impl* @data: 2024-11-21 19:59**/
@Service
public class TestServiceImpl implements TestService {// 省略...@Overridepublic void testResultHandler(HttpServletResponse response) throws IOException {LocalDateTime startDatetime = LocalDateTime.now();// try (MyExportResultHandler handler = new MyExportResultHandler("mysheet1", true, response)) {
// testDao.selectMillionData(handler);
// }BaseExportResultHandler<TestVo> handler = new BaseExportResultHandler<TestVo>("sheetName1", true, response) {/*** 子类重写 fetchDataByStream ,自定义获取数据的方式*/@Overridepublic void fetchDataByStream() {// 这里的this 指的就是 BaseExportResultHandler<TestVo> handler 这个对象,在这里写 mapper 调用获取数据的调用testDao.selectMillionData(this);}};// startExportExcel 方法中调用 fetchDataByStream 方法,// 而 fetchDataByStream 方法中 selectMillionData 方法会调用 handler 中的 handleResult 方法handler.startExportExcel();// 计算两个时间点之间的 DurationDuration duration = Duration.between(startDatetime, LocalDateTime.now());// 获取分钟、秒和毫秒long minutes = duration.toMinutes();long seconds = duration.getSeconds() - minutes * 60;long millis = duration.toMillis() - minutes * 60 * 1000 - seconds * 1000;System.out.printf("数量:100W20行,耗时:%d 分 %d 秒 %d 毫秒\n", minutes, seconds, millis);}
}
导出一波10W行20列的数据,效果如下:
耗时:0 分 17 秒 37 毫秒
四、反射优化
这里使用了反射机制,会消耗性能,
/*** 获取类字段值** @param obj 类实例* @param fieldName 字段名* @return Object* @throws Exception*/public static Object getFieldValueUsingGetter(Object obj, String fieldName) throws Exception {// 获取类对象Class<?> clazz = obj.getClass();// 构造getter方法名String getterName = "get" + Character.toUpperCase(fieldName.charAt(0)) + fieldName.substring(1);// 获取getter方法Method getterMethod = clazz.getMethod(getterName);// 调用getter方法并返回值return getterMethod.invoke(obj);}
使用缓存优化一下,
/*** 获取类字段值** @param obj 类实例* @param fieldName 字段名* @return Object* @throws Exception*/public Object getFieldValueUsingGetter(Object obj, String fieldName) throws Exception {Class<?> clazz = obj.getClass();String getterName = "get" + Character.toUpperCase(fieldName.charAt(0)) + fieldName.substring(1);// 从缓存中获取方法Method getterMethod = methodCache.computeIfAbsent(getterName, k -> {try {return clazz.getMethod(k);} catch (NoSuchMethodException e) {throw new RuntimeException(e);}});// 调用getter方法并返回值return getterMethod.invoke(obj);}
数量:100W20行,耗时:3 分 35 秒 792 毫秒,生成文件大小约 485M。而同样的数据量级下,EasyExcel 需要 4分20秒左右。
可以看到 CPU 跟堆内存使用情况表现良好,
五、导出 Jar 包
新建一个 Maven 项目,
1、修改外部库加载方式
在Java中,System.loadLibrary 方法只能加载系统库路径或 java.library.path 中的库,而不能直接从JAR包内部加载DLL文件。为了从JAR包内加载DLL,我们需要先将DLL提取到一个临时目录,然后再加载它。
/*** @version: V1.0* @author: 余衫马* @description: 从 Jar 包内加载外部库* @data: 2024-12-06 13:46**/
public class NativeUtils {public static void loadLibraryFromJar(String path) throws IOException {// 获取操作系统的临时目录String tempDir = System.getProperty("java.io.tmpdir");File tempFile = new File(tempDir, new File(path).getName());try (InputStream is = NativeUtils.class.getResourceAsStream(path);FileOutputStream fos = new FileOutputStream(tempFile)) {if (is == null) {throw new IllegalArgumentException("File " + path + " was not found inside JAR.");}byte[] buffer = new byte[1024];int bytesRead;while ((bytesRead = is.read(buffer)) != -1) {fos.write(buffer, 0, bytesRead);}}System.load(tempFile.getAbsolutePath());}public static void loadLibrary(String libName) throws IOException {String osName = System.getProperty("os.name").toLowerCase();String libPath = "/libs/";if (osName.contains("win")) {libPath += "windows/";} else if (osName.contains("mac")) {libPath += "macos/";} else if (osName.contains("nix") || osName.contains("nux") || osName.contains("aix")) {libPath += "linux/";} else {throw new UnsupportedOperationException("Unsupported operating system: " + osName);}libPath += "x86_64/";libPath += System.mapLibraryName(libName);loadLibraryFromJar(libPath);}}
/*** @version: V1.0* @author: 余衫马* @description: BaseExcel 导出处理器* @data: 2024-12-03 18:00**/
public abstract class BaseExportResultHandler<T> implements ResultHandler<T> {// 省略...static {try {NativeUtils.loadLibrary("my_excel_writer_lib");} catch (IOException e) {e.printStackTrace();throw new RuntimeException("Failed to load native library", e);}}// 省略...
}
2、重新编译 Rust 库文件
查看支持的 target
rustc --print target-list
分别编译三个平台的库文件,
cargo build --release --target x86_64-pc-windows-msvc
cargo build --release --target x86_64-unknown-linux-gnu
cargo build --release --target x86_64-apple-darwin
win10
cargo build --release --target x86_64-pc-windows-msvc
可以看到 build 成功。
linux
cargo build --release --target x86_64-unknown-linux-gnu
如果编译 x86_64-unknown-linux-gnu 报错,则需要先安装目标架构,
error[E0463]: can't find crate for `core`|= note: the `x86_64-unknown-linux-gnu` target may not be installed= help: consider downloading the target with `rustup target add x86_64-unknown-linux-gnu`For more information about this error, try `rustc --explain E0463`.
error: could not compile `cfg-if` (lib) due to 1 previous error
warning: build failed, waiting for other jobs to finish...
error[E0463]: can't find crate for `std` |= note: the `x86_64-unknown-linux-gnu` target may not be installed= help: consider downloading the target with `rustup target add x86_64-unknown-linux-gnu`
执行指令安装,
# 安装 x86_64-unknown-linux-gnu 目标架构
rustup target add x86_64-unknown-linux-gnu
没有编译环境的话,直接到 Linux 上编译也可以,
这里在 Deepin 系统下 build 成功。
macos
cargo build --release --target x86_64-apple-darwin
如果编译 x86_64-apple-darwin 报错,则需要先安装目标架构,
error[E0463]: can't find crate for `core`|= note: the `x86_64-apple-darwin` target may not be installed= help: consider downloading the target with `rustup target add x86_64-apple-darwin`For more information about this error, try `rustc --explain E0463`.
error: could not compile `cfg-if` (lib) due to 1 previous error
warning: build failed, waiting for other jobs to finish...
error[E0463]: can't find crate for `std`|= note: the `x86_64-apple-darwin` target may not be installed= help: consider downloading the target with `rustup target add x86_64-apple-darwin`
如果是 MacOS 直接安装工具链编译即可,
# rustup toolchain install stable-x86_64-apple-darwin
这里我是 Deepin 系统,即便安装 toolchain 也用不了,不过可以先安装 target,然后交叉编译,
rustup target add x86_64-apple-darwin
直接跑编译会报错,
cargo build --release --target x86_64-apple-darwin
接下来我们安装 MacOS 跨平台编译工具链:
# Step1
git clone https://github.com/tpoechtrager/osxcross
# Step2
cd osxcross
# Step3
wget -nc https://s3.dockerproject.org/darwin/v2/MacOSX10.10.sdk.tar.xz
# Step4
mv MacOSX10.10.sdk.tar.xz tarballs/
# Step5
UNATTENDED=yes OSX_VERSION_MIN=10.10 ./build.sh
需要 clang ,
# Required dependency 'clang' is not installed
# 安装基础构建工具
# sudo apt install build-essential
sudo apt install clang
需要 cmake ,
(base) sam@sam-PC:~/AwesomeWorkSpace/RustStudy/osxcross$ UNATTENDED=yes OSX_VERSION_MIN=10.10 ./build.sh
Required dependency 'cmake' is not installed
sudo apt install cmake
所需环境可以在 readme 文档中看到,
重新执行,
UNATTENDED=yes OSX_VERSION_MIN=10.10 ./build.sh
遇到报错 error while loading shared libraries: libicui18n.so.73
testing i386-apple-darwin14-clang++ -stdlib=libc++ -std=c++11 ... failed (ignored)
testing x86_64-apple-darwin14-clang++ -stdlib=libc++ -std=c++11 ... failed (ignored)testing i386-apple-darwin14-clang ... /home/sam/AwesomeWorkSpace/RustStudy/osxcross/target/bin/i386-apple-darwin14-ld: error while loading shared libraries: libicui18n.so.73: cannot open shared object file: No such file or directory
clang: error: unable to execute command: No such file or directory
clang: error: linker command failed due to signal (use -v to see invocation)
关键在于没有 libicui18n.so 这个库,
sudo apt-get install libicu-dev
系统自带的版本太低了,我们从源码编译安装,
# https://icu.unicode.org/download/73
wget https://github.com/unicode-org/icu/releases/download/release-73-1/icu4c-73_1-src.tgz#编译与安装
tar -xzf icu4c-73_1-src.tgz
cd icu/source
./configure
make
sudo make install
sudo ldconfig
重新执行,
UNATTENDED=yes OSX_VERSION_MIN=10.10 ./build.sh
可以看到已经 build 成功,需要配置系统变量,
sudo vim ~/.bashrc
# 添加路径
export PATH="/home/sam/AwesomeWorkSpace/RustStudy/osxcross/target/bin:$PATH"
# 刷新
source ~/.bashrc
配置交叉编译工具后,回归主线继续编译 x86_64-apple-darwin,修改 Cargo.toml 指定链接器,
[target.x86_64-apple-darwin]
linker = "o64-clang"
ar = "x86_64-apple-darwin-ar"
# 指定交叉编译的编译器
export CC=o64-clang
export CXX=o64-clang++# 告诉 Cargo 使用正确的链接器和存档工具
export CARGO_TARGET_X86_64_APPLE_DARWIN_LINKER=o64-clang
export CARGO_TARGET_X86_64_APPLE_DARWIN_AR=x86_64-apple-darwin-ar# 设置当前会话环境变量
export CC_x86_64_apple_darwin=o64-clang
export CXX_x86_64_apple_darwin=o64-clang++# 设置 SDKROOT 环境变量
# export SDKROOT=/path/to/osxcross/target/SDK/MacOSX10.10.sdk# 清理项目
# cargo clean# 编译
cargo build -Zbuild-std --release --target x86_64-apple-darwin# 查看输出,确保调用了 o64-clang 而不是系统的 cc
# cargo build --target x86_64-apple-darwin -vv
报错版本过高,
warning: deployment target in MACOSX_DEPLOYMENT_TARGET was set to 10.10, but the minimum supported by `rustc` is 10.12
"-mmacosx-version-min=10.12.0"note: osxcross: error: targeted macOS version must be <= 10.10.0 (SDK)osxcross: error: while detecting target
解决:升级 SDK 版本,这里升级到 10.14 版本。
注意:如果是使用自己的 Clang 环境,LLVM 编译的时候需要启用 lld,否则期间不能编译安装 TAPI ,举个例子,
TAPI(Text-based Stubs for System Libraries)是 Apple 用于处理动态库符号的一套工具。在构建 cctools-port(osxcross 使用的工具集)时,如果系统中缺少 libtapi,就会报错 ERROR: Failed TAPI checks in XXX。
# Step1、编译安装 llvm# 下载 llvm-project
wget -nc https://github.com/llvm/llvm-project/releases/download/llvmorg-18.1.8/llvm-project-18.1.8.src.tar.xz
# 解压 llvm-project-18.1.8.src.tar.xz
# cd llvm-project-18.1.8.src# 创建文件夹
mkdir build
cd build
# sudo apt install ninja-build
# 注意将默认的Debug模式换成Release模式.
cmake -G Ninja -DLLVM_ENABLE_PROJECTS=lld -DCMAKE_BUILD_TYPE=Release ../llvm
# 编译安装
sudo cmake --build . -j4 --target installcd ../# Step2、编译安装 clang
# 创建文件夹
mkdir clang-build
cd clang-build
# 注意将默认的Debug模式换成Release模式.
cmake -G Ninja -DLLVM_ENABLE_PROJECTS=lld -DCMAKE_BUILD_TYPE=Release ../clang
# 编译安装
sudo cmake --build . -j4 --target install
3、打包 jar
mvn install
六、使用 Jar 包
在 Resource 文件夹下新建一个 lib 文件夹,把 jar 包放到文件夹下,
添加项目依赖,
现在我们就可以直接使用 jar 包中的方法了,
或者直接在 pom.xml 添加本地 jar 依赖,这适用于 Maven 项目,
<dependency><groupId>com.yushanma</groupId><artifactId>BaseCrazyExcel</artifactId><version>1.0.0</version></dependency>
七、测试效果
Windows 下
1、10W行20列
耗时:0 分 20 秒 625 毫秒
2、50W行20列
耗时:1 分 36 秒 533 毫秒
3、100W行20列
耗时:3 分 23 秒 413 毫秒
补充:远程查看 Linux JVM
Linux 配置策略
找到 Java 主目录,
(base) sam@sam-PC:~$ which java
/usr/bin/java
(base) sam@sam-PC:~$ ls -lrt /usr/bin/java
lrwxrwxrwx 1 root root 22 11月 12 17:38 /usr/bin/java -> /etc/alternatives/java
(base) sam@sam-PC:~$ ls -lrt /etc/alternatives/java
lrwxrwxrwx 1 root root 46 11月 12 17:38 /etc/alternatives/java -> /usr/lib/jvm/java-8-openjdk-amd64/jre/bin/java
在 usr/bin/ 下创建文件,
sudo vim jstatd.all.policy# 写入以下内容
grant codebase "file:/usr/lib/jvm/java-8-openjdk-amd64/lib/tools.jar" {permission java.security.AllPermission;
};
运行 jstatd,
nohup jstatd -J-Djava.security.policy=jstatd.all.policy -p 8066 >/dev/null 2>&1 &# 查看是否启动成功
# ps -ef |grep jstatd
# lsof -i:8066
添加 JVM 启动参数,
-Dcom.sun.management.jmxremote
-Dcom.sun.management.jmxremote.port=XXXX
-Dcom.sun.management.jmxremote.rmi.port=XXXX
-Dcom.sun.management.jmxremote.ssl=false
-Dcom.sun.management.jmxremote.authenticate=false
JVM 调参,
-Xms1024m -Xmx10240m -Xmn3840m -Xss512k
java -Xms1024m -Xmx8192m -Xmn3840m -Xss512k -Djava.io.tmpdir=./tmp -Djava.rmi.server.hostname=0.0.0.0 -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=8067 -Dcom.sun.management.jmxremote.rmi.port=8067 -Dcom.sun.management.jmxremote.authenticate=false -Dcom.sun.management.jmxremote.ssl=false -jar CrazyExcel-0.0.1-SNAPSHOT.jar
在本地电脑打 jdk下的bin目录启动 jvisualvm.exe,
添加远程主机,输入 ip,
添加 JMX 连接,输入端口,
这样就可以连接远程主机监控 JVM 状态了。如果是本地虚拟机,则使用内网 ip 连接:
相关文章:
基于 JNI + Rust 实现一种高性能 Excel 导出方案(下篇)
衡量一个人是否幸福,不应看他有多少高兴的事,而应看他是否为小事烦扰。只有幸福的人,才会把无关痛痒的小事挂心上。那些真正经历巨大灾难和深重痛苦的人,根本无暇顾及这些小事的。因此人们往往在失去幸福之后,才会发现…...
关于Python程序消费Kafka消息不稳定问题的处理方法
在使用Python程序消费Kafka消息的过程中,有时会遇到各种不稳定的情况,如自动提交偏移量无效、CommitFailedError错误等。这些问题不仅影响了数据处理的可靠性,还可能导致重复消费或丢失消息。本文将针对这两个常见问题提供详细的解决方案和最…...
【OpenCV】Canny边缘检测
理论 Canny 边缘检测是一种流行的边缘检测算法。它是由 John F. Canny 在 1986 年提出。 这是一个多阶段算法,我们将介绍算法的每一个步骤。 降噪 由于边缘检测易受图像中的噪声影响,因此第一步是使用 5x5 高斯滤波器去除图像中的噪声。我们在前面的章…...
算法-二进制和位运算
一.二进制 (1).无符号数: 无符号数是一种数据表示方式,它只表示非负整数,即没有符号位,所有的位都用来表示数值大小。在 C 等编程语言中,常见的无符号类型有 unsigned int、unsigned char 等。…...
OpenAI Chatgpt 大语言模型
OpenAI 一个美国人工智能研究实验室,由非营利组织 OpenAI Inc,和其营利组织子公司 OpenAI LP 所组成。该组织于 2015 年由萨姆阿尔特曼、里德霍夫曼、杰西卡利文斯顿、伊隆马斯克、伊尔亚苏茨克维、沃伊切赫萨伦巴、彼得泰尔等人在旧金山成立࿰…...
SpringBoot【九】mybatis-plus之自定义sql零基础教学!
一、前言🔥 环境说明:Windows10 Idea2021.3.2 Jdk1.8 SpringBoot 2.3.1.RELEASE mybatis-plus的基本使用,前两期基本讲的差不多,够日常使用,但是有的小伙伴可能就会抱怨了,若是遇到业务逻辑比较复杂的sq…...
C#,人工智能,深度学习,目标检测,OpenCV级联分类器数据集的制作与《层级分类器一键生成器》源代码
一、目标识别技术概述 1、摘要 目标检测是计算机视觉中最基本和最具挑战性的问题之一,它试图从自然图像中的大量预定义类别中定位目标实例。深度学习技术已成为直接从数据中学习特征表示的强大策略,并在通用目标检测领域取得了显著突破。鉴于这一快速发…...
调度系统:Luigi 的主要特性和功能
Luigi 是一个开源的 Python 工作流管理工具,用于构建批处理作业管道,特别适用于数据工程领域。它被设计用来编排任务和处理任务间的依赖关系,支持自动化复杂的 ETL 流程、数据分析、模型训练等任务。 Luigi 的主要特性和功能: 任…...
C# 探险之旅:第二节 - 定义变量与变量赋值
欢迎再次踏上我们的C#学习之旅。今天,我们要聊一个超级重要又好玩的话题——定义变量与变量赋值。想象一下,你正站在一个魔法森林里,手里拿着一本空白的魔法书(其实就是你的代码编辑器),准备记录下各种神奇…...
AUTOSAR:SOME/IP 概念
文章目录 1. 用例与需求1.1 典型用例1.2 对中间件的要求 2. 协议栈示例3. SOME/IP 概念3.1 中间件整体功能与架构3.2 服务组成元素详细解释 4. 服务发现机制深入剖析5. 总结 1. 用例与需求 1.1 典型用例 信息娱乐系统: 后座娱乐系统连接:允许后排乘客连…...
循序渐进kubenetes Service(Cluster ip、Nodeport、Loadbalancer)
文章目录 部署一个web服务Kubernetes Port ForwardKubernetes ServicesClusterIP ServiceNodePort ServiceLoadBalancer Service 部署一个web服务 准备 Kubernetes 集群后,创建一个名为 web 的新 namespace,然后在该 namespace 中部署一个简单的 web 应…...
深入理解 Apache Shiro:安全框架全解析
亲爱的小伙伴们😘,在求知的漫漫旅途中,若你对深度学习的奥秘、JAVA 、PYTHON与SAP 的奇妙世界,亦或是读研论文的撰写攻略有所探寻🧐,那不妨给我一个小小的关注吧🥰。我会精心筹备,在…...
mac 安装CosyVoice (cpu版本)
CosyVoice 介绍 CosyVoice 是阿里研发的一个tts大模型 官方项目地址:https://github.com/FunAudioLLM/CosyVoice.git 下载项目(非官方) git clone --recursive https://github.com/v3ucn/CosyVoice_for_MacOs.git 进入项目 cd CosyVoic…...
币安移除铭文市场的深度解读:背后原因及其对区块链行业的影响
引言: 就在昨天,2024年12月10号,币安宣布将移除铭文市场(Inscriptions Market)。这一消息引发了全球加密货币社区的广泛关注,尤其是在比特币NFT和数字收藏品市场快速发展的背景下。铭文市场自诞生以来迅速…...
深度学习实战野生动物识别
本文采用YOLOv11作为核心算法框架,结合PyQt5构建用户界面,使用Python3进行开发。YOLOv11以其高效的实时检测能力,在多个目标检测任务中展现出卓越性能。本研究针对野生动物数据集进行训练和优化,该数据集包含丰富的野生动物图像样…...
windows安装使用conda
在Windows系统上安装和使用Conda的详细步骤如下: 一、下载Conda安装包 访问Conda的官方网站Anaconda | The Operating System for AI,点击“Downloads”按钮。在下载页面,选择适合您系统的安装包。通常,对于Windows系统…...
手机租赁系统开发全流程解析与实用指南
内容概要 在如今快速发展的科技时代,手机租赁系统已经成为一种新兴的商业模式,非常符合当下市场需求。那么,在开发这样一个系统的时候,首先要从需求分析和市场调研开始。在这一阶段,你需要了解用户需要什么࿰…...
SpringBoot 开发—— YAML文件深度分析
文章目录 一、YAML概述二、数据表示三、YAML 的语法四、YAML 的应用五、YAML 与其他格式的比较1、YAML vs .properties文件可读性和结构数据类型支持扩展性和灵活性使用场景性能和支持2、YAML vs. JSON3、YAML vs. XML六、使用 YAML 的注意事项七、总结YAML 是非常流行的一种配…...
复合机器人整体解决方案
复合机器人是一种集成移动机器人和协作机器人两项功能为一身的新型机器人,更符合人们想象中“脑、眼、手、脚”融合的机器人终极形态。复合机器人的整体解决方案通常涉及多个方面,包括机器人本体、控制系统、感知系统、执行系统以及周边配套设备等。以下…...
【Oracle11g SQL详解】日期和时间函数:SYSDATE、TO_DATE、TO_CHAR 等
日期和时间函数:SYSDATE、TO_DATE、TO_CHAR 等 在 Oracle 数据库中,日期和时间函数用于处理日期和时间数据。它们在记录创建时间、分析时间间隔、格式化输出等场景中非常重要。本文将详细讲解常用的日期和时间函数及其应用。 一、SYSDATE:获…...
VSCode设置字体
参考文章:【面向小白】vscode最佳实践(2)—— 字体设置(fira code更纱黑体),这篇文章末尾给了安装字体的链接。 配置的字体还是很好看的。 ‘Fira Code Retina’, ‘Sarasa Mono Sc’ 需要注意的一个点&am…...
shell编程入门之提取字符并设置rtc时间
awk用法 awk是一款文本处理工具,通常在Unix和Linux操作系统中使用,用于以行为单位对文本进行处理和操作。它可以读取输入文本,对其进行处理,生成报表、统计信息等,并将结果输出到标准输出设备中。 它主要有以下特点&…...
react 不可变数据更新(Immutable Update)合并对象 类似与Java 的BeanUtils.copyProperties
{ ...state, // 保留原有的 state 的其他部分data: { ...state.data, // 保留 state.data 中的其他字段...action.payload // 使用 action.payload 覆盖 state.data 中需要更新的字段} }这段代码是 Redux 中常见的一种状态更…...
Linux GCC基础用法⑦
在 CentOS 7 系统中使用 GCC 与编写 99 乘法表 一、GCC 简介 GCC(GNU Compiler Collection)是一套功能强大的编程语言编译器,在 CentOS 7 系统中广泛用于编译 C、C等多种编程语言的程序。它能够将源代码转换为可执行文件,让计算…...
PyTorch 切片运算 (Slice Operator)
PyTorch 切片运算 {Slice Operator} 1. [:, -1, :]2. [:, [-1], :]References 1. [:, -1, :] https://github.com/karpathy/llama2.c/blob/master/model.py import torchlogits torch.arange(1, 16) print("logits.shape:", logits.shape) print("logits:\n&…...
SpringSecurity Oauth2 -账号密码实现多因子身份认证
1. 密码策略问题 CREATE TABLE t_storage (id bigint(20) NOT NULL AUTO_INCREMENT COMMENT 自增主键,nameSpace varchar(64) NOT NULL COMMENT 隔离字段,groupId varchar(128) NOT NULL COMMENT 分组,比如不同app,dataId varchar(64) NOT NULL COMMENT 数据存储id…...
【CSS in Depth 2 精译_071】11.4 思考字体颜色的对比效果 + 11.5 本章小结
当前内容所在位置(可进入专栏查看其他译好的章节内容) 第四部分 视觉增强技术 ✔️【第 11 章 颜色与对比】 ✔️ 11.1 通过对比进行交流 11.1.1 模式的建立11.1.2 还原设计稿 11.2 颜色的定义 11.2.1 色域与色彩空间11.2.2 CSS 颜色表示法 11.2.2.1 RGB…...
Y3编辑器文档4:触发器1(对话、装备、特效、行为树、排行榜、不同步问题)
文章目录 一、触发器简介1.1 触发器界面1.2 ECA语句编辑及快捷键1.3 参数设置1.4 变量设置1.5 实体触发器1.6 函数库与触发器复用 二、触发器的多层结构2.1 子触发器(在游戏内对新的事件进行注册)2.2 触发器变量作用域2.3 复合条件2.4 循环2.5 计时器2.6…...
趣味编程:猜拳小游戏
1.简介 这个系列的第一篇以猜拳小游戏开始,这是源于我们生活的灵感,在忙碌的时代中,我们每个人都在为自己的生活各自忙碌着,奔赴着自己所走向的那条路上,即使遍体鳞伤。 但是,生活虽然很苦,也不…...
软件工程 概述
软件 不仅仅是一个程序代码。程序是一个可执行的代码,它提供了一些计算的目的。 软件被认为是集合可执行的程序代码,相关库和文档的软件。当满足一个特定的要求,就被称为软件产品。 工程 是所有有关开发的产品,使用良好定义的&…...
浏览器做单页网站项目/百度排名优化工具
安装 Linux 主机时,如果选择 最小化安装! 配置 vnc 远程桌面可以参考:Linux 配置 VNC 远程桌面 使用 vnc 等工具连接通常显示如下: 也就是无法使用图形化界面,可以通过 yum 直接安装图形化界面: Linux 6: yum groupinstall -y "X Window System" yum gr…...
吉林省吉林市疫情风险等级/抖音seo优化排名
在第一篇教程中有提到MP中的各种元素。 最近今天一直在做MP,出了不少的问题,也对Mp也有更深的了解。 今天说说做MP包的思路问题: 1、如果你要反馈的信息在Mp的类库中没有原始类的话,你就需要定义一个类(ClassType&…...
轻量级WordPress/seo入门培训
对这段时间和以后做个了结 这段时间太多的杂乱思维,太多的浮躁和迷茫与不安。我写下这些,希望对于同样IT职场的小白们有所帮助。 先来说说这段时间的颓废。感觉这大半年都没什么进步,过的也不好不坏,不痛不痒。整天来公司…...
网站开发云南/广州各区最新动态
java环境变量配置新建系统变量JAVA_HOME 和CLASSPATH 变量名:JAVA_HOME 变量值:C:\Program Files\Java\jdk1.7.0_75变量名:CLASSPATH 变量值:.;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar;Path添加变量置变量名࿱…...
seo短视频网页入口引流在线看/木卢seo教程
系统进程管理 进程:进程是程序的一次动态执行守护进程:守护进程是在后台运行并提供系统服务的一些进程 top : 动态显示当前进程标签含义PID进程号PR、NI控制用户优先级,值越高,优先级越低VIRT虚拟内存S状态( S…...
读书郎营销网站/百度广告联盟赚广告费
字符串操作C语言提供了较多的库函数,本题目要求代码中不能使用字符串操作相关的库函数,可以使用malloc。 用例中可以使用中提供的库函数。 实现接口,每个接口实现1个基本操作: unsigned int strlenth(char*s):计算字…...