百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 软件资讯 > 正文

Java 覆盖 jar 包内的方法

ninehua 2024-12-11 16:21 41 浏览

背景

有时候在 java 开发过程中会遇到这样的场景,比如说我们需要用 jar 包里的方法,但是 jar 包里的方法又不能满足当前的业务逻辑需要,而想直接下载 jar 包源码修改的话又会很麻烦,这个时候比较简便快捷能满足需要的办法就是覆写 jar 包中对应的方法,只需要改动你用的类的某一个方法就行,而不用去下载 jar 包源码再打包那么繁杂。

有一点需要注意的是需要保持方法中的参数不变,同时不能删除原有类的方法,但是可以新增一些方法来实现自己业务逻辑。

java 实现

实现代码其实比较简单,就是创建一个和 jar 包中需要重写类的路径,类名完全一样的类,然后 copy jar 包中原类的全部方法到 jar 包外的类中,根据需要改写原方法的业务逻辑或者新增方法来实现自己需要的业务逻辑,由于 jar 包外代码的优先级高于 maven 依赖 jar 包的优先级,这个时候你重写的类就会生效了。

比如 LoginController.class

重写后 LoginController.java

这里你可以在 jar 包外方法中新增自己的方法实现新的业务逻辑。

重写之后新增方法

整体源代码如下,集成了钉钉扫码登录功能

@Controller
public class LoginController extends BaseController
{
    @Autowired
    private DingDingProperties dingDingProperties;
    @Autowired
    private ISysUserService sysUserService;
    @Autowired
    private IDingDingService dingDingService;
    @Autowired
    private PasswordService passwordService;
    @Autowired
    private ISysUserDeptService sysUserDeptService;
    @Autowired
    private UserRoleMapper userRoleMapper;


    @GetMapping("/login")
    public String login(HttpServletRequest request, HttpServletResponse response, ModelMap mmap)
    {
        // 如果是Ajax请求,返回Json字符串。
        if (ServletUtils.isAjaxRequest(request))
        {
            return ServletUtils.renderString(response, "{\"code\":\"1\",\"msg\":\"未登录或登录超时。请重新登录\"}");
        }


        String dingtalkAppId = dingDingProperties.getAppId();
        //获取请求域名
        String scheme = request.getScheme();
        String serverName = request.getServerName();
        String dingtalkProjectUrl = scheme + "://" + serverName + "/";
        mmap.put("appId",dingtalkAppId);
        mmap.put("projectUrl",dingtalkProjectUrl);
        String msg = request.getParameter("msg");
        if (msg != null && msg != "") {
            if ("one".equals(msg)){
                mmap.put("msg","当前用户不存在,请联系管理员!");
            }else if ("two".equals(msg)) {
                mmap.put("msg","您好,请求连接远程服务器失败!");
            }
        }


        return "login";
    }


    @PostMapping("/login")
    @ResponseBody
    public AjaxResult ajaxLogin(String username, String password, Boolean rememberMe)
    {
        UsernamePasswordToken token = new UsernamePasswordToken(username, password, rememberMe);
        Subject subject = SecurityUtils.getSubject();
        try
        {
            subject.login(token);
            return success();
        }
        catch (AuthenticationException e)
        {
            String msg = "用户或密码错误";
            if (StringUtils.isNotEmpty(e.getMessage()))
            {
                msg = e.getMessage();
            }
            return error(msg);
        }
    }


    @GetMapping("/unauth")
    public String unauth()
    {
        return "error/unauth";
    }


    /**
     * 扫码登录并跳转到index
     * @param request
     * @return
     */
    @RequestMapping("/loginScan")
    public String loginSys(HttpServletRequest request)
    {
        String code = request.getParameter("code");
        String msg = "";
        //扫码登录
        try {
            OapiSnsGetuserinfoBycodeResponse.UserInfo userInfo = dingDingService.getUserInfoByCode(code);
            String unionId = userInfo.getUnionid();
            //String unionId = "dingdingceshiunionid";
            //根据unionId 获取用户信息
            SysUser use = sysUserService.selectUserByUnionId(unionId);
            if (use != null) {
                return doLoginSystem(use);
            }else {
                //未绑定 则自动添加账号 根据unionid获取userid
                OapiUserGetbyunionidResponse.UserGetByUnionIdResponse userIdByUnionIdV2 = dingDingService.getUserIdByUnionIdV2(unionId);
                if (userIdByUnionIdV2 != null && StringUtils.isNotEmpty(userIdByUnionIdV2.getUserid())) {
                    //根据userid获取用户详细信息
                    OapiV2UserGetResponse.UserGetResponse user = dingDingService.getUserDetailByUserid(userIdByUnionIdV2.getUserid(), null);
                    if (user != null) {
                        List<UserRole> userRoleList = new ArrayList();
                        //整理需要插入数据库字段
                        String userid = user.getUserid();
                        //判断当前用户是否已经插入过了
                        Long userId = null;
                        SysUser sysUser = sysUserService.selectUserByDingDingUserid(userid);
                        if (sysUser != null) {
                            userId = sysUser.getUserId();
                        }else {
                            sysUser = new SysUser();
                            sysUser.setDingdingUserid(userid);
                            sysUser.setAvatar(user.getAvatar());
                            sysUser.setUnionId(user.getUnionid());
                            sysUser.setPhonenumber(user.getMobile());
                            sysUser.setEmail(user.getEmail());
                            sysUser.setTitle(user.getTitle());
                            String userName = user.getName();
                            sysUser.setUserName(userName);
                            //中文转拼音作为loginName
                            String loginName = PinYinUtils.getPingYin(userName);
                            //根据当前公司人员重名情况,最大重名6人,故此处设置最大重名为10
                            StringBuilder sb = new StringBuilder(loginName).append(",");
                            for (int i = 1; i < 10; i++) {
                                sb.append(loginName+String.valueOf(i)).append(",");
                            }
                            String loginnames = sb.toString();
                            //判断一下当前是否已经存在了loginname
                            SysUser uniqueuser = sysUserService.selectLastUserByLoginName(Convert.toStrArray(loginnames));
                            if (uniqueuser != null) {
                                //获取当前重名用户的序号
                                String number = uniqueuser.getLoginName().replace(loginName, "");
                                //序号增加1在放回去拼接好
                                int i = 0;
                                if (StringUtils.isNotEmpty(number)) {
                                    i = Integer.parseInt(number) + 1;
                                }else {
                                    //说明当前loginname还没有后续数字
                                    i = 1;
                                }
                                sysUser.setLoginName(loginName+i);
                            }else {
                                sysUser.setLoginName(loginName);
                            }
                            sysUser.randomSalt();
                            //初始未编码前password为123456
                            sysUser.setPassword(passwordService.encryptPassword(sysUser.getLoginName(), "123456", sysUser.getSalt()));
                            sysUser.setCreateBy("扫码登录补充用户");
                            sysUser.setCreateTime(new Date());
                            sysUserService.insertSysUser(sysUser);
                            userId=sysUser.getUserId();


                            //插入用户的角色,初始用户都是普通角色
                            UserRole ur = new UserRole();
                            ur.setUserId(userId);
                            if (Constants.ONE_KEY.equals(user.getUserid()) || Constants.TWO_KEY.equals(user.getUserid())) {
                                ur.setRoleId(1L);
                            }else {
                                ur.setRoleId(2L);
                            }
                            userRoleList.add(ur);
                        }
                        //插入用户的部门
                        List<Long> deptIdList = user.getDeptIdList();
                        if (CollectionUtils.isNotEmpty(deptIdList)) {
                            //插入部门之前需要清除之前的人员部门关系,以防钉钉部门变动而系统未感知
                            int dele = sysUserDeptService.deleteSysUserDeptByUserId(userId);
                            //插入人员部门关系表
                            SysUserDept userDept = new SysUserDept();
                            userDept.setUserId(userId);
                            for (Long deptid : deptIdList) {
                                userDept.setDeptId(deptid);
                                userDept.setCreateBy("扫码登录补充用户");
                                userDept.setCreateTime(new Date());
                                sysUserDeptService.insertSysUserDept(userDept);
                            }
                        }


                        //插入角色
                        if (CollectionUtils.isNotEmpty(userRoleList)) {
                            userRoleMapper.batchUserRole(userRoleList);
                        }
                        //登录系统
                        return doLoginSystem(sysUser);


                    }else {
                        msg = "one";
                    }
                }else {
                    msg = "one";
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
            msg = "two";
        }
        return redirect("/login?msg="+msg);
    }


    /**
     * 登录系统
     * @param use
     * @return
     */
    private String doLoginSystem(SysUser use) {
        //已绑定账号则直接登录操作
        MyUsernamePasswordToken token = new MyUsernamePasswordToken(use.getUnionId(), use.getPassword(),false,true);


        // 登陆主流程
        Subject subject = SecurityUtils.getSubject();
        subject.login(token);


        return redirect("/index");
    }
}

复制代码

这里是改写后的登录代码,有需要的可以参考一下。


版权声明: 本文为 InfoQ 作者【六月的雨在infoQ】的原创文章。

原文链接:【https://xie.infoq.cn/article/3b69b24ed50c2e0bb9ec9f111】。

相关推荐

配合制导弹药!土耳其推出反无人机武器系统,可全天候拦截无人机

【军武次位面】作者:FriedrichLau据报道,日前土耳其军队装备了一款名为Sahin"沙欣"的反无人机武器系统。该系统由土耳其Aselsan公司开发,旨在摧毁迷你或小型无人机,...

威纶通常见问题18 威纶通触摸屏密码使用的相关介绍

密码1:屏右下角的系统设定列(向左的箭头)进入密码提醒:上传下载的默认密码均为6个1,若不慎忘记密码,因已加密所以原厂也无法解开,请妥善保管密码。若因不可抗力因素造成的忘记密码,只能将屏恢复出厂设置然...

轻松学会威纶通触摸屏上传与反编译

说到上传,大家并不陌生,上传顾名思义:就是把实际触摸屏画面上传到电脑EasyBuilderPro软件上,上传的作用是什么呢?可以修改触摸屏上原有的画面,所以说在上传这两字在工控中是非常重要的,接下来进...

DLL 可以被反编译吗?(dll文件可以编辑吗)

DLL可以被反编译,但存在一定难度DLL(动态链接库)是一种二进制文件,包含了可执行代码和数据。虽然它不像高级语言源代码那样容易被理解,但可以通过一些反编译工具进行逆向分析。不过,反编译后的结果通常...

复古游戏反编译工具遭任天堂下架 工具名或为诱因

Reddit用户mhjhacker1近日发表文章称,自己开发的一款“复古主机游戏逆向工程”工具被其他用户使用,反编译了一款任天堂四十年前的游戏,因而遭到了Itch.io的下架。而相关的另外三个工具也遭...

protobuf之序列化数据和反序列化数据基础知识

什么是protobufProtocalBuffers(简称protobuf)是谷歌的一项技术,用于结构化的数据序列化、反序列化。ProtocolBuffers是一种语言无关、平台无关、可扩展的...

MT管理器-简单实战-去除启动页(mt管理器去除登录界面)

每次打开APP都要看着启动页2秒,很烦人,既然如此我们把它去掉吧。学过Android开发的都知道,APP的每个界面都是一个Activity,然后你点击桌面上的图标之后,其实是启动了这个...

这两个强大的开源C#反编译逆向工具,探索C#桌面应用的小秘密

C#的应用也比较多,有时候,我们想要研究一下别人的优秀的项目,可能会借助一些非常规手段来学习。下面,我就分享几款开源的C#反编译工具。dnSpydnSpy是一个用C#开发,开源的调试器和.NET汇...

一款开源的.NET程序集反编译、编辑和调试神器

致力于发掘、评测和推荐各种高效实用的工具和软件,帮助您在工作和生活中事半功倍。项目介绍dnSpyEx是dnSpy项目的非官方Fork维护版本,一款开源、功能强大的.NET反编译工具和调试器,可用于在没...

记一次噩梦般的经历--论学会反编译的重要性

最近在做一个android项目,因为非开源,github上也没购有私有库,所以就没有提交github管理。昨天我感觉写到了一定程度,一个小的版本已经出来了,于是就想U盘和网盘都备份一下。结果出现了下面...

奇!AI(人工智能)能不能反编译二进制代码为高级语言Java或者C?

一直很好奇某些软件的制作过程,使用的算法。因为自己实现的,写的稀烂,虽然功能达到了,但处理速度和处理效率很低下,于是也尝试使用一些反编译软件,如IDAPro软件。想看看人家怎么实现,怎么做到效果那么...

你知道 Java 中关键字 enum 是一个语法糖吗?反编译枚举类

持续创作,加速成长!这是我参与「掘金日新计划·10月更文挑战」的第7天,点击查看活动详情写下这篇文章,也纯属于是一个机缘巧合,我一个非常要好的朋友程,也是刚刚踏上工作岗位。这个问题也是他踏上岗位...

SpringBoot项目Jar包加密,防止反编译

场景最近项目要求部署到其他公司的服务器上,但是又不想将源码泄露出去。要求对正式环境的启动包进行安全性处理,防止客户直接通过反编译工具将代码反编译出来。方案第一种方案使用代码混淆采用proguard-m...

安卓移动应用代码安全加固系统设计及实现

安卓平台已经逐渐成为最受欢迎的移动终端操作系统,基于安卓系统的软件应用数量众多,同时安全威胁也在不断增加。介绍了Android系统的安全风险及加固的核心技术,据此提出了一种安卓应用程序的安全加固系统,...

代码安全之代码混淆(代码混淆器是干什么的)

防止核心代码泄露以及代码合并每日辛辛苦苦码的代码被轻松破解?机密核心模块被盗用?数据库地址和语句暴露?……这些都是由代码未混淆加密,从而进行泄露的我们目前DLL文件存在的风险:一、直接引用二、反编译...