Java代码设计模式讲解二十三种设计模式(七)

简介: Java代码设计模式讲解二十三种设计模式

三、结构型模式

一共七种

3.1 适配器模式

(1)概念

适配器模式(Adapter Pattern)是指将一个类的接口转换成客户期望的另一个接口,使

原本的接口不兼容的类可以一起工作,属于结构型设计模式。

(2)适用场景

  • 1、已经存在的类,它的方法和需求不匹配(方法结果相同或相似)的情况。
  • 2、适配器模式不是软件设计阶段考虑的设计模式,是随着软件维护,由于不同产品、不同厂家造成功能类似而接口不相同情况下的解决方案。有点亡羊补牢的感觉。生活中也非常的应用场景,例如电源插转换头、手机充电转换头、显示器转接头。

在中国民用电都是 220V 交流电,但我们手机使用的锂电池使用的 5V 直流电。因此,我

们给手机充电时就需要使用电源适配器来进行转换。

(3)代码示例

创建 AC220 类,表示 220V 交流电:

  • AC220
package com.alibaba.design.adapterpattern.powerapapter;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/29-21:06
 */
public class AC220 {
    public int outputAC220V(){
        int output = 220;
        System.out.println("输出电流" + output + "V");
        return output;
    }
}

创建 DC5 接口,表示 5V 直流电的标准:

  • DC5
package com.alibaba.design.adapterpattern.powerapapter;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/29-21:07
 */
public interface DC5 {
    public int outoupDC5V();
}

创建电源适配器 PowerAdapter 类:

  • PowerAdapter
package com.alibaba.design.adapterpattern.powerapapter;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/29-21:07
 */
public class PowerAdapter implements DC5 {
    private AC220 ac220;
    public PowerAdapter(AC220 ac220) {
        this.ac220 = ac220;
    }
    @Override
    public int outoupDC5V() {
        int adapterInput = ac220.outputAC220V();
        int adapterOutput = adapterInput / 44;
        System.out.println("使用PowerAdapter输入AC:" + adapterInput + "V,输出DC:" + adapterOutput + "V");
        return adapterOutput;
    }
}

客户端测试类PowerAdapterTest

  • PowerAdapterTest
package com.alibaba.design.adapterpattern.test;
import com.alibaba.design.adapterpattern.powerapapter.AC220;
import com.alibaba.design.adapterpattern.powerapapter.DC5;
import com.alibaba.design.adapterpattern.powerapapter.PowerAdapter;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/29-21:10
 */
public class PowerAdapterTest {
    public static void main(String[] args) {
        DC5 dc5 = new PowerAdapter(new AC220());
        dc5.outoupDC5V();
    }
}

上面的案例中,通过增加 PowerAdapter 电源适配器,实现了二者的兼容。

(4)重构第三登录自由适配的业务场景

下面我们来一个实际的业务场景,利用适配模式来解决实际问题。我们很早以前开发的老系统应该都有登录接口,但是随着业务的发展和社会的进步,单纯地依赖用户名密码登录显然不能满足用户需求了。现在,我们大部分系统都已经支持多种登录方式,如 QQ 登录、微信登录、手机登录、微博登录等等,同时保留用户名密码的登录方式。虽然登录形式丰富了,但是登录后的处理逻辑可以不必改,同样是将登录状态保存到 session,遵循开闭原则。首先创建统一的返回结果 ResultMsg 类:

package com.alibaba.design.adapterpattern.loginadapter;
/**
 * Created by Tom.
 */
public class ResultMsg {
    private int code;
    private String msg;
    private Object data;
    public ResultMsg(int code, String msg, Object data) {
        this.code = code;
        this.msg = msg;
        this.data = data;
    }
    public int getCode() {
        return code;
    }
    public void setCode(int code) {
        this.code = code;
    }
    public String getMsg() {
        return msg;
    }
    public void setMsg(String msg) {
        this.msg = msg;
    }
    public Object getData() {
        return data;
    }
    public void setData(Object data) {
        this.data = data;
    }
}

假设老系统的登录逻辑 SiginService:

package com.alibaba.design.adapterpattern.loginadapter.v1.service;
import com.alibaba.design.adapterpattern.loginadapter.Member;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom.
 */
public class SiginService {
    /**
     * 注册方法
     * @param username
     * @param password
     * @return
     */
    public ResultMsg regist(String username, String password){
            return  new ResultMsg(200,"注册成功",new Member());
    }
    /**
     * 登录的方法
     * @param username
     * @param password
     * @return
     */
    public ResultMsg login(String username,String password){
        return null;
    }
}

为了遵循开闭原则,老系统的代码我们不会去修改。那么下面开启代码重构之路,先创建 Member 类:

package com.alibaba.design.adapterpattern.loginadapter;
/**
 * Created by Tom.
 */
public class Member {
    private String username;
    private String password;
    private String mid;
    private String info;
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
    public String getMid() {
        return mid;
    }
    public void setMid(String mid) {
        this.mid = mid;
    }
    public String getInfo() {
        return info;
    }
    public void setInfo(String info) {
        this.info = info;
    }
}

创建一个新的类SinginForThirdService继承原来的逻辑,运行非常稳定的代码我们不去改动:

package com.alibaba.design.adapterpattern.loginadapter.v1.service;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom on 2019/3/16.
 */
public class SinginForThirdService extends SiginService {
    public ResultMsg loginForQQ(String openId){
        //1、openId是全局唯一,我们可以把它当做是一个用户名(加长)
        //2、密码默认为QQ_EMPTY
        //3、注册(在原有系统里面创建一个用户)
        //4、调用原来的登录方法
        return loginForRegist(openId,null);
    }
    public ResultMsg loginForWechat(String openId){
        return null;
    }
    public ResultMsg loginForToken(String token){
        //通过token拿到用户信息,然后再重新登陆了一次
        return  null;
    }
    public ResultMsg loginForTelphone(String telphone,String code){
        return null;
    }
    public ResultMsg loginForRegist(String username,String password){
        super.regist(username,null);
        return super.login(username,null);
    }
}

客户端测试代码

package com.alibaba.design.adapterpattern.loginadapter.v1;
import com.alibaba.design.adapterpattern.loginadapter.v1.service.SinginForThirdService;
/**
 * Created by Tom on 2019/3/16.
 */
public class SiginForThirdServiceTest {
    public static void main(String[] args) {
        SinginForThirdService service = new SinginForThirdService();
        service.login("tom","123456");
        service.loginForQQ("sdfasdfasf");
        service.loginForWechat("sdfasfsa");
    }
}

通过这么一个简单的适配,完成了代码兼容。当然,我们代码还可以更加优雅,根据不

同的登录方式,创建不同的 Adapter。首先,创建 LoginAdapter 接口:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * 在适配器里面,这个接口是可有可无,不要跟模板模式混淆
 * 模板模式一定是抽象类,而这里仅仅只是一个接口
 * Created by Tom 
 */
public interface LoginAdapter {
    boolean support(Object adapter);
    ResultMsg login(String id, Object adapter);
}

分别实现不同的登录适配,QQ 登录 LoginForQQAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom on 2019/3/16.
 */
public class LoginForQQAdapter implements LoginAdapter {
    @Override
    public boolean support(Object adapter) {
        return adapter instanceof LoginForQQAdapter;
    }
    @Override
    public ResultMsg login(String id, Object adapter) {
        return null;
    }
}

新浪微博登录 LoginForSinaAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom.
 */
public class LoginForSinaAdapter implements LoginAdapter {
    @Override
    public boolean support(Object adapter) {
        return adapter instanceof LoginForSinaAdapter;
    }
    @Override
    public ResultMsg login(String id, Object adapter) {
        return null;
    }
}

手机号登录 LoginForTelAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom.
 */
public class LoginForTelAdapter implements LoginAdapter {
    @Override
    public boolean support(Object adapter) {
        return adapter instanceof LoginForTelAdapter;
    }
    @Override
    public ResultMsg login(String id, Object adapter) {
        return null;
    }
}

Token 自动登录 LoginForTokenAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom.
 */
public class LoginForTokenAdapter implements LoginAdapter {
    @Override
    public boolean support(Object adapter) {
        return adapter instanceof LoginForTokenAdapter;
    }
    @Override
    public ResultMsg login(String id, Object adapter) {
        return null;
    }
}

微信登录 LoginForWechatAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2.adapters;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * Created by Tom.
 */
public class LoginForWechatAdapter implements LoginAdapter {
    @Override
    public boolean support(Object adapter) {
        return adapter instanceof LoginForWechatAdapter;
    }
    @Override
    public ResultMsg login(String id, Object adapter) {
        return null;
    }
}

然后,创建第三方登录兼容接口 IPassportForThird:

package com.alibaba.design.adapterpattern.loginadapter.v2;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
/**
 * 只扩展
 * Created by Tom on 2019/3/16.
 */
public interface IPassportForThird {
    /**
     * QQ登录
     * @param id
     * @return
     */
    ResultMsg loginForQQ(String id);
    /**
     * 微信登录
     * @param id
     * @return
     */
    ResultMsg loginForWechat(String id);
    /**
     * 记住登录状态后自动登录
     * @param token
     * @return
     */
    ResultMsg loginForToken(String token);
    /**
     * 手机号登录
     * @param telphone
     * @param code
     * @return
     */
    ResultMsg loginForTelphone(String telphone, String code);
    /**
     * 注册后自动登录
     * @param username
     * @param passport
     * @return
     */
    ResultMsg loginForRegist(String username, String passport);
}

实现兼容 PassportForThirdAdapter:

package com.alibaba.design.adapterpattern.loginadapter.v2;
import com.alibaba.design.adapterpattern.loginadapter.ResultMsg;
import com.alibaba.design.adapterpattern.loginadapter.v1.service.SiginService;
import com.alibaba.design.adapterpattern.loginadapter.v2.adapters.*;
/**
 * 结合策略模式、工厂模式、适配器模式
 * Created by Tom on 2019/3/16.
 */
public class PassportForThirdAdapter extends SiginService implements IPassportForThird {
    @Override
    public ResultMsg loginForQQ(String id) {
//        return processLogin(id,RegistForQQAdapter.class);
        return processLogin(id, LoginForQQAdapter.class);
    }
    @Override
    public ResultMsg loginForWechat(String id) {
        return processLogin(id, LoginForWechatAdapter.class);
    }
    @Override
    public ResultMsg loginForToken(String token) {
        return processLogin(token, LoginForTokenAdapter.class);
    }
    @Override
    public ResultMsg loginForTelphone(String telphone, String code) {
        return processLogin(telphone, LoginForTelAdapter.class);
    }
    @Override
    public ResultMsg loginForRegist(String username, String passport) {
        super.regist(username,passport);
        return super.login(username,passport);
    }
    private ResultMsg processLogin(String key,Class<? extends LoginAdapter> clazz){
        try{
            //适配器不一定要实现接口
            LoginAdapter adapter = clazz.newInstance();
            //判断传过来的适配器是否能处理指定的逻辑
            if(adapter.support(adapter)){
                return adapter.login(key,adapter);
            }
        }catch (Exception e){
            e.printStackTrace();
        }
        return null;
    }
    //类图的快捷键  Ctrl + Alt + Shift + U
}

客户端测试代码:

package com.alibaba.design.adapterpattern.loginadapter.v2;
/**
 * Created by Tom.
 */
public class PassportTest {
    public static void main(String[] args) {
        IPassportForThird passportForThird = new PassportForThirdAdapter();
        passportForThird.loginForQQ("");
    }
}

最后,来看一下类图:

至此,我们在遵循开闭原则的前提下,完整地实现了一个兼容多平台登录的业务场景。

(5)模式在源码中的体现

Spring 中适配器模式也应用得非常广泛,例如:SpringAOP 中的 AdvisorAdapter 类,它有三个实现类 MethodBeforeAdviceAdapter、AfterReturningAdviceAdapter 和ThrowsAdviceAdapter,先来看顶层接口 AdvisorAdapter 的源代码:

package org.springframework.aop.framework.adapter;
import org.aopalliance.aop.Advice;
import org.aopalliance.intercept.MethodInterceptor;
import org.springframework.aop.Advisor;
public interface AdvisorAdapter {
  boolean supportsAdvice(Advice var1);
  MethodInterceptor getInterceptor(Advisor var1);
}

再看 MethodBeforeAdviceAdapter 类:

package org.springframework.aop.framework.adapter;
import java.io.Serializable;
import org.aopalliance.aop.Advice;
import org.aopalliance.intercept.MethodInterceptor;
import org.springframework.aop.Advisor;
import org.springframework.aop.MethodBeforeAdvice;
class MethodBeforeAdviceAdapter implements AdvisorAdapter, Serializable {
    MethodBeforeAdviceAdapter() {
    }
    public boolean supportsAdvice(Advice advice) {
      return advice instanceof MethodBeforeAdvice;
    }
    public MethodInterceptor getInterceptor(Advisor advisor) {
        MethodBeforeAdvice advice = (MethodBeforeAdvice)advisor.getAdvice();
        return new MethodBeforeAdviceInterceptor(advice);
    }
}

Spring 会根据不同的 AOP 配置来确定使用对应的 Advice,跟策略模式不同的一个方法可以同时拥有多个 Advice。

(6)适配器模式的优缺点

  • 优点:
    1、能提高类的透明性和复用,现有的类复用但不需要改变。
    2、目标类和适配器类解耦,提高程序的扩展性。
    3、在很多业务场景中符合开闭原则。
  • 缺点:
    1、适配器编写过程需要全面考虑,可能会增加系统的复杂性。
    2、增加代码阅读难度,降低代码可读性,过多使用适配器会使系统代码变得凌乱。

3.2 代理模式

(1) 概念

代理模式(ProxyPattern)的定义也非常简单,是指为其他对象提供一种代理,以控制对这个对象的访问。

代理对象在客服端和目标对象之间起到中介作用,代理模式属于结构型设计模式。

(2)适用场景

使用代理模式主要有两个目的:一保护目标对象,二增强目标对象。

Subject 是顶层接口,RealSubject 是真实对象(被代理对象),Proxy 是代理对象,代理对象持有被代理对象的引用,客户端调用代理对象方法,同时也调用被代理对象的方法,但是在代理对象前后增加一些处理。在代码中,我们想到代理,就会理解为是代码增强,其实就是在原本逻辑前后增加一些逻辑,而调用者无感知。代理模式属于结构型模式,有静态代理和动态代理

(3)代码示例

1. 静态代理模式

举个例子:人到了适婚年龄,父母总是迫不及待希望早点抱孙子。而现在社会的人在各种压力之下,都选择晚婚晚育。于是着急的父母就开始到处为自己的子女相亲,比子女自己还着急。这个相亲的过程,就是一种我们人人都有份的代理。来看代码实现:

顶层接口 Person:

package com.alibaba.design.proxypattern;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-10:54
 */
public interface Person {
    public void findLove();
}

儿子要找对象,实现 Son 类:

package com.alibaba.design.proxypattern.staticproxy;
import com.alibaba.design.proxypattern.Person;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-10:55
 */
public class Son implements Person {
    @Override
    public void findLove() {
        System.out.println("儿子要求:肤白貌美大长腿");
    }
}

父亲要帮儿子相亲,实现 Father 类:

package com.alibaba.design.proxypattern.staticproxy;
import com.alibaba.design.proxypattern.Person;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-10:55
 */
public class Father implements Person {
    private Son son;
    public Father(Son son){
        this.son = son;
    }
    @Override
    public  void findLove(){
        System.out.println("父亲物色对象");
        this.son.findLove();
        System.out.println("双方父母同意,确立关系");
    }
    public void findJob(){
    }
}

来看测试代码:

  • FatherTest
package com.alibaba.design.proxypattern.staticproxy;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-10:58
 */
public class FatherTest {
    public static void main(String[] args) {
        Father father = new Father(new Son());
        father.findLove();
        //农村,媒婆
        //婚介所
        //大家每天都在用的一种静态代理的形式
        //对数据库进行分库分表
        //ThreadLocal
        //进行数据源动态切换
    }
}

2. 动态代理模式

动态代理和静态对比基本思路是一致的,只不过动态代理功能更加强大,随着业务的扩展适应性更强。如果还以找对象为例,使用动态代理相当于是能够适应复杂的业务场景。不仅仅只是父亲给儿子找对象,如果找对象这项业务发展成了一个产业,进而出现了媒婆、婚介所等这样的形式。那么,此时用静态代理成本就更大了,需要一个更加通用的解决方案,要满足任何单身人士找对象的需求。我们升级一下代码,先来看 JDK 实现方式:

创建媒婆(婚介)JDKMeipo 类:

package com.alibaba.design.proxypattern.dynamicproxy.jdkproxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-10:52
 */
public class JDKMeipo implements InvocationHandler {
    private Object target;
    public Object getInstance(Object target) throws Exception{
        this.target = target;
        Class<?> clazz = target.getClass();
        return Proxy.newProxyInstance(clazz.getClassLoader(),clazz.getInterfaces(),this);
    }
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        Object obj = method.invoke(this.target,args);
        after();
        return obj;
    }
    private void before(){
        System.out.println("我是媒婆,我要给你找对象,现在已经确认你的需求");
        System.out.println("开始物色");
    }
    private void after(){
        System.out.println("OK的话,准备办事");
    }
}

创建单身客户 Customer 类:

package com.alibaba.design.proxypattern.dynamicproxy.jdkproxy;
import com.alibaba.design.proxypattern.Person;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-11:11
 */
public class Customer implements Person {
    @Override
    public void findLove() {
        System.out.println("高富帅");
        System.out.println("身高180cm");
        System.out.println("有6块腹肌");
    }
}

测试代码

package com.alibaba.design.proxypattern.dynamicproxy.jdkproxy;
import java.lang.reflect.Method;
/**
 * @author zhouyanxiang
 * @create 2020-07-2020/7/30-11:11
 */
public class JDKProxyTest {
    public static void main(String[] args) {
        try {
            Object obj = new JDKMeipo().getInstance(new Customer());
            Method method = obj.getClass().getMethod("findLove",null);
            method.invoke(obj);
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

(4) 模式在源码中的体现

代理模式在 Spring 源码中的应用,先看 ProxyFactoryBean 核心的方法就是 getObject()方法,我们来看一下源码:

public Object getObject() throws BeansException {
    initializeAdvisorChain();
    if (isSingleton()) {
      return getSingletonInstance();
    }else {
      if (this.targetName == null) {
      logger.warn("Using non-singleton proxies with singleton targets is often undesirable. " + "Enable prototype proxies by setting the 'targetName' property.");
      } 
      return newPrototypeInstance();
    }
}

在 getObject()方法中,主要调用 getSingletonInstance()和 newPrototypeInstance();在 Spring 的配置中,如果不做任何设置,那么 Spring 代理生成的 Bean 都是单例对象。如果修改 scope 则每次创建一个新的原型对象。

Spring 利用动态代理实现 AOP 有两个非常重要的类,一个是 JdkDynamicAopProxy 类

和 CglibAopProxy 类,来看一下类图:

Spring 中的代理选择原则

  • 1、当 Bean 有实现接口时,Spring 就会用 JDK 的动态代理。
  • 2、当 Bean 没有实现接口时,Spring 选择 CGLib。
  • 3、Spring 可以通过配置强制使用 CGLib,只需在 Spring 的配置文件中加入如下代码:
<aop:aspectj-autoproxy proxy-target-class="true"/>

(5)代理模式的优缺点

  • 优点:
    1、代理模式能将代理对象与真实被调用的目标对象分离。
    2、一定程度上降低了系统的耦合度,扩展性好。
    3、可以起到保护目标对象的作用。
    4、可以对目标对象的功能增强。
  • 缺点:
    1、代理模式会造成系统设计中类的数量增加。
    2、在客户端和目标对象增加一个代理对象,会造成请求处理速度变慢。
    3、增加了系统的复杂度。

(6) 动态代理模式和静态代理模式的区别

  • 1、静态代理只能通过手动完成代理操作,如果被代理类增加新的方法,代理类需要同步
    新增,违背开闭原则。
  • 2、动态代理采用在运行时动态生成代码的方式,取消了对被代理类的扩展限制,遵循开
    闭原则。
  • 3、若动态代理要对目标类的增强逻辑扩展,结合策略模式,只需要新增策略类便可完成,
    无需修改代理类的代码。

3.3 装饰器模式

(1)概念

装饰者模式(Decorator Pattern)是指在不改变原有对象的基础之上,将功能附加到对象上,提供了比继承更有弹性的替代方案(扩展原有对象的功能),属于结构型模式。

(2)适用场景

装饰者模式在我们生活中应用也比较多如给煎饼加鸡蛋;给蛋糕加上一些水果;给房子装修等,为对象扩展一些额外的职责。装饰者在代码程序中适用于以下场景:

1、用于扩展一个类的功能或给一个类添加附加职责。

2、动态的给一个对象添加功能,这些功能可以再动态的撤销。


相关文章
|
2天前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
|
13天前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
|
16天前
|
设计模式 开发者 Python
Python编程中的设计模式:工厂方法模式###
本文深入浅出地探讨了Python编程中的一种重要设计模式——工厂方法模式。通过具体案例和代码示例,我们将了解工厂方法模式的定义、应用场景、实现步骤以及其优势与潜在缺点。无论你是Python新手还是有经验的开发者,都能从本文中获得关于如何在实际项目中有效应用工厂方法模式的启发。 ###
|
16天前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
26 5
Java反射机制:解锁代码的无限可能
|
9天前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
24 1
|
13天前
|
jenkins Java 测试技术
如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例详细说明
本文介绍了如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例,详细说明了从 Jenkins 安装配置到自动构建、测试和部署的全流程。文中还提供了一个 Jenkinsfile 示例,并分享了实践经验,强调了版本控制、自动化测试等关键点的重要性。
42 3
|
18天前
|
存储 安全 Java
系统安全架构的深度解析与实践:Java代码实现
【11月更文挑战第1天】系统安全架构是保护信息系统免受各种威胁和攻击的关键。作为系统架构师,设计一套完善的系统安全架构不仅需要对各种安全威胁有深入理解,还需要熟练掌握各种安全技术和工具。
50 10
|
14天前
|
分布式计算 Java MaxCompute
ODPS MR节点跑graph连通分量计算代码报错java heap space如何解决
任务启动命令:jar -resources odps-graph-connect-family-2.0-SNAPSHOT.jar -classpath ./odps-graph-connect-family-2.0-SNAPSHOT.jar ConnectFamily 若是设置参数该如何设置
|
12天前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
20天前
|
搜索推荐 Java 数据库连接
Java|在 IDEA 里自动生成 MyBatis 模板代码
基于 MyBatis 开发的项目,新增数据库表以后,总是需要编写对应的 Entity、Mapper 和 Service 等等 Class 的代码,这些都是重复的工作,我们可以想一些办法来自动生成这些代码。
28 6

热门文章

最新文章

  • 1
    C++一分钟之-设计模式:工厂模式与抽象工厂
    42
  • 2
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    46
  • 3
    C++一分钟之-C++中的设计模式:单例模式
    53
  • 4
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    37
  • 5
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    61
  • 6
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    56
  • 7
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    40
  • 8
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    49
  • 9
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    105
  • 10
    Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
    75