java – Lwxyz https://yun.lwxyz.cn Don't Forget To Be Awesome! Fri, 16 Jun 2023 22:20:57 +0000 zh-CN hourly 1 https://wordpress.org/?v=5.1.13 基于TOTP的双向认证算法 https://yun.lwxyz.cn/2018/09/18/%e5%9f%ba%e4%ba%8etotp%e7%9a%84%e5%8f%8c%e5%90%91%e8%ae%a4%e8%af%81%e7%ae%97%e6%b3%95/ https://yun.lwxyz.cn/2018/09/18/%e5%9f%ba%e4%ba%8etotp%e7%9a%84%e5%8f%8c%e5%90%91%e8%ae%a4%e8%af%81%e7%ae%97%e6%b3%95/#respond Tue, 18 Sep 2018 06:02:12 +0000 https://blog.lwxyz.org/?p=217 阅读更多]]> 转自:https://www.jianshu.com/p/e1031d36888b

基于otp算法的双向认证

先举例一个应用场景吧,我们应该都用U盾,或者将军令这种生成动态密钥的工具,其实它内部就是基于OTP算法来实现的。

谷歌也有开源这样的开源项目,本文就是博主通读了谷歌开源项目源码来实现的,项目地址:

Google Android端项目源码

Google 后台项目源码

算法概要

TOTP(基于时间的一次性密码算法)是支持时间作为动态因素基于HMAC一次性密码算法的扩展。

本算法是一个对称算法,也就是说,后台和移动端采用同样的密钥,同时这个算法是依赖于当前的系统时间的,所以可以用于动态验证。

TOTP = HMAC-SHA-1(K, (T – T0) / X)

  • K 共享密钥
  • T 当前时间戳
  • T0 开始的时间戳
  • X 时间步长

多唠叨几句:我们整体算法采用处理密钥的方式,是要将密钥转换成byte数组之后进行操作的,还有就是谷歌的这套双向认证算法中严格的采用了Base32字符串的方式,Base32中只有A-Z和2-7这些字符。

代码效果:

2017/08/09 20:39:56
983452
2017/08/09 20:39:57
983452
2017/08/09 20:39:58
983452
2017/08/09 20:39:59
983452
2017/08/09 20:40:00
977560
2017/08/09 20:40:01
977560
2017/08/09 20:40:02
977560
2017/08/09 20:40:03
977560
2017/08/09 20:40:04
977560

应用在app中的效果图:

应用在app中的效果图:

核心代码算法:

算法GitHub地址:https://github.com/linsir6/TOTP

算法的核心类:

/**
 * Created by linSir
 * date at 2017/8/8.
 * describe: 算法的核心类
 */

public class PasscodeGenerator {
    private static final int MAX_PASSCODE_LENGTH = 9;

    private static final int[] DIGITS_POWER
            // 0 1  2   3    4     5      6       7        8         9
            = {1,10,100,1000,10000,100000,1000000,10000000,100000000,1000000000};

    private final Signer signer;
    private final int codeLength;

    interface Signer {
        byte[] sign(byte[] data) throws GeneralSecurityException;
    }

    public PasscodeGenerator(Signer signer, int passCodeLength) {
        if ((passCodeLength < 0) || (passCodeLength > MAX_PASSCODE_LENGTH)) {
            throw new IllegalArgumentException(
                    "PassCodeLength must be between 1 and " + MAX_PASSCODE_LENGTH
                            + " digits.");
        }
        this.signer = signer;
        this.codeLength = passCodeLength;
    }

    private String padOutput(int value) {
        String result = Integer.toString(value);
        for (int i = result.length(); i < codeLength; i++) {
            result = "0" + result;
        }
        return result;
    }

    public String generateResponseCode(long state)
            throws GeneralSecurityException {
        byte[] value = ByteBuffer.allocate(8).putLong(state).array();
        return generateResponseCode(value);
    }

    public String generateResponseCode(byte[] challenge)
            throws GeneralSecurityException {
        byte[] hash = signer.sign(challenge);

        int offset = hash[hash.length - 1] & 0xF;
        int truncatedHash = hashToInt(hash, offset) & 0x7FFFFFFF;
        int pinValue = truncatedHash % DIGITS_POWER[codeLength];
        return padOutput(pinValue);
    }

    private int hashToInt(byte[] bytes, int start) {
        DataInput input = new DataInputStream(
                new ByteArrayInputStream(bytes, start, bytes.length - start));
        int val;
        try {
            val = input.readInt();
        } catch (IOException e) {
            throw new IllegalStateException(e);
        }
        return val;
    }
}

]]>
https://yun.lwxyz.cn/2018/09/18/%e5%9f%ba%e4%ba%8etotp%e7%9a%84%e5%8f%8c%e5%90%91%e8%ae%a4%e8%af%81%e7%ae%97%e6%b3%95/feed/ 0
maven中scope详解 https://yun.lwxyz.cn/2018/09/06/maven%e4%b8%adscope%e8%af%a6%e8%a7%a3/ https://yun.lwxyz.cn/2018/09/06/maven%e4%b8%adscope%e8%af%a6%e8%a7%a3/#respond Thu, 06 Sep 2018 03:21:20 +0000 https://blog.lwxyz.org/?p=212 阅读更多]]> 参考:https://blog.csdn.net/cd18333612683/article/details/66478332

scope的分类

compile、test、runtime、provided、system、import (2.0.9之后新增)

1.compile:默认值 他表示被依赖项目需要参与当前项目的编译,还有后续的测试,运行周期也参与其中,是一个比较强的依赖。打包的时候通常需要包含进去

2.test:依赖项目仅仅参与测试相关的工作,包括测试代码的编译和执行,不会被打包,例如:junit

3.runtime:表示被依赖项目无需参与项目的编译,不过后期的测试和运行周期需要其参与。与compile相比,跳过了编译而已。例如JDBC驱动,适用运行和测试阶段

4.provided:打包的时候可以不用包进去,别的设施会提供。事实上该依赖理论上可以参与编译,测试,运行等周期。相当于compile,但是打包阶段做了exclude操作

5.system:从参与度来说,和provided相同,不过被依赖项不会从maven仓库下载,而是从本地文件系统拿。需要添加systemPath的属性来定义路径

6.import:从它只使用在<dependencyManagement>中,表示从其它的pom中导入dependency的配置

]]>
https://yun.lwxyz.cn/2018/09/06/maven%e4%b8%adscope%e8%af%a6%e8%a7%a3/feed/ 0
各种学习网站 https://yun.lwxyz.cn/2018/08/10/%e5%b9%b6%e5%8f%91%e7%bc%96%e7%a8%8b%e7%bd%91/ https://yun.lwxyz.cn/2018/08/10/%e5%b9%b6%e5%8f%91%e7%bc%96%e7%a8%8b%e7%bd%91/#respond Fri, 10 Aug 2018 01:34:38 +0000 https://blog.lwxyz.org/?p=176 阅读更多]]> 1.并发编程网

http://ifeve.com/

 

2.spring学习网站

http://www.spring4all.com/

]]>
https://yun.lwxyz.cn/2018/08/10/%e5%b9%b6%e5%8f%91%e7%bc%96%e7%a8%8b%e7%bd%91/feed/ 0
Java各种对象(PO,BO,VO,DTO,POJO,DAO,Entity,JavaBean,JavaBeans)的区分 https://yun.lwxyz.cn/2018/08/06/java%e5%90%84%e7%a7%8d%e5%af%b9%e8%b1%a1%ef%bc%88pobovodtopojodaoentityjavabeanjavabeans%ef%bc%89%e7%9a%84%e5%8c%ba%e5%88%86/ https://yun.lwxyz.cn/2018/08/06/java%e5%90%84%e7%a7%8d%e5%af%b9%e8%b1%a1%ef%bc%88pobovodtopojodaoentityjavabeanjavabeans%ef%bc%89%e7%9a%84%e5%8c%ba%e5%88%86/#respond Mon, 06 Aug 2018 06:30:15 +0000 https://blog.lwxyz.org/?p=168 阅读更多]]> 转自:https://www.cnblogs.com/lyjin/p/6389349.html

PO:持久对象 (persistent object),po(persistent object)就是在Object/Relation Mapping框架中的Entity,po的每个属性基本上都对应数据库表里面的某个字段。完全是一个符合Java Bean规范的纯Java对象,没有增加别的属性和方法。持久对象是由insert数据库创建,由数据库delete删除的。基本上持久对象生命周期和数据库密切相关。


VO:值对象(Value Object),通常用于业务层之间的数据传递,和PO一样也是仅仅包含数据而已。但应是抽象出的业务对象,可以和表对应,也可以不,这根据业务的需要。

表现层对象(View Object),主要对应展示界面显示的数据对象,用一个VO对象来封装整个界面展示所需要的对象数据。


BO:业务对象层的缩写(Business Object),封装业务逻辑的java对象,通过调用DAO方法,结合PO,VO进行业务操作。具体可以看网上的一个例子:

比如一个简历,有教育经历、工作经历、社会关系等等。
我们可以把教育经历对应一个PO,工作经历对应一个PO,社会关系对应一个PO。
建立一个对应简历的BO对象处理简历,每个BO包含这些PO。
这样处理业务逻辑时,我们就可以针对BO去处理。


DTO:数据传输对象(Data Transfer Object),是一种设计模式之间传输数据的软件应用系统。数据传输目标往往是数据访问对象从数据库中检索数据。数据传输对象与数据交互对象或数据访问对象之间的差异是一个以不具有任何行为除了存储和检索的数据(访问和存取器)。简单来说,当我们需要一个对象10个字段的内容,但这个对象总共有20个字段,我们不需要把整个PO对象全部字段传输到客户端,而是可以用DTO重新封装,传递到客户端。此时,如果这个对象用来对应界面的展现,就叫VO。


JavaBean:一种可重用组件,即“一次性编写,任何地方执行,任何地方重用”。满足三个条件①类必须是具体的和公共的②具有无参构造器③提供一致性设计模式的公共方法将内部域暴露成员属性。

主要用途:可以用在 功能、处理、值、数据库访问和JSP中任何可以用Java代码创造的对象。

有两种:一种是有用户界面(UI,User Interface)的JavaBean;还有一种是没有用户界面,主要负责处理事务(如数据运算,操纵数据库)的JavaBean。JSP通常访问的是后一种JavaBean。

分类:通常有Session bean,Entity bean,MessageDrivenBean三大类

——Session bean会话构件,是短暂的对象,运行在服务器上,并执行一些应用逻辑处理,它由客户端应用程序建立,其数据需要自己来管理。分为无状态和有状态两种。
——Entity bean实体构件,是持久对象,可以被其他对象调用。在建立时指定一个唯一标示的标识,并允许客户程序,根据实体bean标识来定位beans实例。多个实体可以并发访问实体bean,事务间的协调由容器来完成。
——MessageDriven Bean消息构件,是专门用来处理JMS(Java Message System)消息的规范(EIB2.0)。JMS是一种与厂商无关的API,用来访问消息收发系统,并提供了与厂商无关的访问方法,以此来访问消息收发服务。JMS客户机可以用来发送消息而不必等待回应。

JavaBeans:JavaBeans 从狭义来说,指的是 JavaBeans 规范也就是位于 java.beans 包中的一组 API。从广义上来说,JavaBeans 指的是 API 集合,比如 Enterprise JavaBeans。


POJO:POJO(Plain Ordinary Java Object)简单的Java对象,实际就是普通JavaBeans,是为了避免和EJB混淆所创造的简称。通指没有使用Entity Beans的普通java对象,可以把POJO作为支持业务逻辑的协助类。

POJO实质上可以理解为简单的实体类,顾名思义POJO类的作用是方便程序员使用数据库中的数据表,对于广大的程序员,可以很方便的将POJO类当做对象来进行使用,当然也是可以方便的调用其get,set方法。POJO类也给我们在struts框架中的配置带来了很大的方便。
一个POJO持久化以后就是PO
直接用它传递、传递过程中就是DTO
直接用来对应表示层就是VO

DAO: 数据访问对象是第一个面向对象的数据库接口,是一个数据访问接口(Data Access Object)。它可以把POJO持久化为PO,用PO组装出来VO、DTO。

DAO模式是标准的J2EE设计模式之一.开发人员使用这个模式把底层的数据访问操作和上层的商务逻辑分开.一个典型的DAO实现有下列几个组件:
1. 一个DAO工厂类;
2. 一个DAO接口;
3. 一个实现DAO接口的具体类;
4. 数据传递对象(有些时候叫做值对象).
具体的DAO类包含了从特定的数据源访问数据的逻辑,一般一个DAO类和一张表对应,每个操作要和事务关联。

]]>
https://yun.lwxyz.cn/2018/08/06/java%e5%90%84%e7%a7%8d%e5%af%b9%e8%b1%a1%ef%bc%88pobovodtopojodaoentityjavabeanjavabeans%ef%bc%89%e7%9a%84%e5%8c%ba%e5%88%86/feed/ 0