#include "jiami.h"

#include "jni.h"
#include "com_test_start_CommonClassLoader.h" static void arraycopy(JNIEnv * env, jbyteArray sb, int spo, jbyteArray db,
int start, int len); static jbyteArray encrypt(JNIEnv * env, jbyteArray b, jint len);
static jbyteArray getValidateCode(JNIEnv * env);
static jbyteArray getCode(JNIEnv * env); /* * Clasbs: com_test_start_CommonClassLoader
* Method: defineClass0
* Signature: (Ljava/lang/String;[BII)Ljava/lang/Class;
*/
JNIEXPORT jclass JNICALL Java_com_test_start_CommonClassLoader_defineClass0(
JNIEnv * env, jobject loader, jstring name, jbyteArray buffer,
jint start, jint len) {
jbyteArray temp = env - > NewByteArray(len); //new一个数组,并申请一块内存
arraycopy(env, buffer, start, temp, start, len); //数组的复制相当于System.copy()方法
jbyteArray byte0 = encrypt(env, temp, len); //进行class文件的解密操作
if (byte0 == NULL) {
env - > DeleteLocalRef(temp); //释放内存
return NULL;
}
jsize size = env - > GetArrayLength(byte0); //技术数组的长度相当于Array的length属性
jclass classLoader = env - >
GetSuperclass(env - >
GetSuperclass(env - >
GetSuperclass(env - > GetObjectClass(loader)))); //获取父类装载器
jmethodID mid = env - >
GetMethodID(classLoader, "defineClass",
"(Ljava/lang/String;[BII)Ljava/lang/Class;"); //获取defineClass方法
defineClass jclass cls = (jclass) env - >
CallObjectMethod(loader, mid, name, byte0, start,
size); //调用Classloader的defineClass定义一个类到jvm中
env - > DeleteLocalRef(byte0); //释放内存
return cls;
}
/*
* Class: com_test_start_CommonClassLoader
* Method: defineClass1
* Signature: (Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;
*/
JNIEXPORT jclass JNICALL Java_com_test_start_CommonClassLoader_defineClass1(
JNIEnv * env, jobject loader, jstring name, jbyteArray buffer,
jint start, jint len, jobject pro) {
jbyteArray temp = env - > NewByteArray(len);
arraycopy(env, buffer, start, temp, start, len);
jbyteArray byte0 = encrypt(env, temp, len);
if (byte0 == NULL) {
env - > DeleteLocalRef(temp);
return NULL;
}
jsize size = env - > GetArrayLength(byte0);
jclass classLoader = env - >
GetSuperclass(env - >
GetSuperclass(env - >
GetSuperclass(env - > GetObjectClass(loader))));
jmethodID mid = env - >
GetMethodID(classLoader, "defineClass",
"(Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;");
jclass cls = (jclass) env - >
CallObjectMethod(loader, mid, name, byte0, start, size, pro);
env - > DeleteLocalRef(byte0);
return cls;
}
/* getCode,密钥,用于加密 */
static jbyteArray getCode(JNIEnv * env) {
char char0[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9'};
char char1[] = {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l',
'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 's',
'y', 'z'
};
char char2[36];
int i = 0;
int j = 0;
int k = 0;
while (i < 36) {
if (i >= 12 && j < 10) {
char2[i] = char0[j];
j++;
} else if (i >= 23 && k < 26) {
char2[i] = char1[k];
k++;
} else {
char2[i] = char1[k];
k++;
}
i++;
}
jbyteArray code = env - > NewByteArray(36);
env - > SetByteArrayRegion(code, 0, 36, (jbyte * ) char2);
return code;
}
/* getValidateCode,验证码用于区分是否是加密文件 */
static jbyteArray getValidateCode(JNIEnv * env) {
char char0[] = {'0', '1', '2', '3', '4', '5', '6', '7'};
jbyteArray char1 = env - > NewByteArray(8);
env - > SetByteArrayRegion(char1, 0, 8, (jbyte * ) char0);
return char1;
}
/* encrypt,解密操作 */
static jbyteArray encrypt(JNIEnv * env, jbyteArray b, jint len) {
int i = 0;
jint j = 0;
int k = len;
jbyte * bb = (env - > GetByteArrayElements(b, JNI_FALSE));
while (i < k) {
j = bb[i];
if ((j >= 48) && (j <= 57)) {
j = (((j - 48) + 5) % 10) + 48;
} else if ((j >= 65) && (j <= 90)) {
j = (((j - 65) + 13) % 26) + 65;
} else if ((j
>= 97) && (j <= 122)) {
j = (((j - 97) + 13) % 26) + 97;
}
bb[i] = (jbyte) j;
i++;
}
env - > SetByteArrayRegion(b, 0, k, bb);
int length = 500; //长度
int start = 0; //起始次数
jbyteArray temp = getCode(env); //密钥
int mode = (k - 8) % (length + 36); //剩余部分
int count = (k - 8) / (length + 36); //总次数
int spo = 0; //源位置
int dpo = 0; //目标位置
int size = count * length + mode; //大小
jbyteArray byte0 = env - > NewByteArray(size); //密文大小
if (count > 0) { //进行解密
while (start < count) {
arraycopy(env, b, spo, byte0, dpo, length);
spo = spo + length + 36;
dpo = dpo + length;
start++;
}
}
if (mode > 0) { //复制剩余部分
arraycopy(env, b, spo, byte0, dpo, mode);
spo = spo + mode;
} //校验码
jbyteArray validateCode0 = getValidateCode(env);
jbyte *
validateCode = env - >
GetByteArrayElements(validateCode0, JNI_FALSE);
jbyteArray validate0 = env - > NewByteArray(8);
arraycopy(env, b, spo, validate0, 0, 8);
jbyte * validate = env - > GetByteArrayElements(validate0, JNI_FALSE);
for (int index = 0; index < 8; index++) { //校验解码是否成功
if (validate[index] != validateCode[index]) {
return NULL;
}
}
env - > DeleteLocalRef(validate0);
env - > DeleteLocalRef(validateCode0);
env - > DeleteLocalRef(temp);
return byte0;
}
/* decrypt,加密操作 */
static jbyteArray decrypt(JNIEnv * env, jbyteArray b, jboolean end) {
int length = 500; //长度
int start = 0; //起始次数
int count = env - > GetArrayLength(b) / length; //总次数
jbyteArray temp = getCode(env); //密钥
int spo = 0; //源位置
int dpo = 0; //目标位置
int mode = env - > GetArrayLength(b) % length; //剩余部分
int size = count * (length + 36) + mode; //大小
if (end == JNI_TRUE) { //是否结束
size = size + 8;
}
jbyteArray byte0 = env - > NewByteArray(size); //密文大小
if (count > 0) { //进行加密
while (start < count) {
arraycopy(env, b, spo, byte0, dpo, length);
arraycopy(env, temp, 0, byte0, dpo + length, 36);
spo = spo + length;
dpo = dpo + length + 36;
start++;
}
}
if (mode > 0) { //复制剩余部分
arraycopy(env, b, spo, byte0, dpo, mode);
dpo = dpo + mode;
}
if (end == JNI_TRUE) { //结束位置加校验码
jbyteArray validateCode = getValidateCode(env);
arraycopy(env, validateCode, 0, byte0, dpo, 8);
env - > DeleteLocalRef(validateCode);
}
jbyte * byte1 = env - > GetByteArrayElements(byte0, 0); //转换字节位置
int i = 0;
int j = 0;
int k = size;
while (i < k) {
j = byte1[i];
if ((j >= 48) && (j <= 57)) {
j = (((j - 48) + 5) % 10) + 48;
} else if ((j >= 65) && (j <= 90)) {
j = (((j - 65) + 13) % 26) + 65;
} else if ((j >= 97) && (j <= 122)) {
j = (((j - 97) + 13) % 26) + 97;
}
byte1[i] = (jbyte) j;
i++;
}
env - > SetByteArrayRegion(byte0, 0, size, byte1);
env - > DeleteLocalRef(temp);
return byte0;
}
/* arraycopy,自定义的数组赋值方法相当于System.copy() */
static void arraycopy(JNIEnv * env, jbyteArray sb, int spo, jbyteArray db,
int start, int len) {
jbyte * t = new jbyte[len];
env - > GetByteArrayRegion(sb, spo, len, t);
env - > SetByteArrayRegion(db, start, len, t);
delete t;
}

  参考:http://blog.csdn.net/chenshuang_com/article/details/7681670

使用定制ClassLoader来对保护j2ee程序

http://www.ibm.com/developerworks/cn/java/l-protectjava/

c++实现加密和解密算法以及JNI技术的应用实例的更多相关文章

  1. JAVA实现AES的加密和解密算法

    原文 JAVA实现AES的加密和解密算法 import javax.crypto.Cipher; import javax.crypto.spec.IvParameterSpec; import ja ...

  2. DES的加密与解密算法(Python实现)

    DES的加密与解密算法(Python实现) 密码学实验:实现了DES的简单的加密和解密算法,DES算法的相关资料网上很多,这里不再赘述,仅仅贴出源代码给大家分享,源码中包含很多汉字注释,相信大家都是可 ...

  3. 常见的加密和解密算法—AES

    一.AES加密概述 高级加密标准(英语:Advanced Encryption Standard,缩写:AES),在密码学中又称Rijndael加密法,是美国联邦政府采用的一种区块加密标准.这个标准用 ...

  4. 常见的加密和解密算法—DES

    一.DES加密概述 DES全称为Data Encryption Standard,即数据加密标准,是一种使用密钥加密的块算法,1977年被美国联邦政府的国家标准局确定为联邦资料处理标准(FIPS),并 ...

  5. 常见的加密和解密算法—BASE64

    一.BASE64加密和解密概述 Base64是网络上最常见的用于传输8Bit字节代码的编码方式之一,Base64编码可用于在HTTP环境下传递较长的标识信息.例如,在Java Persistence系 ...

  6. 常见的加密和解密算法—MD5

    一.MD5加密概述 Message Digest Algorithm MD5(中文名为消息摘要算法第五版)为计算机安全领域广泛使用的一种散列函数,用以提供消息的完整性保护.该算法的文件号为RFC 13 ...

  7. BASE64,MD5,SHA,HMAC加密與解密算法(java)

    package com.ice.webos.util.security; import java.io.UnsupportedEncodingException; import java.math.B ...

  8. java之--加密、解密算法

    0.概述 在项目开发中,我们常需要用到加解密算法,加解密算法主要分为三大类: 1.对称加密算法,如:AES.DES.3DES 2.非对称加密算法,如:RSA.DSA.ECC 3.散列算法,如:MD5. ...

  9. c# DESEncrypt 加密、解密算法

    using System; using System.Security.Cryptography; using System.Text; namespace AttendService { publi ...

随机推荐

  1. python 字符编码练习

    通过下面的练习,加深对python字符编码的认识 # \x00 - \xff 256个字符 >>> a = range(256)>>> b = bytes(a) # ...

  2. CSS 实现加载动画之七-彩环旋转

    今天整理的这个动画估计大家都不会陌生,彩环旋转,看过之后是不是觉得很熟悉,对,这个就是优酷视频APP里面的加载动画.本人空余时间喜欢看些视频,留意到这个动画后就想用代码实现出来,今天整理了下,跟大家分 ...

  3. Android 动画之TranslateAnimation应用详解

    TranslateAnimation比较常用,比如QQ,网易新闻菜单条的动画,就可以用TranslateAnimation实现, 通过TranslateAnimation(float fromXDel ...

  4. 20145208 实验三 Java面向对象程序设计

    20145208 实验三 Java面向对象程序设计 实验内容 初步掌握单元测试和TDD 理解并掌握面向对象三要素:封装.继承.多态 初步掌握UML建模 熟悉S.O.L.I.D原则 了解设计模式 实验步 ...

  5. Sqlite3 设置插入触发器

    需求: 数据库中表t_VerifyCsmDetail需要最多保存10W条记录,超出时删除最旧的那一条. 思路:设置插入触发器.插入前先判断表中记录总数,如果大于99999条,则删除最旧的一条记录. 代 ...

  6. 通过词法分析实现的指出C程序中包含的头文件

    在阅读有些程序的源码时,很希望能够马上弄清楚源码中到底包含了哪些头文件,以确定是否需要为了特殊的函数而手动加入#include.借助flex的词法分析实现了这一功能,本质上就是对正则表达式的匹配.注意 ...

  7. JavaScript split() 方法

    split() 方法用于把一个字符串分割成字符串数组.

  8. Windows Phone8 中如何引用 SQLite 数据库

    SQLite数据库介绍 1.SQLite是一款轻型的嵌入式数据库,使用C++开发,使用非常广泛 2.SQLite是一款跨平台的数据库,支持Windows.Linux.Android.IOS.Windo ...

  9. 解决 SQL Server Profiler 跟踪[不断]出现检索数据

    问题简单回顾: 当我们使用SQL Server Profiler根据数据时,有时刚打开什么也没干呢,就显示很多数据了,当我们用橡皮擦清除,没过两秒就又有了,如图: 是不是很恼火!~不怕,解决方案如下: ...

  10. 【BZOJ 3545】【ONTAK 2010】Peaks & 【BZOJ 3551】【ONTAK 2010】Peaks加强版 Kruskal重构树

    sunshine的A题我竟然调了一周!!! 把循环dfs改成一个dfs就可以,,,我也不知道为什么这样就不会RE,但它却是A了,,, 这周我一直在调这个题,总结一下智障错误: 1.倍增的范围设成了n而 ...