开发支持类库

UUID类

  • UUID是一种生成无重复字符串的一种程序类,这种程序类的主要功能是根据时间戳实现一个自动的无重复的字符串定义;
  • 一般在获取UUID的时候往往都是随机生成一个内容,所以可以通过如下方式获取:
    • 获取UUID对象:public static UUID randomUUID()
    • 根据字符串获取UUID内容:public static UUID fromString(String name)
import java.util.UUID;
public class UUIDApi {
    public static void main(String[] args) {
        UUID uid = UUID.randomUUID();
        System.out.println(uid);
    }
}
  • 在对一些文件进行自动命名处理的情况下,UUID类型非常好用;

Optional类

  • Optional类的主要功能是进行null的相关处理,为了防止空指向异常,往往可以追加有null的验证;
package com.company.OptionalApi;
interface IMessage {
    public String getContent();
}
class MessageImpl implements IMessage {
    @Override
    public String getContent() {
        return "Hello World";
    }
}
class MessageUtil {
    private MessageUtil() {
    }
    public static IMessage getMessage() {
        return null;
    }
    public static void useMessage(IMessage msg) {
        if (msg != null) {
            System.out.println(msg.getContent());       //有可能会出现null而导致空指向
        }
    }
}
public class OptionalApi {
    public static void main(String[] args) throws Exception {
        MessageUtil.useMessage(MessageUtil.getMessage());
    }
}
  • 在引用接收的一方往往都是被动的进行判断,所以为了解决这种被动的处理操作,在Java类中提供有一个Optional类,这个类可以实现null的处理操作,提供有如下的一些操作方法:
    • 返回空的数据:public static <T> Optional<T> empty()
    • 获取数据:public T get()
    • 保存数据,但是不允许出现null:public static <T> Optional<T> of(T value),如果保存时出现了null则会抛出NullPointerException的异常
    • 保存数据,允许为null:public static <T> Optional<T> ofNullable(T value)
    • 空的时候,返回其他数据:public T orElse(T other)
此图来源于李兴华老师
package com.company.OptionalApi;
import java.util.Optional;
interface IMessage {
    public String getContent();
}
class MessageImpl implements IMessage {
    @Override
    public String getContent() {
        return "Hello World";
    }
}
class MessageUtil {
    private MessageUtil() {
    }
    public static Optional<IMessage> getMessage() {
        return Optional.of(new MessageImpl());
    }
    public static void useMessage(IMessage msg) {
        if (msg != null) {
            System.out.println(msg.getContent());       //有可能会出现null而导致空指向
        }
    }
}
public class OptionalApi {
    public static void main(String[] args) throws Exception {
        IMessage msg = MessageUtil.getMessage().get();
        MessageUtil.useMessage(msg);
    }
}
  • 如果现在数据保存的内容是null,则会在保存处出现异常;
public static Optional<IMessage> getMessage() {
    return Optional.of(null);
}
  • 由于Optional类中允许保存有null的内容,所以数据在获取的时候也可以进行null的处理,如果此时为null,在get()时就会抛出异常,此时可以更换为orElse()方法;
public class OptionalApi {
    public static void main(String[] args) throws Exception {
        IMessage msg = MessageUtil.getMessage().orElse(new MessageImpl());
        MessageUtil.useMessage(msg);
    }
}

ThreadLocal类

package com.company.ThreadLocal;
class Channel {
    private static Message message;
    private Channel() {
    }
    public static void setMessage(Message m) {
        message = m;
    }
    public static void send() {
        System.out.println("【消息发送】" + message.getInfo());
    }
}
class Message {
    private String info;
    public void setInfo(String info) {
        this.info = info;
    }
    public String getInfo() {
        return info;
    }
}
public class ThreadLocalApi {
    public static void main(String[] args) throws Exception {
        Message msg = new Message();
        msg.setInfo("Hello World");
        Channel.setMessage(msg);
        Channel.send();
    }
}
  • 上面所展示程序是以单线程的模式来进行处理的,那么如果在多线程的状态下能否实现完全一致的操作效果呢?为此我们启用三个线程进行处理;
public class ThreadLocalApi {
    public static void main(String[] args) throws Exception {
        for (int x = 0; x < 3; x++) {
            int num = x;
            new Thread(() -> {
                Message msg = new Message();
                msg.setInfo("第" + num + "个消息");
                Channel.setMessage(msg);
                Channel.send();
            }, "线程" + num).start();
        }
    }
}
  • 这个时候消息的处理产生了影响;
此图来源于李兴华老师
  • 在保持Channel(所有消息发送的通道)核心结构不发生改变的情况下,需要考虑到每个线程的独立操作问题,这样的情况下就发现对于Channel类而言除了要保留有发送的消息之外,还应该多存放有一个每一个线程的标记(当前线程),那么这个时候就可以通过ThreadLocal类来存放数据;
  • 在ThreadLocal类里面提供有如下的操作方法:
    • 构造方法:public ThreadLocal()
    • 设置数据:public void set(T value)
    • 取出数据:public T get()
    • 删除数据:public void remove()
    • public class ThreadLocal<T> extends Object
此图来源于李兴华老师
package com.company.ThreadLocal;
class Channel {
    private static final ThreadLocal<Message> THREADLOCAL = new ThreadLocal<Message>();
    private Channel() {}
    public static void setMessage(Message m) {
        THREADLOCAL.set(m);
    }
    public static void send() {
        System.out.println(Thread.currentThread().getName() + "、【消息发送】" + THREADLOCAL.get().getInfo());
    }
}
class Message {
    private String info;
    public void setInfo(String info) {
        this.info = info;
    }
    public String getInfo() {
        return info;
    }
}
public class ThreadLocalApi {
    public static void main(String[] args) throws Exception {
        for (int x = 0; x < 3; x++) {
            int num = x;
            new Thread(() -> {
                Message msg = new Message();
                msg.setInfo("第" + num + "个消息");
                Channel.setMessage(msg);
                Channel.send();
            }, "线程" + num).start();
        }
    }
}
  • 每一个线程通过ThreadLocal只允许保存一个数据;

定时器

  • 定时器的主要操作是进行定时任务的处理,在Java中提供有定时任务的支持,但是这种任务的处理只是实现了一种间隔触发的操作;
  • java.util.TimerTask类:实现定时任务处理;
  • java.util.Timer类:进行任务的启动,启动的方法:
    • 任务启动:public void schedule(TimerTask task, long delay),延迟单位为毫秒;
    • 间隔触发:public void scheduleAtFixedRate(TimerTask task, Date firstTime, long period)
package com.company.TimerApi;
import java.util.Timer;
import java.util.TimerTask;
class MyTask extends TimerTask {
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "、定时任务,当前时间:" + System.currentTimeMillis());
    }
}
public class TimerApi {
    public static void main(String[] args) throws Exception {
        Timer timer = new Timer();
        timer.scheduleAtFixedRate(new MyTask(), 100, 1000);
        //timer.schedule(new MyTask(), 1000);
    }
}

Base64加密与解密

  • 正常来讲加密基本上永远都要伴随着解密,所谓的加密或者是解密往往都需要有一些规则,从JDK1.8开始提供有新的加密处理操作类,Base64处理,在这个类里面有两个内部类:
    • Base64.Encoder:进行加密处理,public byte[] encode(byte [] src)
    • Base64.Decoder:进行解密处理,public byte[] decode(byte [] src)
package com.company.Base64APi;
import java.util.Base64;
public class Base64Api {
    public static void main(String[] args) throws Exception {
        String msg = "Hello World";
        String enMsg = new String(Base64.getEncoder().encode(msg.getBytes()));
        System.out.println(enMsg);
        String deMsg = new String(Base64.getDecoder().decode(enMsg));
        System.out.println(deMsg);
    }
}
  • 虽然Base64可以实现加密和解密的处理,但是其由于是一个公版的算法,所以如果直接对数据进行加密往往并不安全,那么最好的做法是使用盐值操作:
String salt = "Mars";
String msg = "Hello World" + "{" + salt + "}";
  • 即便有盐值实际上发现加密效果也不好,最好的做法是多次加密;
package com.company.Base64APi;
import java.util.Base64;
class StringUtil {
    private static final String SALT = "Mars";
    private static final int REPEAT = 5;
    /** 
    * @Description: 加密处理 
    * @Param: [str要加密的字符串需要与盐值整合]
    * @return: java.lang.String 加密后的数据
    * @Author: mars_wu
    * @Date: 2020/12/25 下午6:14
    */ 
    public static String encode(String str) {
        String temp = str + "{" + SALT + "}";
        byte [] data = temp.getBytes();
        for (int x = 0; x < REPEAT; x++) {
            data = Base64.getEncoder().encode(data);
        }
        return new String(data);
    }
    /** 
    * @Description: 解密处理 
    * @Param: [str要解密的内容]
    * @return: java.lang.String 解密后的原始数据
    * @Author: mars_wu
    * @Date: 2020/12/25 下午6:22
    */ 
    public static String decode(String str) {
        byte [] data = str.getBytes();
        for (int x = 0; x < REPEAT; x++) {
            data = Base64.getDecoder().decode(data);
        }
        return new String(data).replaceAll("\\{\\w+\\}", "");
    }
}
public class Base64Api {
    public static void main(String[] args) throws Exception {
        String str = StringUtil.encode("Hello World");
        System.out.println(StringUtil.decode(str));
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,542评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,822评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,912评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,449评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,500评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,370评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,193评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,074评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,505评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,722评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,841评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,569评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,168评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,783评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,918评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,962评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,781评论 2 354

推荐阅读更多精彩内容