Java调用native本地方法实例:控制台下的中英文字符对齐问题
liebian365 2024-10-30 04:50 18 浏览 0 评论
小伙伴们在初学Java的时候一般都是采用Eclipse或其他IDE环境,中英文混合时的对齐问题想必都或多或少地困扰过大家。
在线上训练营的项目里,很多小伙伴也都反映说遇到了这个问题,也太烧脑了吧!
于是窖头花了2天时间,专门来研究这个问题,大雄强烈推荐强迫症患者服用哦~
比如下面的代码和在Eclipse中的显示效果:
Java字符串格式构建代码:
public String toString() { String str = String.format("%-8s%-4d\t%-8s\t%.2f", name, level, getLevelName(), face); return str; }
跟我们设想的并不一样。网上有个比较简单的解决方案,就是在%s后添加\t:
public String toString() { String str = String.format("%-8s\t%-4d\t%-8s\t%.2f", name, level, getLevelName(), face); return str; }
效果如下:
好了,对于没有强迫症的小伙伴,本文结束,大家按照上面的解决方案修改代码即可。
二、使用JNI调用C/C++实现中英文对齐
JNI,即Java Native Interface,Java本地接口。是Java平台提供的调用本地C/C++代码进行互操作的API。
2.1 本次示例所用的代码如下:
/** * 后宫佳丽 * @author 老九学堂·窖头 * */ public class Beauty { private static String[] levelNames = {"秀女", "答应", "常在", "贵人", "嫔", "妃", "贵妃", "皇贵妃", "皇后", "皇太后", "太皇太后", "太皇太后还要往后"}; private String name; private int level; private String levelName; private double face; //颜值,可以通过图像AI获取 public Beauty(String name, int level, double face) { this.name = name; setLevel(level); this.levelName = getLevelName(); this.face = face; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getLevel() { return level; } public void setLevel(int level) { if(level < 0 || level > levelNames.length) { this.level = 1; return; } this.level = level; } public String getLevelName() { if(level < 0 || level > levelNames.length) { return levelNames[0]; } return levelNames[level - 1]; } public void setLevelName(String levelName) { this.levelName = levelName; } public double getFace() { return face; } public void setFace(double face) { this.face = face; } } /** * 使用单例模式的打印类 * @author 窖头 * */ public class Printer { private static Printer printer = null; private Printer() {} /** * 调用native方法打印后宫佳丽的信息 * @param beauty */ public native void printf(Beauty beauty); public static Printer getInstance() { if(null == printer) { printer = new Printer(); } return printer; } }
下图是在Eclipse中创建的工程和class:
2.2 命令行下执行javah命令,得到包含该本地方法声明的头文件(.h文件)
win+r -> cmd,进入工程根目录的bin目录,输入以下指令:
//包名及类名请根据自己的定义进行修改 javah -jni com.xuetang9.kenny.util.Printer
这里如果出现错误,请检查并重新配置Java的环境变量
获得头文件:com_xuetang9_kenny_util_Printer.h
头文件以包名_方法名的方式命名,内容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_xuetang9_kenny_util_Printer */ #ifndef _Included_com_xuetang9_kenny_util_Printer #define _Included_com_xuetang9_kenny_util_Printer #ifdef __cplusplus extern "C" { #endif /* * Class: com_xuetang9_kenny_util_Printer * Method: printf * Signature: (Lcom/xuetang9/kenny/entity/Beauty;)V */ JNIEXPORT void JNICALL Java_com_xuetang9_kenny_util_Printer_printf (JNIEnv *, jobject, jobject); /** 自定义函数:将Java传来的字符串转换为GB2312以便显示 */ char* jstringToWindows(JNIEnv *, jstring); /** 自定义函数:将gb2312转换为UTF8/16,以便传回给Java能够正常显示 */ jstring WindowsTojstring(JNIEnv* env, const char * ); //关于为什么使用两个自定义转换函数请参见:http://wiki.xuetang9.com/?p=5270 #ifdef __cplusplus } #endif #endif
2.3 下面根据头文件,书写C++代码,实现本地方法
在头文件旁创建C++源文件:com_xuetang9_kenny_util_Printer.cpp
文件名不变,后缀名修改为cpp,实现代码如下:
#include "com_xuetang9_kenny_util_Printer.h" #include <stdio.h> #include <iostream> #include <iomanip> #include <stdlib.h> #include <malloc.h> #include <memory.h> #include <Windows.h> using namespace std; JNIEXPORT void JNICALL Java_com_xuetang9_kenny_util_Printer_printf(JNIEnv * env, jobject jobj, jobject jbeauty) { jclass beautyClass = env->GetObjectClass(jbeauty); //获得Java传来的后宫佳丽对象 //获得属性句柄(ID) jfieldID nameFid = env->GetFieldID(beautyClass, "name", "Ljava/lang/String;"); jfieldID levelFid = env->GetFieldID(beautyClass, "level", "I"); //整型为I,double类型为D jfieldID levelNameFid = env->GetFieldID(beautyClass, "levelName", "Ljava/lang/String;"); jfieldID faceFid = env->GetFieldID(beautyClass, "face", "D"); //获得name属性的值 jstring jNameField = (jstring)env->GetObjectField(jbeauty, nameFid); jint jLevelField = (jint)env->GetIntField(jbeauty, levelFid); jstring jLevelNameField = (jstring)env->GetObjectField(jbeauty, levelNameFid); jdouble jFaceField = env->GetDoubleField(jbeauty, faceFid); //const char * cNameField = env->GetStringUTFChars(jNameField, NULL); //const char * cLevelNameField = env->GetStringUTFChars(jLevelNameField, NULL); //C++中的打印格式控制,左对齐,单独设置每个元素的宽度 cout.setf(ios::left); cout << setw(12) << jstringToWindows(env, jNameField); cout << setw(4) << jLevelField; cout << setw(8) << jstringToWindows(env, jLevelNameField); cout << setw(7) << jFaceField << endl; //释放字符串所占的空间 //env->ReleaseStringUTFChars(jNameField, NULL); //env->ReleaseStringUTFChars(jLevelNameField, cLevelNameField); } //字符串转换函数,了解做什么的即可 /** * 将Java传来的UTF8/16编码转换为C/C++能够正常显示的GB2312编码 */ char* jstringToWindows( JNIEnv *env, jstring jstr ) { int length = (env)->GetStringLength(jstr); const jchar* jcstr = (env)->GetStringChars(jstr, 0); char* rtn = (char*)malloc(length*2 + 1); int size = 0; size = WideCharToMultiByte( CP_ACP, 0, (LPCWSTR)jcstr, length, rtn,(length*2+1), NULL, NULL); if( size <= 0 ) return NULL; (env)->ReleaseStringChars(jstr, jcstr); rtn[size] = 0; return rtn; } /** * 将C/C++中的GB2312编码转换成UTF8/16编码 */ jstring WindowsTojstring( JNIEnv* env, const char* str ) { jstring rtn = 0; int slen = strlen(str); unsigned short * buffer = 0; if( slen == 0 ) { rtn = (env)->NewStringUTF(str ); } else { int length = MultiByteToWideChar( CP_ACP, 0, (LPCSTR)str, slen, NULL, 0 ); buffer = (unsigned short *)malloc( length*2 + 1 ); if( MultiByteToWideChar( CP_ACP, 0, (LPCSTR)str, slen, (LPWSTR)buffer, length ) >0 ) rtn = (env)->NewString( (jchar*)buffer, length ); } if(buffer) free( buffer ); return rtn; }
2.4 使用Gcc编译生成共享库dll文件
配置好MinGw的环境变量后,键入下面的命令:
g++ -m64 -static-libgcc -static-libstdc++ -I"C:\Program Files\Java\jdk1.8.0_201\include" -I"C:\Program Files\Java\jdk1.8.0_201\include\win32" -shared -o Printer.dll com_xuetang9_kenny_util_Printer.cpp
1、路径C:\Program Files\Java\jdk1.8.0_201\include和
C:\Program Files\Java\jdk1.8.0_201\include\win32
分别包含了JNI的头文件,<jni.h>和<jni_md.h>,请大家根据自己机器配置的不同,自行修改路径
2、-m64表示生成64位dll库文件
2.5 在Java中调用本地库文件
书写Java测试类:
import java.io.File; import com.xuetang9.kenny.entity.Beauty; import com.xuetang9.kenny.util.Printer; public class TestPrinter { public static void main(String[] args) { //请大家自行修改成自己机器的路径 String path = "C:\\Users\\窖头\\eclipse-workspace\\PrintMsgByCpp\\bin\\Printer.dll"; File file = new File(path); //加载本地dll库 System.load(file.getAbsolutePath()); Beauty[] beauties = new Beauty[5]; for(int i = 0; i < beauties.length; i++) { beauties[i] = new Beauty(); } beauties[0] = new Beauty("貂蝉1号", 5, 86.25); beauties[1] = new Beauty("a赵飞燕b", 6, 76.25); beauties[2] = new Beauty("ab西施bc", 7, 56.25); beauties[3] = new Beauty("北岸初晴", 8, 66.25); beauties[4] = new Beauty("龙a女d", 9, 96.25); for(int i = 0; i < beauties.length; i++) { //调用本地C++方法打印对象的内容 Printer.getInstance().printf(beauties[i]); } } }
如果直接在Eclipse中运行这个main方法,会抛出异常:java.lang.UnsatisfiedLinkError: %1 不是有效的 Win32 应用程序
反正未来我们开发完成的程序也不可能在Eclipse中执行,所以我们直接在控制台下执行并观察
结果:
java com.xuetang9.kenny.TestPrinter
显示效果非常完美,大功告成!
需要更多学习笔记干货的小伙伴、欢迎关注公众号【老九学堂】(づ ̄3 ̄)づ╭?~
相关推荐
- “版本末期”了?下周平衡补丁!国服最强5套牌!上分首选
-
明天,酒馆战棋就将迎来大更新,也聊了很多天战棋相关的内容了,趁此机会,给兄弟们穿插一篇构筑模式的卡组推荐!老规矩,我们先来看10职业胜率。目前10职业胜率排名与一周前基本类似,没有太多的变化。平衡补丁...
- VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符"
-
首先,程序中头文件的选择,要选择头文件,在文件中是没有对M_PI的定义的。选择:项目——>”XXX属性"——>配置属性——>C/C++——>预处理器——>预处理器定义,...
- 东营交警实名曝光一批酒驾人员名单 88人受处罚
-
齐鲁网·闪电新闻5月24日讯酒后驾驶是对自己和他人生命安全极不负责的行为,为守护大家的平安出行路,东营交警一直将酒驾作为重点打击对象。5月23日,东营交警公布最新一批饮酒、醉酒名单。对以下驾驶人醉酒...
- Qt界面——搭配QCustomPlot(qt platform)
-
这是我第一个使用QCustomPlot控件的上位机,通过串口精确的5ms发送一次数据,再将读取的数据绘制到图表中。界面方面,尝试卡片式设计,外加QSS简单的配了个色。QCustomPlot官网:Qt...
- 大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写
-
老友相聚,仗剑江湖!《大话西游2》2021全民PK季4月激燃打响,各PK玩法鏖战齐开,零门槛参与热情高涨。PK季期间,不仅各种玩法奖励丰厚,参与PK趣闻录活动,投稿自己在PK季遇到的趣事,还有机会带走...
- 测试谷歌VS Code AI 编程插件 Gemini Code Assist
-
用ClaudeSonnet3.7的天气测试编码,让谷歌VSCodeAI编程插件GeminiCodeAssist自动编程。生成的文件在浏览器中的效果如下:(附源代码)VSCode...
- 顾爷想知道第4.5期 国服便利性到底需优化啥?
-
前段时间DNF国服推出了名为“阿拉德B计划”的系列改版计划,截至目前我们已经看到了两项实装。不过关于便利性上,国服似乎还有很多路要走。自从顾爷回归DNF以来,几乎每天都在跟我抱怨关于DNF里面各种各样...
- 掌握Visual Studio项目配置【基础篇】
-
1.前言VisualStudio是Windows上最常用的C++集成开发环境之一,简称VS。VS功能十分强大,对应的,其配置系统较为复杂。不管是对于初学者还是有一定开发经验的开发者来说,捋清楚VS...
- 还嫌LED驱动设计套路深?那就来看看这篇文章吧
-
随着LED在各个领域的不同应用需求,LED驱动电路也在不断进步和发展。本文从LED的特性入手,推导出适合LED的电源驱动类型,再进一步介绍各类LED驱动设计。设计必读:LED四个关键特性特性一:非线...
- Visual Studio Community 2022(VS2022)安装图文方法
-
直接上步骤:1,首先可以下载安装一个VisualStudio安装器,叫做VisualStudioinstaller。这个安装文件很小,很快就安装完成了。2,打开VisualStudioins...
- Qt添加MSVC构建套件的方法(qt添加c++11)
-
前言有些时候,在Windows下因为某些需求需要使用MSVC编译器对程序进行编译,假设我们安装Qt的时候又只是安装了MingW构建套件,那么此时我们该如何给现有的Qt添加一个MSVC构建套件呢?本文以...
- Qt为什么站稳c++GUI的top1(qt c)
-
为什么现在QT越来越成为c++界面编程的第一选择,从事QT编程多年,在这之前做C++界面都是基于MFC。当时为什么会从MFC转到QT?主要原因是MFC开发界面想做得好看一些十分困难,引用第三方基于MF...
- qt开发IDE应该选择VS还是qt creator
-
如果一个公司选择了qt来开发自己的产品,在面临IDE的选择时会出现vs或者qtcreator,选择qt的IDE需要结合产品需求、部署平台、项目定位、程序猿本身和公司战略,因为大的软件产品需要明确IDE...
- Qt 5.14.2超详细安装教程,不会来打我
-
Qt简介Qt(官方发音[kju:t],音同cute)是一个跨平台的C++开库,主要用来开发图形用户界面(GraphicalUserInterface,GUI)程序。Qt是纯C++开...
- Cygwin配置与使用(四)——VI字体和颜色的配置
-
简介:VI的操作模式,基本上VI可以分为三种状态,分别是命令模式(commandmode)、插入模式(Insertmode)和底行模式(lastlinemode),各模式的功能区分如下:1)...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- “版本末期”了?下周平衡补丁!国服最强5套牌!上分首选
- VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符"
- 东营交警实名曝光一批酒驾人员名单 88人受处罚
- Qt界面——搭配QCustomPlot(qt platform)
- 大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写
- 测试谷歌VS Code AI 编程插件 Gemini Code Assist
- 顾爷想知道第4.5期 国服便利性到底需优化啥?
- 掌握Visual Studio项目配置【基础篇】
- 还嫌LED驱动设计套路深?那就来看看这篇文章吧
- Visual Studio Community 2022(VS2022)安装图文方法
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)