03.Eclipse下Ndk开发(以文件加密为例模拟一下开发过程)

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介: (创建于2017/12/2)1.编写native方法package com.example.ndk_file_encrpty;public class Cryptor { static{ System.

(创建于2017/12/2)

1.编写native方法

package com.example.ndk_file_encrpty;

public class Cryptor {
    
    static{
        System.loadLibrary("ndk_file_encrpty");
    }
    
    public native static void encrypt(String normal_path,String crypt_path);
    
    public native static void decrypt(String crypt_path,String decrypt_path);

}

2.进入eclipse项目src目录下(cd 进入到src目录下,使用命令生成头文件 javah 包名+类型(如 com.renzhenming.utils.JniUtils)),如果是进入的bin目录下使用此命令行的话,得到的头文件只有一些声明,没有生成相应的jni方法

bin目录下生成的是这样的

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_example_ndk_file_encrpty_Cryptor */

#ifndef _Included_com_example_ndk_file_encrpty_Cryptor
#define _Included_com_example_ndk_file_encrpty_Cryptor
#ifdef __cplusplus
extern "C" {
#endif


#ifdef __cplusplus
}
#endif
#endif

src目录下生成的才是我们需要的

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_example_ndk_file_encrpty_Cryptor */

#ifndef _Included_com_example_ndk_file_encrpty_Cryptor
#define _Included_com_example_ndk_file_encrpty_Cryptor
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_example_ndk_file_encrpty_Cryptor
 * Method:    encrypt
 * Signature: (Ljava/lang/String;Ljava/lang/String;)V
 */
JNIEXPORT void JNICALL Java_com_example_ndk_1file_1encrpty_Cryptor_encrypt
  (JNIEnv *, jclass, jstring, jstring);

/*
 * Class:     com_example_ndk_file_encrpty_Cryptor
 * Method:    decrypt
 * Signature: (Ljava/lang/String;Ljava/lang/String;)V
 */
JNIEXPORT void JNICALL Java_com_example_ndk_1file_1encrpty_Cryptor_decrypt
  (JNIEnv *, jclass, jstring, jstring);

#ifdef __cplusplus
}
#endif
#endif

有一点需要注意的是:对于像这样包名这样命名的情况


img_aa02906b1e5b7c7c5c07ec45649892d8.png
8392687.png

因为存在下划线,所以如果你直接按照平常的那种写法去手写,比如写成这样

JNIEXPORT void JNICALL Java_com_example_ndk_file_encrpty_Cryptor_encrypt

是错误的,正确的写法是如上边生成的

JNIEXPORT void JNICALL Java_com_example_ndk_1file_1encrpty_Cryptor_encrypt

在个别下划线的前边加了数字1,目前我也不知道为何,所以遇到这种包名,建议使用命令行生成

3.创建jni目录(eclipse工程是直接在项目根目录创建,将头文件复制进去,然后编写c文件,将对应的方法名复制进去,引入需要的
头文件)

4.添加native支持(右键工程如图,点击add native support后弹出对话框(创建出头文件之后),设置c文件名称,确定即可,然后会自动给你生成Android.mk文件,c文件,然后创建Application.mk文件)


img_661b544ae66324ebd6335b8fa90aac29.png
9041421.png

img_af29a3ca319d0616559ce54ce88ef611.png
9129687.png

5.确认你的eclipse已经配置了ndk,注意,ndk的版本会有可能发生问题,这样看你的eclipse最多能支持多大版本的ndk了
如果ndk版本过高,会导致你配置目录之后仍然提示你有问题,如图not a valid NDK directory,所以需要降低版本,我设置了r10之后可以了


img_d48c02107794a8010f6b876000c6680c.png
9272984.png

6.添加ndk相关头文件的支持(右键项目打开properties,发现里边由C/C++General选项,这两个选项是你添加了native surpport之后才有的)


img_eb8af9423b434369ed184e59a7b66e6a.png
9398312.png

在右边的add选项,添加这三个目录进去(一定要是你ndk中存在的目录,不同版本有不同)
D:\application\java\android-ndk-r10e\toolchains\arm-linux-androideabi-4.8\prebuilt\windows-x86_64\lib\gcc\arm-linux-androideabi\4.8\include

D:\application\java\android-ndk-r10e\toolchains\arm-linux-androideabi-4.8\prebuilt\windows-x86_64\lib\gcc\arm-linux-androideabi\4.8\include-fixed

D:\application\java\android-ndk-r10e\platforms\android-18\arch-arm\usr\include

7,补充c文件的方法实现

#include <jni.h>
#include <com_example_ndk_file_encrpty_Cryptor.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>


char password[] = "renzhenming";

//加密
JNIEXPORT void JNICALL Java_com_example_ndk_1file_1encrpty_Cryptor_encrypt(
        JNIEnv *env, jclass jcls, jstring normal_path_jstr,jstring crypt_path_jstr) {
    //jstring -> char*
    const char* normal_path = (*env)->GetStringUTFChars(env,normal_path_jstr,JNI_FALSE);
    const char* crypt_path = (*env)->GetStringUTFChars(env,crypt_path_jstr,JNI_FALSE);

    //打开文件
    FILE *normal_fp = fopen(normal_path, "rb");
    FILE *crypt_fp = fopen(crypt_path, "wb");
    //一次读取一个字符
    int ch;
    int i = 0; //循环使用密码中的字母进行异或运算
    int pwd_len = strlen(password); //密码的长度
    while ((ch = fgetc(normal_fp)) != EOF) { //End of File
        //写入(异或运算)
        fputc(ch ^ password[i % pwd_len], crypt_fp);
        i++;
    }
    //关闭
    fclose(crypt_fp);
    fclose(normal_fp);
}

//解密
JNIEXPORT void JNICALL Java_com_example_ndk_1file_1encrpty_Cryptor_decrypt(
        JNIEnv * env, jclass jcls, jstring crypt_path_jstr, jstring decrypt_path_jstr) {
    const char* crypt_path = (*env)->GetStringUTFChars(env,crypt_path_jstr,JNI_FALSE);
    const char* decrypt_path = (*env)->GetStringUTFChars(env,decrypt_path_jstr,JNI_FALSE);

    //打开文件
    FILE *normal_fp = fopen(crypt_path, "rb");
    FILE *crypt_fp = fopen(decrypt_path, "wb");
    //一次读取一个字符
    int ch;
    int i = 0; //循环使用密码中的字母进行异或运算
    int pwd_len = strlen(password); //密码的长度
    while ((ch = fgetc(normal_fp)) != EOF) { //End of File
        //写入(异或运算)
        fputc(ch ^ password[i % pwd_len], crypt_fp);
        i++;
    }
    //关闭
    fclose(crypt_fp);
    fclose(normal_fp);
}

8.在需要的地方调用

package com.example.ndk_file_encrpty;


import java.io.File;

import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    
    public void encrypt(View view){
        String path = Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+"man.jpg";
        String new_path = Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+"man_encrypt.jpg";
        Cryptor.encrypt(path, new_path);
    }
    
    public void decrypt(View view){
        String path = Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+"man_encrypt.jpg";
        String new_path = Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+"man_decrypt.jpg";
        Cryptor.decrypt(path, new_path);
    }

}

成功!

相关文章
|
19天前
|
NoSQL Java Redis
shiro学习四:使用springboot整合shiro,正常的企业级后端开发shiro认证鉴权流程。使用redis做token的过滤。md5做密码的加密。
这篇文章介绍了如何使用Spring Boot整合Apache Shiro框架进行后端开发,包括认证和授权流程,并使用Redis存储Token以及MD5加密用户密码。
21 0
shiro学习四:使用springboot整合shiro,正常的企业级后端开发shiro认证鉴权流程。使用redis做token的过滤。md5做密码的加密。
|
3月前
|
安全 网络安全 Android开发
安卓与iOS开发:选择的艺术网络安全与信息安全:漏洞、加密与意识的交织
【8月更文挑战第20天】在数字时代,安卓和iOS两大平台如同两座巍峨的山峰,分别占据着移动互联网的半壁江山。它们各自拥有独特的魅力和优势,吸引着无数开发者投身其中。本文将探讨这两个平台的特点、优势以及它们在移动应用开发中的地位,帮助读者更好地理解这两个平台的差异,并为那些正在面临选择的开发者提供一些启示。
120 56
|
Java 关系型数据库 MySQL
SSM度假村管理系统开发mysql数据库web结构java编程计算机网页源码eclipse项目
SSM 度假村管理系统是一套完善的信息系统,结合SSM框架完成本系统,对理解JSP java编程开发语言有帮助系统采用SSM框架(MVC模式开发),系统具有完整的源代码和数据库,系统主要采用B/S模式开发。 前段主要技术html div js 后端主要技术 SSM框架 java jdbc 数据库 mysql 开发工具 eclipse JDK1.8 TOMCAT 8.5
54 0
|
3月前
|
安全 网络安全 数据安全/隐私保护
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。
|
5月前
|
JavaScript 前端开发 安全
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
|
4月前
|
存储 Oracle Java
Java面试题:描述如何使用Eclipse或IntelliJ IDEA进行Java开发?
Java面试题:描述如何使用Eclipse或IntelliJ IDEA进行Java开发?
45 0
|
6月前
|
安全 网络安全 量子技术
网络安全与信息安全:漏洞、加密技术与安全意识的探索安卓应用开发中的内存管理策略
【5月更文挑战第31天】随着互联网的普及,网络安全问题日益严重。本文将深入探讨网络安全漏洞、加密技术以及安全意识等方面的问题,以期提高公众对网络安全的认识和防范能力。
|
5月前
|
开发工具 Android开发 git
合作开发(Eclipse+git +码云)
合作开发(Eclipse+git +码云)
66 0
|
6月前
|
存储 安全 算法
【PHP开发专栏】PHP加密与解密技术
【4月更文挑战第29天】本文探讨了PHP中的加密解密技术,涵盖基本概念如对称加密(AES、DES)、非对称加密(RSA、DSA)和哈希函数(MD5、SHA)。PHP提供内置函数支持加密,如`openssl_encrypt`、`openssl_pkey_new`、`hash`和`password_hash`。文章强调了最佳实践,如使用安全密钥、密钥管理和HTTPS,并给出用户注册登录的加密实战示例。通过理解和应用这些技术,开发者能增强Web应用的数据安全性。
203 1
|
6月前
|
JavaScript 前端开发 关系型数据库
金融技术解决方案:用Python和Vue开发加密货币交易平台
【4月更文挑战第11天】本文介绍了如何使用Python和Vue.js构建加密货币交易平台。首先确保安装了Python、Node.js、数据库系统和Git。后端可选择Flask或Django框架,通过RESTful API处理交易。前端利用Vue.js、Vuex和Vue Router创建用户友好的界面,并用Axios与后端通信。这种架构促进团队协作,提升代码质量和平台功能。
61 0

推荐镜像

更多