zoukankan      html  css  js  c++  java
  • JAVA加密系列(四)- 位运算加密(异或加密)


    位运算介绍
    程序中的所有数在计算机内存中都是以二进制的形式储存的。位运算就是直接对整数在内存中的二进制位进行操作。位运算虽说是语言的基础,但是应用层基本很少有涉及到此,很多开发同学可能了解的不是蛮清楚,本文只对算法用的位运算做一个简单介绍,其他运算符将一笔带过,感兴趣的可以在好好的深入学习一下

    "^" 异或运算
    相同为0 不同为1

    例子 2^3=1

    0010^0011=0001

    "&" 与运算
    只要有一个为0,就为0

    例子 2&3=2

    0010&0011=0010


    ">>>" 无符号右移
    10进制转二进制的时候,因为二进制数一般分8位、 16位、32位以及64位 表示一个十进制数,所以在转换过程中,最高位会补零。
    在计算机中负数采用二进制的补码表示,10进制转为二进制得到的是源码,将源码按位取反得到的是反码,反码加1得到补码
    二进制的最高位是符号位,0表示正,1表示负。

    其他算法符 "|"、"~"、">>"、"<<"...感兴趣自己研究

    使用异或写对称性加密
    虽然安全性相对来说没有AES等高,但是优点也是异常突出,加密速度比单向加密都快,消耗的性能非常的小。

    public class SecurityUtil {

    public static void main(String[] args) {
    //字符串编码
    System.out.println("编码结果:"+encrypt("hello world"));
    //字符串解码
    System.out.println("编码结果:"+dencrypt("KSFaWC4VMl9GXA=="));
    }


    private final static byte[] ENCRYPT_VAL = {
    65, 68, 54, 52, 65, 53, 69, 48,
    52, 56, 57, 57, 69, 56, 56, 69,
    68, 48, 70, 55, 70, 50, 49, 51,
    65, 52, 68, 69, 54, 65, 53, 48
    };

    /**
    * 加密
    *
    * @param str
    * @return
    */
    public static String encrypt(String str) {
    byte[] bytes = str.getBytes();
    for (int i = 0; i < bytes.length; i++) {
    bytes[i] = (byte) (bytes[i] ^ ENCRYPT_VAL[i % ENCRYPT_VAL.length]);
    }
    return new String(Base64.getEncoder().encode(bytes));
    }

    /**
    * 解密
    *
    * @param str
    * @return
    */
    public static String dencrypt(String str) {
    byte[] bytes = Base64.getDecoder().decode(str);
    for (int i = 0; i < bytes.length; i++) {
    bytes[i] = (byte) (bytes[i] ^ ENCRYPT_VAL[i % ENCRYPT_VAL.length]);
    }
    return new String(bytes);
    }
    }
    log
    编码结果:KSFaWC4VMl9GXA==
    编码结果:hello word

    使用位运算对MD5加密进行魔改
    以下只是一个例子,抛转引玉以下,大家可以随意魔改,加密和验证的逻辑一致即可

    public class MD5Util {
    public static void main(String[] args) {
    //字符串编码
    System.out.println("编码结果:"+encryptMD5("hello world"));
    }

    /**
    * 16进制字符
    */
    static String[] chars ="如果我没有眼睛,我也一定可以看见你迷人的美丽".split("");

    public static String encryptMD5(String str){
    try {
    byte[] bytes=str.getBytes();
    MessageDigest messageDigest=MessageDigest.getInstance("MD5");
    messageDigest.update(bytes);
    bytes = messageDigest.digest();
    // digest()最后确定返回md5 hash值,返回值为8位字符串。因为md5 hash值是16位的hex值,实际上就是8位的字符
    StringBuffer sb = new StringBuffer();
    for (int i = 0; i < bytes.length; i++) {

    // 一个字节对应两个字符
    byte x = bytes[i];
    // 取得高位
    int h = 0x0f & (x >>> 4);
    // 取得低位
    int l = 0x0f & x;
    sb.append(chars[h]).append(chars[l]);
    }
    return sb.toString();
    } catch (NoSuchAlgorithmException e) {
    e.printStackTrace();
    return null;
    }
    }

    }
    log

    {
    "-do-not-delete-":"second"
    }
    编码结果:果没眼,有看见如以眼我定眼如见一定没我看可我有果看见看没也以见有
    源码下载

    作者:lance_小超
    链接:https://www.jianshu.com/p/47d8c41132f8
    来源:简书
    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 相关阅读:
    再学 GDI+[47]: 路径 CloseFigure
    再学 GDI+[46]: 路径 Create、FillPath、DrawPath
    学习官方示例 TApplication.ExeName
    再学 GDI+[45]: 文本输出 在矩形中格式化输出
    再学 GDI+[50]: 路径 GetPathPoints、GetPathTypes、TPathData、GetPathData
    再学 GDI+[49]: 路径 GetPointCount、GetPathPoints、GetLastPoint、GetBounds
    再学 GDI+[48]: 路径 StartFigure、CloseFigure、CloseAllFigures
    微软DevWow博客达人征文大赛获奖名单
    开始组建博客园北京俱乐部
    .NET技术大会
  • 原文地址:https://www.cnblogs.com/sidesky/p/12726366.html
Copyright © 2011-2022 走看看