Java 之 实现C++库函数等价函数遇到的问题

背景

项目中之前使用到的库大部分是C++编写的,后面因业务需要,要求把部分库用java实现,期间遇到一些问题,这里记录下

问题

写入字符数组类型时,数组大小不同问题

java 复制代码
 //C++源代码
 char fileName_[100];
 fwrite(fileName_, sizeof(fileName_), 1, pf);
// 问题:因该写入100字节,但是写入时发现只按照实际大小写入
//java 修改后的方法
 this.fileName = new byte[100]; // 固定大小为100字节
 //这里是入参
 byte[] nameBytes = fileName.getBytes(StandardCharsets.UTF_8);
 //这句是关键,这样再写入时,就是100字节了
 System.arraycopy(nameBytes, 0, this.fileName, 0, Math.min(nameBytes.length, this.fileName.length));

写入整数类型时,遇到大小端问题

C++‌:其数据在内存中的存储顺序(字节序)‌直接依赖于底层CPU架构和编译器的实现‌。例如,在x86架构上,C++程序默认使用小端模式存储数据,而在PowerPC等架构上则可能使用大端模式。因此,C++开发者需要关注并处理跨平台或跨系统通信时的字节序转换问题。

Java‌:其设计目标是"一次编写,到处运行",因此‌在语言层面统一规定使用大端(Big-Endian)字节序‌,也称为网络字节序。Java虚拟机(JVM)负责屏蔽底层硬件差异,对所有基本数据类型(如int、float)的存储和网络传输都采用大端模式,开发者通常无需关心底层硬件的字节序。

java 复制代码
//java中按小端写入整数
private static void writeLittleInt(RandomAccessFile fos, int value) {
    try {
        // 分配4字节缓冲区
        ByteBuffer buffer = ByteBuffer.allocate(4);
        // 设置为小端序
        buffer.order(ByteOrder.LITTLE_ENDIAN);
        // 将整数写入缓冲区
        buffer.putInt(value);
        // 写入文件
        fos.write(buffer.array());
    } catch (Exception e) {
        e.printStackTrace();
    }
}

日志记录

如果不想依赖第三方日志库,可以直接使用jdk提供的java.util.logging

java 复制代码
import java.util.logging.*;
import java.nio.file.Files;
import java.text.SimpleDateFormat;
import java.util.Date;
/**
 * 日志记录器
 */
private static final Logger logger = Logger.getLogger(TestLib.class.getName());
static {
    try {
        // 日志文件名称
        String logFileName = "mylib.log";
        // 使用项目根目录下的 logs 文件夹
        String projectRoot = new File("").getAbsolutePath();
        String logDirPath = projectRoot + File.separator + "logs";
        Path logDir = Paths.get(logDirPath);
        if (!logDir.toFile().exists()){
            // 确保目录存在
            Files.createDirectories(logDir);
        }
        // true表示追加模式
        FileHandler fileHandler = new FileHandler(logDirPath + File.separator + logFileName, true);
        fileHandler.setFormatter(new Formatter() {
            @Override
            public String format(LogRecord record) {
                return String.format("[%s] %s: %s%n",
                        new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date(record.getMillis())),
                        record.getLevel(),
                        record.getMessage());
            }
        });
        // 添加到当前类的 logger
        logger.addHandler(fileHandler);
        logger.setLevel(Level.INFO);
        logger.setUseParentHandlers(false);

        // 可选:也添加到根日志记录器
        Logger rootLogger = Logger.getLogger("");
        rootLogger.addHandler(fileHandler);

        logger.info("Log file initialization successful: log file: " + logFileName);

    } catch (Exception e) {
        System.err.println("Log file init failed: " + e.getMessage());
    }
}

这样,当你调用logger.info函数写日志时,当前程序目录下的logs/mylib.log文件就有内容了。

总结

这里没有涉及到整个结构体的序列化写入,只是基础类型的操作。

相关推荐
小bo波4 小时前
使用Thread子类创建线程 VS 使用Runnable接口创建线程的区别
java·多线程·thread·并发编程·runnable
SamDeepThinking5 小时前
高并发场景下,CompletableFuture与ForkJoinPool该如何取舍?
java·后端·面试
张不才8 小时前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1119 小时前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev12 小时前
单例模式 → object 声明
android·java·kotlin
用户2986985301413 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
SimonKing14 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员
咖啡八杯1 天前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户128526116021 天前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java