详解Java实践之建造者模式


Posted in Java/Android onJune 18, 2021
目录
  • 一、前言
  • 二、开发环境
  • 三、建造者模式介绍
  • 四、案例场景模拟
    • 4.1、场景模拟工程
    • 4.2、场景简述
      • 4.2.1、物料接口
      • 4.2.2、吊顶(ceiling)
      • 4.2.3、涂料(coat)
      • 4.2.4、地板(floor)
      • 4.2.5、地砖(tile)
  • 五、代码实现
    • 5.1、工程结构
    • 5.2、ifelse实现需求
    • 5.3、 测试验证
  • 六、建造者模式重构代码
    • 6.1、工程结构
    • 6.2、代码实现
      • 6.2.1、定义装修包接口
      • 6.2.2、装修包实现
      • 6.2.3、建造者方法
    • 6.3、测试验证
  • 七、总结

 

一、前言

无论承接什么样的需求,是不是身边总有那么几个人代码写的烂,但是却时常有测试小姐姐过来聊天(求改bug)、有产品小伙伴送吃的(求写需求)、有业务小妹妹陪着改代码(求上线),直至领导都认为他的工作很重要,而在旁边的你只能蹭点吃的。

这样的小伙伴,可能把代码写的很直接,ifelse多用一点,满足于先临时支持一下,想着这也没什么的。而且这样的业务需求要的急又都是增删改查的内容,实在不想做设计。而如果有人提到说好好设计下,可能也会被反对不要过渡设计。

第一次完成产品需求实在是很快,但互联网的代码不比传统企业。在传统行业可能一套代码能用十年,但在互联网高速的迭代下你的工程,一年就要变动几十次。如果从一开始就想着只要完成功能就可以,那么随之而来的是后续的需求难以承接,每次看着成片成片的代码,实在不知如何下手。

在研发流程规范下执行,才能写出好程序!

一个项目的上线往往要经历业务需求产品设计研发实现测试验证上线部署正式开量,而这其中对研发非常重要的一换就是研发实现的过程,又可以包括为;架构选型功能设计设计评审代码实现代码评审单测覆盖率检查编写文档提交测试。所以在一些流程规范下,其实很难让你随意开发代码。

开发代码的过程不是炫技,就像盖房子如果不按照图纸来修建,回首就在山墙上搭一个厨房卫浴!可能在现实场景中这很荒唐,但在功能开发中却总有这样的代码。

所以我们也需要一些设计模式的标准思想,去建设代码结构,提升全局把控能力。

 

二、开发环境

1.JDK 1.8

2.Idea + Maven

工程 描述
itstack-demo-design-3-00 场景模拟工程,模拟装修过程中的套餐选择(豪华、田园、简约)
itstack-demo-design-3-01 使用一坨代码实现业务需求,也是对ifelse的使用
itstack-demo-design-3-02 通过设计模式优化改造代码,产生对比性从而学习

 

三、建造者模式介绍

详解Java实践之建造者模式

建造者模式所完成的内容就是通过将多个简单对象通过一步步的组装构建出一个复杂对象的过程。

那么,哪里有这样的场景呢?

例如你玩王者荣耀的时的初始化界面;有三条路、有树木、有野怪、有守卫塔等等,甚至依赖于你的网络情况会控制清晰度。而当你换一个场景进行其他不同模式的选择时,同样会建设道路、树木、野怪等等,但是他们的摆放和大小都有不同。这里就可以用到建造者模式来初始化游戏元素。

而这样的根据相同的物料,不同的组装所产生出的具体的内容,就是建造者模式的最终意图,也就是;将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。

 

四、案例场景模拟

详解Java实践之建造者模式

这里我们模拟装修公司对于设计出一些套餐装修服务的场景。

很多装修公司都会给出自家的套餐服务,一般有;欧式豪华、轻奢田园、现代简约等等,而这些套餐的后面是不同的商品的组合。例如;一级&二级吊顶、多乐士涂料、圣象地板、马可波罗地砖等等,按照不同的套餐的价格选取不同的品牌组合,最终再按照装修面积给出一个整体的报价。

这里我们就模拟装修公司想推出一些套餐装修服务,按照不同的价格设定品牌选择组合,以达到使用建造者模式的过程。

 

4.1、场景模拟工程

itstack-demo-design-3-00

└── src

    └── main

        └── java

            └── org.itstack.demo.design

                ├── ceilling

                │   ├── LevelOneCeiling.java

                │   └── LevelTwoCeiling.java

                ├── coat

                │   ├── DuluxCoat.java

                │   └── LiBangCoat.java

                │   └── LevelTwoCeiling.java

                ├── floor

                │   ├── DerFloor.java

                │   └── ShengXiangFloor.java

                ├── tile

                │   ├── DongPengTile.java

                │   └── MarcoPoloTile.java

                └── Matter.java

在模拟工程中提供了装修中所需要的物料;ceilling(吊顶)coat(涂料)floor(地板)tile(地砖),这么四项内容。(实际的装修物料要比这个多的多

 

4.2、场景简述

 

4.2.1、物料接口

public interface Matter {

    String scene();      // 场景;地板、地砖、涂料、吊顶

    String brand();      // 品牌

    String model();      // 型号

    BigDecimal price();  // 价格

    String desc();       // 描述

}

物料接口提供了基本的信息,以保证所有的装修材料都可以按照统一标准进行获取。

 

4.2.2、吊顶(ceiling)

一级顶

public class LevelOneCeiling implements Matter {

    public String scene() {
        return "吊顶";
    }

    public String brand() {
        return "装修公司自带";
    }

    public String model() {
        return "一级顶";
    }

    public BigDecimal price() {
        return new BigDecimal(260);
    }

    public String desc() {
        return "造型只做低一级,只有一个层次的吊顶,一般离顶120-150mm";
    }

}

二级顶

public class LevelTwoCeiling  implements Matter {

    public String scene() {
        return "吊顶";
    }

    public String brand() {
        return "装修公司自带";
    }

    public String model() {
        return "二级顶";
    }

    public BigDecimal price() {
        return new BigDecimal(850);
    }

    public String desc() {
        return "两个层次的吊顶,二级吊顶高度一般就往下吊20cm,要是层高很高,也可增加每级的厚度";
    }
    
}

 

4.2.3、涂料(coat)

多乐士

public class DuluxCoat  implements Matter {

    public String scene() {
        return "涂料";
    }

    public String brand() {
        return "多乐士(Dulux)";
    }

    public String model() {
        return "第二代";
    }

    public BigDecimal price() {
        return new BigDecimal(719);
    }

    public String desc() {
        return "多乐士是阿克苏诺贝尔旗下的著名建筑装饰油漆品牌,产品畅销于全球100个国家,每年全球有5000万户家庭使用多乐士油漆。";
    }
    
}

立邦

public class LiBangCoat implements Matter {

    public String scene() {
        return "涂料";
    }

    public String brand() {
        return "立邦";
    }

    public String model() {
        return "默认级别";
    }

    public BigDecimal price() {
        return new BigDecimal(650);
    }

    public String desc() {
        return "立邦始终以开发绿色产品、注重高科技、高品质为目标,以技术力量不断推进科研和开发,满足消费者需求。";
    }

}

 

4.2.4、地板(floor)

德尔

public class DerFloor implements Matter {

    public String scene() {
        return "地板";
    }

    public String brand() {
        return "德尔(Der)";
    }

    public String model() {
        return "A+";
    }

    public BigDecimal price() {
        return new BigDecimal(119);
    }

    public String desc() {
        return "DER德尔集团是全球领先的专业木地板制造商,北京2008年奥运会家装和公装地板供应商";
    }
    
}

圣象

public class ShengXiangFloor implements Matter {

    public String scene() {
        return "地板";
    }

    public String brand() {
        return "圣象";
    }

    public String model() {
        return "一级";
    }

    public BigDecimal price() {
        return new BigDecimal(318);
    }

    public String desc() {
        return "圣象地板是中国地板行业著名品牌。圣象地板拥有中国驰名商标、中国名牌、国家免检、中国环境标志认证等多项荣誉。";
    }

}

 

4.2.5、地砖(tile)

东鹏

public class DongPengTile implements Matter {

    public String scene() {
        return "地砖";
    }

    public String brand() {
        return "东鹏瓷砖";
    }

    public String model() {
        return "10001";
    }

    public BigDecimal price() {
        return new BigDecimal(102);
    }

    public String desc() {
        return "东鹏瓷砖以品质铸就品牌,科技推动品牌,口碑传播品牌为宗旨,2014年品牌价值132.35亿元,位列建陶行业榜首。";
    }

}

马可波罗

public class MarcoPoloTile implements Matter {

    public String scene() {
        return "地砖";
    }

    public String brand() {
        return "马可波罗(MARCO POLO)";
    }

    public String model() {
        return "缺省";
    }

    public BigDecimal price() {
        return new BigDecimal(140);
    }

    public String desc() {
        return "“马可波罗”品牌诞生于1996年,作为国内最早品牌化的建陶品牌,以“文化陶瓷”占领市场,享有“仿古砖至尊”的美誉。";
    }

}

以上就是本次装修公司所提供的装修配置单,接下我们会通过案例去使用不同的物料组合出不同的套餐服务。

 

五、代码实现

讲道理没有ifelse解决不了的逻辑,不行就在加一行!

每一个章节中我们都会使用这样很直白的方式去把功能实现出来,在通过设计模式去优化完善。这样的代码结构也都是非常简单的,没有复杂的类关系结构,都是直来直去的代码。除了我们经常强调的这样的代码不能很好的扩展外,做一些例子demo工程还是可以的。

 

5.1、工程结构

itstack-demo-design-3-01

└── src

    └── main

        └── java

            └── org.itstack.demo.design

                └── DecorationPackageController.java

一个类几千行的代码你是否见过,嚯?那今天就让你见识一下有这样潜质的类!

 

5.2、ifelse实现需求

public class DecorationPackageController {

    public String getMatterList(BigDecimal area, Integer level) {

        List<Matter> list = new ArrayList<Matter>(); // 装修清单
        BigDecimal price = BigDecimal.ZERO;          // 装修价格

        // 豪华欧式
        if (1 == level) {

            LevelTwoCeiling levelTwoCeiling = new LevelTwoCeiling(); // 吊顶,二级顶
            DuluxCoat duluxCoat = new DuluxCoat();                   // 涂料,多乐士
            ShengXiangFloor shengXiangFloor = new ShengXiangFloor(); // 地板,圣象

            list.add(levelTwoCeiling);
            list.add(duluxCoat);
            list.add(shengXiangFloor);

            price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelTwoCeiling.price()));
            price = price.add(area.multiply(new BigDecimal("1.4")).multiply(duluxCoat.price()));
            price = price.add(area.multiply(shengXiangFloor.price()));

        }

        // 轻奢田园
        if (2 == level) {

            LevelTwoCeiling levelTwoCeiling = new LevelTwoCeiling(); // 吊顶,二级顶
            LiBangCoat liBangCoat = new LiBangCoat();                // 涂料,立邦
            MarcoPoloTile marcoPoloTile = new MarcoPoloTile();       // 地砖,马可波罗

            list.add(levelTwoCeiling);
            list.add(liBangCoat);
            list.add(marcoPoloTile);

            price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelTwoCeiling.price()));
            price = price.add(area.multiply(new BigDecimal("1.4")).multiply(liBangCoat.price()));
            price = price.add(area.multiply(marcoPoloTile.price()));

        }

        // 现代简约
        if (3 == level) {

            LevelOneCeiling levelOneCeiling = new LevelOneCeiling();  // 吊顶,二级顶
            LiBangCoat liBangCoat = new LiBangCoat();                 // 涂料,立邦
            DongPengTile dongPengTile = new DongPengTile();           // 地砖,东鹏

            list.add(levelOneCeiling);
            list.add(liBangCoat);
            list.add(dongPengTile);

            price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelOneCeiling.price()));
            price = price.add(area.multiply(new BigDecimal("1.4")).multiply(liBangCoat.price()));
            price = price.add(area.multiply(dongPengTile.price()));
        }

        StringBuilder detail = new StringBuilder("\r\n-------------------------------------------------------\r\n" +
                "装修清单" + "\r\n" +
                "套餐等级:" + level + "\r\n" +
                "套餐价格:" + price.setScale(2, BigDecimal.ROUND_HALF_UP) + " 元\r\n" +
                "房屋面积:" + area.doubleValue() + " 平米\r\n" +
                "材料清单:\r\n");

        for (Matter matter: list) {
            detail.append(matter.scene()).append(":").append(matter.brand()).append("、").append(matter.model()).append("、平米价格:").append(matter.price()).append(" 元。\n");
        }

        return detail.toString();

    }

}
  • 首先这段代码所要解决的问题就是接收入参;装修面积(area)、装修等级(level),根据不同类型的装修等级选择不同的材料。
  • 其次在实现过程中可以看到每一段if块里,都包含着不通的材料(吊顶,二级顶、涂料,立邦、地砖,马可波罗),最终生成装修清单和装修成本。
  • 最后提供获取装修详细信息的方法,返回给调用方,用于知道装修清单。

 

5.3、 测试验证

接下来我们通过junit单元测试的方式验证接口服务,强调日常编写好单测可以更好的提高系统的健壮度。

编写测试类:

@Test
public void test_DecorationPackageController(){
    DecorationPackageController decoration = new DecorationPackageController();
    // 豪华欧式
    System.out.println(decoration.getMatterList(new BigDecimal("132.52"),1));
    // 轻奢田园
    System.out.println(decoration.getMatterList(new BigDecimal("98.25"),2));
    // 现代简约
    System.out.println(decoration.getMatterList(new BigDecimal("85.43"),3));
}

结果:

-------------------------------------------------------

装修清单

套餐等级:1

套餐价格:198064.39 元

房屋面积:132.52 平米

材料清单:

吊顶:装修公司自带、二级顶、平米价格:850 元。

涂料:多乐士(Dulux)、第二代、平米价格:719 元。

地板:圣象、一级、平米价格:318 元。

 

 

-------------------------------------------------------

装修清单

套餐等级:2

套餐价格:119865.00 元

房屋面积:98.25 平米

材料清单:

吊顶:装修公司自带、二级顶、平米价格:850 元。

涂料:立邦、默认级别、平米价格:650 元。

地砖:马可波罗(MARCO POLO)、缺省、平米价格:140 元。

 

 

-------------------------------------------------------

装修清单

套餐等级:3

套餐价格:90897.52 元

房屋面积:85.43 平米

材料清单:

吊顶:装修公司自带、一级顶、平米价格:260 元。

涂料:立邦、默认级别、平米价格:650 元。

地砖:东鹏瓷砖、10001、平米价格:102 元。

 

 

Process finished with exit code 0

看到输出的这个结果,已经很有装修公司提供报价单的感觉了。以上这段使用ifelse方式实现的代码,目前已经满足的我们的也许功能。但随着老板对业务的快速发展要求,会提供很多的套餐针对不同的户型。那么这段实现代码将迅速扩增到几千行,甚至在修修改改中,已经像膏药一样难以维护。

 

六、建造者模式重构代码

接下来使用建造者模式来进行代码优化,也算是一次很小的重构。

建造者模式主要解决的问题是在软件系统中,有时候面临着"一个复杂对象"的创建工作,其通常由各个部分的子对象用一定的过程构成;由于需求的变化,这个复杂对象的各个部分经常面临着重大的变化,但是将它们组合在一起的过程却相对稳定。

这里我们会把构建的过程交给创建者类,而创建者通过使用我们的构建工具包,去构建出不同的装修套餐

 

6.1、工程结构

itstack-demo-design-3-02

└── src

    ├── main

    │   └── java

    │       └── org.itstack.demo.design

    │           ├── Builder.java    

    │           ├── DecorationPackageMenu.java

    │           └── IMenu.java 

    └── test

         └── java

             └── org.itstack.demo.design.test

                 └── ApiTest.java

建造者模型结构

详解Java实践之建造者模式

工程中有三个核心类和一个测试类,核心类是建造者模式的具体实现。与ifelse实现方式相比,多出来了两个二外的类。具体功能如下;

Builder,建造者类具体的各种组装由此类实现。DecorationPackageMenu,是IMenu接口的实现类,主要是承载建造过程中的填充器。相当于这是一套承载物料和创建者中间衔接的内容。

好,那么接下来会分别讲解几个类的具体实现。

 

6.2、代码实现

 

6.2.1、定义装修包接口

public interface IMenu {

    IMenu appendCeiling(Matter matter); // 吊顶

    IMenu appendCoat(Matter matter);    // 涂料

    IMenu appendFloor(Matter matter);   // 地板

    IMenu appendTile(Matter matter);    // 地砖

    String getDetail();                 // 明细 

}

接口类中定义了填充各项物料的方法;吊顶涂料地板地砖,以及最终提供获取全部明细的方法。

 

6.2.2、装修包实现

public class DecorationPackageMenu implements IMenu {

    private List<Matter> list = new ArrayList<Matter>();  // 装修清单
    private BigDecimal price = BigDecimal.ZERO;      // 装修价格

    private BigDecimal area;  // 面积
    private String grade;     // 装修等级;豪华欧式、轻奢田园、现代简约

    private DecorationPackageMenu() {
    }

    public DecorationPackageMenu(Double area, String grade) {
        this.area = new BigDecimal(area);
        this.grade = grade;
    }

    public IMenu appendCeiling(Matter matter) {
        list.add(matter);
        price = price.add(area.multiply(new BigDecimal("0.2")).multiply(matter.price()));
        return this;
    }

    public IMenu appendCoat(Matter matter) {
        list.add(matter);
        price = price.add(area.multiply(new BigDecimal("1.4")).multiply(matter.price()));
        return this;
    }

    public IMenu appendFloor(Matter matter) {
        list.add(matter);
        price = price.add(area.multiply(matter.price()));
        return this;
    }

    public IMenu appendTile(Matter matter) {
        list.add(matter);
        price = price.add(area.multiply(matter.price()));
        return this;
    }

    public String getDetail() {

        StringBuilder detail = new StringBuilder("\r\n-------------------------------------------------------\r\n" +
                "装修清单" + "\r\n" +
                "套餐等级:" + grade + "\r\n" +
                "套餐价格:" + price.setScale(2, BigDecimal.ROUND_HALF_UP) + " 元\r\n" +
                "房屋面积:" + area.doubleValue() + " 平米\r\n" +
                "材料清单:\r\n");

        for (Matter matter: list) {
            detail.append(matter.scene()).append(":").append(matter.brand()).append("、").append(matter.model()).append("、平米价格:").append(matter.price()).append(" 元。\n");
        }

        return detail.toString();
    }

}
  • 装修包的实现中每一个方法都会了 this,也就可以非常方便的用于连续填充各项物料。
  • 同时在填充时也会根据物料计算平米数下的报价,吊顶和涂料按照平米数适量乘以常熟计算。
  • 最后同样提供了统一的获取装修清单的明细方法。

 

6.2.3、建造者方法

public class Builder {

    public IMenu levelOne(Double area) {
        return new DecorationPackageMenu(area, "豪华欧式")
                .appendCeiling(new LevelTwoCeiling())    // 吊顶,二级顶
                .appendCoat(new DuluxCoat())             // 涂料,多乐士
                .appendFloor(new ShengXiangFloor());     // 地板,圣象
    }

    public IMenu levelTwo(Double area){
        return new DecorationPackageMenu(area, "轻奢田园")
                .appendCeiling(new LevelTwoCeiling())   // 吊顶,二级顶
                .appendCoat(new LiBangCoat())           // 涂料,立邦
                .appendTile(new MarcoPoloTile());       // 地砖,马可波罗
    }

    public IMenu levelThree(Double area){
        return new DecorationPackageMenu(area, "现代简约")
                .appendCeiling(new LevelOneCeiling())   // 吊顶,二级顶
                .appendCoat(new LiBangCoat())           // 涂料,立邦
                .appendTile(new DongPengTile());        // 地砖,东鹏
    }

}

建造者的使用中就已经非常容易了,统一的建造方式,通过不同物料填充出不同的装修风格;豪华欧式轻奢田园现代简约,如果将来业务扩展也可以将这部分内容配置到数据库自动生成。但整体的思想还可以使用创建者模式进行搭建。

 

6.3、测试验证

编写测试类:

@Test
public void test_Builder(){
    Builder builder = new Builder();
    // 豪华欧式
    System.out.println(builder.levelOne(132.52D).getDetail());
    // 轻奢田园
    System.out.println(builder.levelTwo(98.25D).getDetail());
    // 现代简约
    System.out.println(builder.levelThree(85.43D).getDetail());
}

结果:

-------------------------------------------------------

装修清单

套餐等级:豪华欧式

套餐价格:198064.39 元

房屋面积:132.52 平米

材料清单:

吊顶:装修公司自带、二级顶、平米价格:850 元。

涂料:多乐士(Dulux)、第二代、平米价格:719 元。

地板:圣象、一级、平米价格:318 元。

 

 

-------------------------------------------------------

装修清单

套餐等级:轻奢田园

套餐价格:119865.00 元

房屋面积:98.25 平米

材料清单:

吊顶:装修公司自带、二级顶、平米价格:850 元。

涂料:立邦、默认级别、平米价格:650 元。

地砖:马可波罗(MARCO POLO)、缺省、平米价格:140 元。

 

 

-------------------------------------------------------

装修清单

套餐等级:现代简约

套餐价格:90897.52 元

房屋面积:85.43 平米

材料清单:

吊顶:装修公司自带、一级顶、平米价格:260 元。

涂料:立邦、默认级别、平米价格:650 元。

地砖:东鹏瓷砖、10001、平米价格:102 元。

 

Process finished with exit code 0

测试结果是一样的,调用方式也基本类似。但是目前的代码结构却可以让你很方便的很有调理的进行扩展业务开发。而不是以往一样把所有代码都写到ifelse里面。

 

七、总结

通过上面对建造者模式的使用,已经可以摸索出一点心得。那就是什么时候会选择这样的设计模式,当:一些基本物料不会变,而其组合经常变化的时候,就可以选择这样的设计模式来构建代码。

此设计模式满足了单一职责原则以及可复用的技术、建造者独立、易扩展、便于控制细节风险。但同时当出现特别多的物料以及很多的组合后,类的不断扩展也会造成难以维护的问题。但这种设计结构模型可以把重复的内容抽象到数据库中,按照需要配置。这样就可以减少代码中大量的重复。

设计模式能带给你的是一些思想,但在平时的开发中怎么样清晰的提炼出符合此思路的建造模块,是比较难的。需要经过一些锻炼和不断承接更多的项目,从而获得这部分经验。有的时候你的代码写的好,往往是倒逼的,复杂的业务频繁的变化,不断的挑战!

以上就是详解Java实践之建造者模式的详细内容,更多关于Java 建造者模式的资料请关注三水点靠木其它相关文章!

Java/Android 相关文章推荐
详解Java实践之适配器模式
Jun 18 Java/Android
springBoot基于webSocket实现扫码登录
Jun 22 Java/Android
使用logback实现按自己的需求打印日志到自定义的文件里
Aug 30 Java/Android
IDEA2021.2配置docker如何将springboot项目打成镜像一键发布部署
Sep 25 Java/Android
SSM项目使用拦截器实现登录验证功能
Jan 22 Java/Android
JVM之方法返回地址详解
Feb 28 Java/Android
你知道Java Spring的两种事务吗
Mar 16 Java/Android
java高级用法JNA强大的Memory和Pointer
Apr 19 Java/Android
Springboot-cli 开发脚手架,权限认证,附demo演示
Apr 28 Java/Android
Java实现添加条码或二维码到Word文档
Jun 01 Java/Android
SpringCloud超详细讲解Feign声明式服务调用
Jun 21 Java/Android
Java中的Kafka为什么性能这么快及4大核心详析
Sep 23 Java/Android
新手入门Jvm-- JVM对象创建与内存分配机制
springboot利用redis、Redisson处理并发问题的操作
ConstraintValidator类如何实现自定义注解校验前端传参
SpringBoot项目中控制台日志的保存配置操作
Jun 18 #Java/Android
浅谈@Value和@Bean的执行顺序问题
Jun 16 #Java/Android
SpringBoot2 参数管理实践之入参出参与校验的方式
Jun 16 #Java/Android
SpringBoot生成License的实现示例
You might like
PHP strncasecmp字符串比较的小技巧
2011/01/04 PHP
php文件打包 下载之使用PHP自带的ZipArchive压缩文件并下载打包好的文件
2012/06/13 PHP
PHP实现微信退款功能
2018/10/02 PHP
php生成随机数/生成随机字符串的方法小结【5种方法】
2020/05/27 PHP
数据结构之利用PHP实现二分搜索树
2020/10/25 PHP
JS获取scrollHeight问题想到的标准问题
2007/05/27 Javascript
js 图片轮播(5张图片)
2008/12/30 Javascript
javascript string字符串优化问题
2011/07/31 Javascript
js 使FORM表单的所有元素不可编辑的示例代码
2013/10/17 Javascript
window.print打印指定div实例代码
2013/12/13 Javascript
jQuery鼠标事件总结
2016/10/13 Javascript
浅谈使用splice函数对数组中的元素进行删除时的注意事项
2016/12/04 Javascript
BootStrap 下拉菜单点击之后不会出现下拉菜单(下拉菜单不弹出)的解决方案
2016/12/14 Javascript
Jquery中.bind()、.live()、.delegate()和.on()之间的区别详解
2017/08/01 jQuery
jQuery实现腾讯信用界面(自制刻度尺)样式
2017/08/15 jQuery
Echarts之悬浮框中的数据排序问题
2018/11/08 Javascript
vue router 用户登陆功能的实例代码
2019/04/24 Javascript
layui button 按钮弹出提示窗口,确定才进行的方法
2019/09/06 Javascript
详解Vue的组件中data选项为什么必须是函数
2020/08/17 Javascript
[00:47]DOTA2荣耀之路6:天火,天火!
2018/05/30 DOTA
Python 输入一个数字判断成绩分数等级的方法
2018/11/15 Python
django的csrf实现过程详解
2019/07/26 Python
Python3网络爬虫开发实战之极验滑动验证码的识别
2019/08/02 Python
在Django中实现添加user到group并查看
2019/11/18 Python
Python pysnmp使用方法及代码实例
2020/08/24 Python
Python调用Redis的示例代码
2020/11/24 Python
乌克兰最大的家用电器和电子产品连锁店:Eldorado
2019/10/02 全球购物
领导检查欢迎词
2014/01/14 职场文书
教师专业自荐书范文
2014/02/10 职场文书
《欢乐的泼水节》教学反思
2014/04/22 职场文书
本科生求职信
2014/06/17 职场文书
员工试用期自我评价
2014/09/18 职场文书
食品仓管员岗位职责
2015/04/01 职场文书
2015年高校就业工作总结
2015/05/04 职场文书
申论不会写怎么办?教您掌握这6点思维和原则
2019/07/17 职场文书
《好妈妈胜过好老师》:每个孩子的优秀都是有源头的
2020/01/03 职场文书