Java程序员的日常—— 基于类的策略模式、List<?>与List、泛型编译警告、同比和环比

简介:

早晨起得太早,昨晚睡得太晚,一天都迷迷糊糊的。中午虽然睡了半个小时,可是依然没有缓过来。整个下午都在混沌中....不过今天下载了一款手游——《剑侠情缘》,感觉不错,喜欢这种类型的游戏。

今天主要的工作还是做业务需求,不过下午状态不好,看了下《Effective java》,正好重构了下代码。

effective java 通过函数来作为策略

通过函数作为策略有两个要注意的地方:

  • 使用接口作为策略传入
  • 如果长期调用,应该设置为静态内部类,避免频繁创建过多的匿名对象

下面举个简单的例子,针对Engineer类提供不同的策略做排序,比如按照年龄或者按照员工级别:

class Engineer{
    private String name;
    private int age;
    private int level;
    public Engineer(String name,int age,int level) {
        this.name = name;
        this.age = age;
        this.level = level;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public int getLevel() {
        return level;
    }
    public void setLevel(int level) {
        this.level = level;
    }
}
public class Strategy {
    public static final Comparator<Engineer> AGE_ORDER = new AgeComparator();
    public static final Comparator<Engineer> LEVEL_ORDER = new LevelComparator();
    private static class AgeComparator implements Comparator<Engineer>{
        @Override
        public int compare(Engineer e1, Engineer e2) {
            return e1.getAge()-e2.getAge();
        }
    }
    private static class LevelComparator implements Comparator<Engineer>{
        @Override
        public int compare(Engineer e1, Engineer e2) {
            return e1.getLevel()-e2.getLevel();
        }
    }
    public static void main(String[] args) {
        List<Engineer> el = new ArrayList<>();
        el.add(new Engineer("zhangsan", 26, 3));
        el.add(new Engineer("lisi", 30, 2));
        el.add(new Engineer("wangwu", 28, 1));
        Collections.sort(el,AGE_ORDER);
        System.out.println(JSON.toJSON(el));
        Collections.sort(el,LEVEL_ORDER);
        System.out.println(JSON.toJSON(el));
    }
}

在上面的例子中,采用静态成员变量声明,可以在多次使用的时候节省创建对象的成本。而且静态成员在堆内存的分配上也更简单,不会每次都创建新的对象。

在真实的场景中,是在某个请求方法里面,返回一个List对象,需要对它按照日期排序。如果是普通的Collections.sort(list,new Comparator<xx>{})这种方式,会在每次返回结果的时候,都创建一个匿名类,很显然会浪费不少内存空间,增加垃圾回收的压力。使用静态成员变量的方式,可以减少这种不必要的浪费。

List<?>与List

由于在1.5之前的版本,java是没有泛型概念的。因此在引入泛型后,需要考虑到以前代码的移植。

没有泛型的时候,如果使用List,可以往里面插入任意类型的值。但是在取得时候,如果类型不对就有问题了:

List list = new ArrayList();
list.add(1);

String list0 = list.get(0);//出错

为了避免这种问题,1.5引入泛型,这样一套代码可以适用于多种类型;还能在编译器就检查类型是否一致。

除了这种List<E> xxx标准的泛型,java还提供了无限制性的泛型:

<?>意思是未知类型,就是不设上下限
<? extend Object>意思是继承于Object的未知类型
<? super Object>意思是Object的祖先类型

所以,尽量使用标准的格式,在某些情况下已知的一些通配限制,还可以使用<?>号加以限制。

记得最开始自己写代码的时候,满满的都是黄色标记,师兄就纠正我的做法,让我把这些警告全都去掉。其实随时保证没有警告的代码,才是最负责的做法。不管是自己屏蔽掉,还是做相应的解决,都好过编译的时候爆出一大堆警告好。

编译器警告

Java是一门编译型的语言,需要经过编译,变成class字节码才能执行。但是在编写泛型相关的代码时,总是会遇到一些警告。比如参数仅仅声明为Map,没有声明具体内部的内容等等。

在Eclipse中可以通过加入@SuppressWarning注解来忽略警告,但是不推荐这种做法。除非你对自己的代码非常自信,保证不会出现其他的类型,而导致ClassCastException。所以尽量在写代码的时候不要产生警告,如果想要忽略,尽量考虑清楚入口出口是否不会出现意外。

常用的就是unckecked和rawtypes,一个是不检查内部变量,一个是不检查参数类型。

all to suppress all warnings
boxing to suppress warnings relative to boxing/unboxing operations
cast to suppress warnings relative to cast operations
dep-ann to suppress warnings relative to deprecated annotation
deprecation to suppress warnings relative to deprecation
fallthrough to suppress warnings relative to missing breaks in switch statements
finally to suppress warnings relative to finally block that don’t return
hiding to suppress warnings relative to locals that hide variable
incomplete-switch to suppress warnings relative to missing entries in a switch statement (enum case)
nls to suppress warnings relative to non-nls string literals
null to suppress warnings relative to null analysis
rawtypes to suppress warnings relative to un-specific types when using generics on class params
restriction to suppress warnings relative to usage of discouraged or forbidden references
serial to suppress warnings relative to missing serialVersionUID field for a serializable class
static-access to suppress warnings relative to incorrect static access
synthetic-access to suppress warnings relative to unoptimized access from inner classes
unchecked to suppress warnings relative to unchecked operations
unqualified-field-access to suppress warnings relative to field access unqualified
unused to suppress warnings relative to unused code

什么是同比和环比

做业务需求,还是需要了解些业务知识才行。无论是电商环境,还是传统企业,环比和同比是最常见的数据分析手段,可以通过对比明显的看到当前业务的变化趋势,有利于管理层即使做出调整,那么什么是环比,什么是同比呢?

  • 环比就是现在的统计周期和上一个统计周期比较。
  • 同比是与历史时期作比较。

举个例子:

  • 2016年4月和2016年的3月相比,就是环比
  • 2016年的10月和2015年的10月相比,就是同比

太业务化的东西,就不说了,免得设计到什么尴尬的信息。

睡觉时间到,养好精神,才能专注...

本文转自博客园xingoo的博客,原文链接:Java程序员的日常—— 基于类的策略模式、List<?>与List、泛型编译警告、同比和环比,如需转载请自行联系原博主。
相关文章
|
8天前
|
JavaScript Java 编译器
Java包装类和泛型的知识点详解
Java包装类和泛型的知识点的深度理解
|
2天前
|
Java 编译器
Java Character 类
4月更文挑战第13天
|
3天前
|
存储 Java
Java基础教程(7)-Java中的面向对象和类
【4月更文挑战第7天】Java是面向对象编程(OOP)语言,强调将事务抽象成对象。面向对象与面向过程的区别在于,前者通过对象间的交互解决问题,后者按步骤顺序执行。类是对象的模板,对象是类的实例。创建类使用`class`关键字,对象通过`new`运算符动态分配内存。方法包括构造函数和一般方法,构造函数用于对象初始化,一般方法处理逻辑。方法可以有0个或多个参数,可变参数用`类型...`定义。`this`关键字用于访问当前对象的属性。
|
7天前
|
Java Shell
Java 21颠覆传统:未命名类与实例Main方法的编码变革
Java 21颠覆传统:未命名类与实例Main方法的编码变革
10 0
|
7天前
|
Java
Java 15 神秘登场:隐藏类解析未知领域
Java 15 神秘登场:隐藏类解析未知领域
11 0
|
7天前
|
存储 监控 安全
泛型魔法:解码Java中的类型参数
泛型魔法:解码Java中的类型参数
30 0
泛型魔法:解码Java中的类型参数
|
9天前
|
安全 Java
append在Java中是哪个类下的方法
append在Java中是哪个类下的方法
21 9
|
9天前
|
Java API
Java基础—笔记—内部类、枚举、泛型篇
本文介绍了Java编程中的内部类、枚举和泛型概念。匿名内部类用于简化类的创建,常作为方法参数,其原理是生成一个隐含的子类。枚举用于表示有限的固定数量的值,常用于系统配置或switch语句中。泛型则用来在编译时增强类型安全性,接收特定数据类型,包括泛型类、泛型接口和泛型方法。
9 0
|
9天前
|
JavaScript Java 测试技术
基于Java的网络类课程思政学习系统的设计与实现(源码+lw+部署文档+讲解等)
基于Java的网络类课程思政学习系统的设计与实现(源码+lw+部署文档+讲解等)
25 0
基于Java的网络类课程思政学习系统的设计与实现(源码+lw+部署文档+讲解等)
|
10天前
|
存储 安全 Java
java多线程之原子操作类
java多线程之原子操作类