Android App开发之利用JNI实现加密和解密操作实战(附源码 简单易懂)

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介: Android App开发之利用JNI实现加密和解密操作实战(附源码 简单易懂)

运行有问题或需要源码请点赞关注收藏后评论区留言~~~

一、JNI实现加密和解密

在实际开发中 JNI主要应用于以下场景

1:对关键业务数据进行加密和解密

Java代码容易遭到破解,JNI加密更加安全

2:底层的网络操作与设备操作

Java作为一门高级程序设计语言 与硬件和网络操作的隔阂比C/C++大,它不想它俩那样容易驾驭硬件和网络的操作

3:对运行效率要求较高的场合

同样的操作C/C++执行效率比Java高很多,另外,图像处理,音视频处理等需要大量运算的场合,其底层算法也都是用C/C++实现。

4:跨平台的应用移植

移动设备的操作系统不是Android就是IOS,如果部分业务功能采用C/C++实现,那么不但Android可以通过JNI调用,而且IOS能直接编译运行,一份代码可以同时被两个平台复用,省时省力

接下来尝试使用JNI完成加解密操作,采用的是AES算法C++的开源代码,主要的改造工作是给C++源代码配上JNI接口

效果如下 分别输入原始字符串并调用JNI接口进行加密,并且对已加密的字符串进行JNI解密操作

代码如下

Java类

package com.example.ebook;
import android.os.Bundle;
import android.text.TextUtils;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
public class JniSecretActivity extends AppCompatActivity {
    private EditText et_origin; // 声明一个用于输入原始字符串的编辑框对象
    private TextView tv_encrypt; // 声明一个文本视图对象
    private TextView tv_decrypt; // 声明一个文本视图对象
    private String mKey = "123456789abcdef"; // 该算法要求密钥串的长度为16位
    private String mEncrypt; // 加密串
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_jni_secret);
        et_origin = findViewById(R.id.et_origin);
        tv_encrypt = findViewById(R.id.tv_encrypt);
        tv_decrypt = findViewById(R.id.tv_decrypt);
        findViewById(R.id.btn_encrypt).setOnClickListener(v -> {
            // 调用JNI方法encryptFromJNI获得加密后的字符串
            mEncrypt = encryptFromJNI(et_origin.getText().toString(), mKey);
            tv_encrypt.setText("jni加密结果为:"+mEncrypt);
        });
        findViewById(R.id.btn_decrypt).setOnClickListener(v -> {
            if (TextUtils.isEmpty(mEncrypt)) {
                Toast.makeText(this, "请先加密后再解密", Toast.LENGTH_SHORT).show();
                return;
            }
            // 调用JNI方法decryptFromJNI获得解密后的字符串
            String raw = decryptFromJNI(mEncrypt, mKey);
            tv_decrypt.setText("jni解密结果为:"+raw);
        });
    }
    // 声明encryptFromJNI是来自于JNI的原生方法
    public native String encryptFromJNI(String raw, String key);
    // 声明decryptFromJNI是来自于JNI的原生方法
    public native String decryptFromJNI(String des, String key);
    // 在加载当前类时就去加载libcommon.so,加载动作发生在页面启动之前
    static {
        System.loadLibrary("common");
    }
}

XML文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:focusable="true"
    android:focusableInTouchMode="true"
    android:orientation="vertical"
    android:padding="5dp" >
    <EditText
        android:id="@+id/et_origin"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:padding="5dp"
        android:background="@drawable/editext_selector"
        android:inputType="text"
        android:hint="请输入待加密的字符串"
        android:textColor="@color/black"
        android:textSize="17sp" />
    <Button
        android:id="@+id/btn_encrypt"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="调用jni接口获取加密串"
        android:textColor="@color/black"
        android:textSize="17sp" />
    <TextView
        android:id="@+id/tv_encrypt"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="这里显示jni加密结果"
        android:textColor="@color/black"
        android:textSize="17sp" />
    <Button
        android:id="@+id/btn_decrypt"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="调用jni接口获取解密串"
        android:textColor="@color/black"
        android:textSize="17sp" />
    <TextView
        android:id="@+id/tv_decrypt"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="这里显示jni解密结果"
        android:textColor="@color/black"
        android:textSize="17sp" />
</LinearLayout>

创作不易 觉得有帮助请点赞关注收藏~~~

相关文章
|
3月前
|
安全 Android开发 Kotlin
Android经典实战之SurfaceView原理和实践
本文介绍了 `SurfaceView` 这一强大的 UI 组件,尤其适合高性能绘制任务,如视频播放和游戏。文章详细讲解了 `SurfaceView` 的原理、与 `Surface` 类的关系及其实现示例,并强调了使用时需注意的线程安全、生命周期管理和性能优化等问题。
176 8
|
4月前
|
Java Android开发 C++
Android Studio JNI 使用模板:c/cpp源文件的集成编译,快速上手
本文提供了一个Android Studio中JNI使用的模板,包括创建C/C++源文件、编辑CMakeLists.txt、编写JNI接口代码、配置build.gradle以及编译生成.so库的详细步骤,以帮助开发者快速上手Android平台的JNI开发和编译过程。
329 1
|
29天前
|
JSON JavaScript 前端开发
harmony-chatroom 自研纯血鸿蒙OS Next 5.0聊天APP实战案例
HarmonyOS-Chat是一个基于纯血鸿蒙OS Next5.0 API12实战开发的聊天应用程序。这个项目使用了ArkUI和ArkTS技术栈,实现了类似微信的消息UI布局、输入框光标处插入文字、emoji表情图片/GIF动图、图片预览、红包、语音/位置UI、长按语音面板等功能。
62 2
|
2月前
|
缓存 前端开发 Android开发
Android实战之如何截取Activity或者Fragment的内容?
本文首发于公众号“AntDream”,介绍了如何在Android中截取Activity或Fragment的屏幕内容并保存为图片。包括截取整个Activity、特定控件或区域的方法,以及处理包含RecyclerView的复杂情况。
26 3
|
3月前
|
Android开发 开发者 索引
Android实战经验之如何使用DiffUtil提升RecyclerView的刷新性能
本文介绍如何使用 `DiffUtil` 实现 `RecyclerView` 数据集的高效更新,避免不必要的全局刷新,尤其适用于处理大量数据场景。通过定义 `DiffUtil.Callback`、计算差异并应用到适配器,可以显著提升性能。同时,文章还列举了常见错误及原因,帮助开发者避免陷阱。
261 9
|
2月前
|
JavaScript 小程序 开发者
uni-app开发实战:利用Vue混入(mixin)实现微信小程序全局分享功能,一键发送给朋友、分享到朋友圈、复制链接
uni-app开发实战:利用Vue混入(mixin)实现微信小程序全局分享功能,一键发送给朋友、分享到朋友圈、复制链接
453 0
|
3月前
|
开发工具 Android开发 git
Android实战之组件化中如何进行版本控制和依赖管理
本文介绍了 Git Submodules 的功能及其在组件化开发中的应用。Submodules 允许将一个 Git 仓库作为另一个仓库的子目录,有助于保持模块独立、代码重用和版本控制。虽然存在一些缺点,如增加复杂性和初始化时间,但通过最佳实践可以有效利用其优势。
44 3
|
2月前
|
Android开发
Android实战之如何快速实现自动轮播图
本文介绍了在 Android 中使用 `ViewPager2` 和自定义适配器实现轮播图的方法,包括添加依赖、布局配置、创建适配器及实现自动轮播等步骤。
86 0
|
3月前
|
Java Android开发 UED
🧠Android多线程与异步编程实战!告别卡顿,让应用响应如丝般顺滑!🧵
在Android开发中,为应对复杂应用场景和繁重计算任务,多线程与异步编程成为保证UI流畅性的关键。本文将介绍Android中的多线程基础,包括Thread、Handler、Looper、AsyncTask及ExecutorService等,并通过示例代码展示其实用性。AsyncTask适用于简单后台操作,而ExecutorService则能更好地管理复杂并发任务。合理运用这些技术,可显著提升应用性能和用户体验,避免内存泄漏和线程安全问题,确保UI更新顺畅。
121 5
|
2月前
|
Android开发
Android开发显示头部Bar的需求解决方案--Android应用实战
Android开发显示头部Bar的需求解决方案--Android应用实战
24 0