我的编程空间,编程开发者的网络收藏夹
学习永远不晚

怎么在Java中实现自定义注解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

怎么在Java中实现自定义注解

怎么在Java中实现自定义注解?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

注解是什么?

①、引用自维基百科的内容:
Java注解又称Java标注,是JDK5.0版本开始支持加入源代码的特殊语法 元数据 。

Java语言中的类、方法、变量、参数和包等都可以被标注。和Javadoc不同,Java标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java虚拟机可以保留标注内容,在运行时可以获取到标注内容。 当然它也支持自定义Java标注。

②、引用自网络的内容:
Java 注解是在 JDK5 时引入的新特性,注解(也被称为 元数据 )为我们在代码中添加信息提供了一种形式化的方法,使我们可以在稍后某个时刻非常方便地使用这些数据。

元注解是什么?

元注解 的作用就是负责注解其他注解。Java5.0定义了4个标准的meta-annotation(元注解)类型,它们被用来提供对其它 annotation类型作说明。

标准的元注解:

@Target
@Retention
@Documented
@Inherited
在详细说这四个元数据的含义之前,先来看一个在工作中会经常使用到的 @Autowired 注解,进入这个注解里面瞧瞧: 此注解中使用到了@Target、@Retention、@Documented 这三个元注解 。

@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface Autowired {    boolean required() default true;}

@Target元注解:

@Target注解,是专门用来限定某个自定义注解能够被应用在哪些Java元素上面的,标明作用范围;取值在java.lang.annotation.ElementType 进行定义的。

public enum ElementType {        TYPE,        FIELD,        METHOD,        PARAMETER,        CONSTRUCTOR,        LOCAL_VARIABLE,        ANNOTATION_TYPE,        PACKAGE}

根据此处可以知道 @Autowired 注解的作用范围:

// 可以作用在 构造方法、方法、方法形参、属性、注解类型 上@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})

@Retention元注解:

@Retention注解,翻译为持久力、保持力。即用来修饰自定义注解的生命周期。

注解的生命周期有三个阶段:

  • Java源文件阶段;

  • 编译到class文件阶段;

  • 运行期阶段;

同样使用了RetentionPolicy 枚举类型对这三个阶段进行了定义:

public enum RetentionPolicy {        SOURCE,        CLASS,        RUNTIME}

再详细描述下这三个阶段:

①、如果被定义为 RetentionPolicy.SOURCE,则它将被限定在Java源文件中,那么这个注解即不会参与编译也不会在运行期起任何作用,这个注解就和一个注释是一样的效果,只能被阅读Java文件的人看到;

②、如果被定义为 RetentionPolicy.CLASS,则它将被编译到Class文件中,那么编译器可以在编译时根据注解做一些处理动作,但是运行时JVM(Java虚拟机)会忽略它,并且在运行期也不能读取到;

③、如果被定义为 RetentionPolicy.RUNTIME,那么这个注解可以在运行期的加载阶段被加载到Class对象中。那么在程序运行阶段,可以通过反射得到这个注解,并通过判断是否有这个注解或这个注解中属性的值,从而执行不同的程序代码段。

注意:实际开发中的自定义注解几乎都是使用的 RetentionPolicy.RUNTIME 。

@Documented元注解:

@Documented注解,是被用来指定自定义注解是否能随着被定义的java文件生成到JavaDoc文档当中。

@Inherited元注解:

@Inherited注解,是指定某个自定义注解如果写在了父类的声明部分,那么子类的声明部分也能自动拥有该注解。

@Inherited注解只对那些@Target被定义为 ElementType.TYPE 的自定义注解起作用。

自定义注解实现:

在了解了上面的内容后,我们来尝试实现一个自定义注解:

怎么在Java中实现自定义注解

根据上面自定义注解中使用到的元注解得知:

①、此注解的作用范围,可以使用在类(接口、枚举)、方法上;

②、此注解的生命周期,被编译器保存在class文件中,而且在运行时会被JVM保留,可以通过反射读取;

自定义注解的简单使用:

上面已经创建了一个自定义的注解,那该怎么使用呢?下面首先描述下它简单的用法,后面将会使用其结合拦截器和AOP切面编程进行实战应用;

怎么在Java中实现自定义注解

应用场景实现

在了解了上面注解的知识后,我们乘胜追击,看看它的实际应用场景是肿么样的,以此加深下我们的理解;

实现的 Demo 项目是以 SpringBoot 实现的,项目工程结构图如下:

怎么在Java中实现自定义注解

场景一:自定义注解 + 拦截器 = 实现接口响应的包装

使用自定义注解 结合 拦截器 优雅的实现对API接口响应的包装。

在介绍自定义实现的方式之前,先简单介绍下普遍的实现方式,通过两者的对比,才能更加明显的发现谁最优雅。

普通的接口响应包装方式:
现在项目绝大部分都采用的前后端分离方式,所以需要前端和后端通过接口进行交互;目前在接口交互中使用最多的数据格式是 json,然后后端返回给前端的最为常见的响应格式如下:

{    #返回状态码    code:integer,           #返回信息描述    message:string,    #返回数据值    data:object}

项目中经常使用枚举类定义状态码和消息,代码如下:

public enum ResponseCode {    SUCCESS(1200, "请求成功"),    ERROR(1400, "请求失败");    private Integer code;    private String message;    private ResponseCode(Integer code, String message) {        this.code = code;        this.message = message;    }    public Integer code() {        return this.code;    }    public String message() {        return this.message;    }}

同时项目中也会设计一个返回响应包装类,代码如下:

import com.alibaba.fastjson.JSONObject;import java.io.Serializable;@SuppressWarnings("serial")public class Response<T> implements Serializable {        private T date;        private Integer code;        private String message;    public Response(T date, Integer code, String message) {        super();        this.date = date;        this.code = code;        this.message = message;    }    public T getDate() {        return date;    }    public void setDate(T date) {        this.date = date;    }    public Integer getCode() {        return code;    }    public void setCode(Integer code) {        this.code = code;    }    public String getMessage() {        return message;    }    public void setMessage(String message) {        this.message = message;    }    @Override    public String toString() {        return JSONObject.toJSONString(this);    }}

最后就是使用响应包装类和状态码枚举类 来实现返回响应的包装了:

@GetMapping("/user/findAllUser")public Response<List<User>> findAllUser() {    logger.info("开始查询所有数据...");    List<User> findAllUser = new ArrayList<>();    findAllUser.add(new User("木子雷", 26));    findAllUser.add(new User("公众号", 28));    // 返回响应进行包装    Response response = new Response(findAllUser, ResponseCode.SUCCESS.code(), ResponseCode.SUCCESS.message());    logger.info("response: {} \n", response.toString());    return response;}

在浏览器中输入网址: http://127.0.0.1:8080/v1/api/user/findAllUser 然后点击回车,得到如下数据:

{    "code": 1200,    "date": [        {            "age": 26,            "name": "木子雷"        },        {            "age": 28,            "name": "公众号"        }    ],    "message": "请求成功"}

通过看这中实现响应包装的方式,我们能发现什么问题吗?

答:代码很冗余,需要在每个接口方法中都进行响应的包装;使得接口方法包含了很多非业务逻辑代码;

有没有版本进行优化下呢? en en 思考中。。。。。 啊,自定义注解 + 拦截器可以实现呀!

自定义注解实现接口响应包装:
①、首先创建一个进行响应包装的自定义注解:

@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface ResponseResult {}

②、创建一个拦截器,实现对请求的拦截,看看请求的方法或类上是否使用了自定义的注解:

@Componentpublic class ResponseResultInterceptor implements HandlerInterceptor {        public static final String RESPONSE_ANNOTATION = "RESPONSE_ANNOTATION";        @Override    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)            throws Exception {        // 请求的接口方法        if (handler instanceof HandlerMethod) {            final HandlerMethod handlerMethod = (HandlerMethod) handler;            final Class<?> clazz = handlerMethod.getBeanType();            final Method method = handlerMethod.getMethod();            // 判断是否在类对象上加了注解            if (clazz.isAnnotationPresent(ResponseResult.class)) {                // 在请求中设置需要进行响应包装的属性标志,在下面的ResponseBodyAdvice增强中进行处理                request.setAttribute(RESPONSE_ANNOTATION, clazz.getAnnotation(ResponseResult.class));            } else if (method.isAnnotationPresent(ResponseResult.class)) {                // 在请求中设置需要进行响应包装的属性标志,在下面的ResponseBodyAdvice增强中进行处理                request.setAttribute(RESPONSE_ANNOTATION, method.getAnnotation(ResponseResult.class));            }        }        return true;    }}

③、创建一个增强Controller,实现对返回响应进行包装的增强处理:

@ControllerAdvicepublic class ResponseResultHandler implements ResponseBodyAdvice<Object> {    private final Logger logger = LoggerFactory.getLogger(this.getClass());        public static final String RESPONSE_ANNOTATION = "RESPONSE_ANNOTATION";        @Override    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {        ServletRequestAttributes ra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();        HttpServletRequest sr = (HttpServletRequest) ra.getRequest();        // 查询是否需要进行响应包装的标志        ResponseResult responseResult = (ResponseResult) sr.getAttribute(RESPONSE_ANNOTATION);        return responseResult == null ? false : true;    }        @Override    public Object beforeBodyWrite(Object responseBody, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {        logger.info("返回响应 包装进行中。。。");        Response response;        // boolean类型时判断一些数据库新增、更新、删除的操作是否成功        if (responseBody instanceof Boolean) {            if ((Boolean) responseBody) {                response = new Response(responseBody, ResponseCode.SUCCESS.code(), ResponseCode.SUCCESS.message());            } else {                response = new Response(responseBody, ResponseCode.ERROR.code(), ResponseCode.ERROR.message());            }        } else {            // 判断像查询一些返回数据的情况,查询不到数据返回 null;            if (null != responseBody) {                response = new Response(responseBody, ResponseCode.SUCCESS.code(), ResponseCode.SUCCESS.message());            } else {                response = new Response(responseBody, ResponseCode.ERROR.code(), ResponseCode.ERROR.message());            }        }        return response;    }}

④、最后在 Controller 中使用上我们的自定义注解;在 Controller 类上或者 方法上使用@ResponseResult自定义注解即可; 在浏览器中输入网址: http://127.0.0.1:8080/v1/api/user/findAllUserByAnnotation 进行查看:

// 自定义注解用在了方法上@ResponseResult@GetMapping("/user/findAllUserByAnnotation")public List<User> findAllUserByAnnotation() {    logger.info("开始查询所有数据...");    List<User> findAllUser = new ArrayList<>();    findAllUser.add(new User("木子雷", 26));    findAllUser.add(new User("公众号", 28));    logger.info("使用 @ResponseResult 自定义注解进行响应的包装,使controller代码更加简介");    return findAllUser;}

至此我们的接口返回响应包装自定义注解实现设计完成,看看代码是不是又简洁,又优雅呢。

总结:本文针对此方案只是进行了简单的实现,如果有兴趣的朋友可以进行更好的优化。

场景二:自定义注解 + AOP = 实现优雅的使用分布式锁

分布式锁的最常见的使用流程:

怎么在Java中实现自定义注解

先看看最为常见的分布式锁使用方式的实现,然后再聊聊自定义注解怎么优雅的实现分布式锁的使用。

普通的分布式锁使用方式:

怎么在Java中实现自定义注解

通过上面的代码可以得到一个信息:如果有很多方法中需要使用分布式锁,那么每个方法中都必须有获取分布式锁和释放分布式锁的代码,这样一来就会出现代码冗余;

那有什么好的解决方案吗? 自定义注解使代码变得更加简洁、优雅;

自定义注解优雅的使用分布式锁:
①、首先实现一个标记分布式锁使用的自定义注解:

@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface GetDistributedLock {    // 分布式锁 key    String lockKey();    // 分布式锁 value,默认为 lockValue    String lockValue() default "lockValue";    // 过期时间,默认为 300秒    int expireTime() default 300;}

②、定义一个切面,在切面中对使用了 @GetDistributedLock 自定义注解的方法进行环绕增强通知:

@Component@Aspectpublic class DistributedLockAspect {    private final Logger logger = LoggerFactory.getLogger(this.getClass());    @Autowired    RedisService redisService;        @Around(value = "@annotation(com.lyl.annotation.GetDistributedLock)")    public Boolean handlerDistributedLock(ProceedingJoinPoint joinPoint) {        // 通过反射获取自定义注解对象        GetDistributedLock getDistributedLock = ((MethodSignature) joinPoint.getSignature())                .getMethod().getAnnotation(GetDistributedLock.class);        // 获取自定义注解对象中的属性值        String lockKey = getDistributedLock.lockKey();        String LockValue = getDistributedLock.lockValue();        int expireTime = getDistributedLock.expireTime();        if (redisService.tryGetDistributedLock(lockKey, LockValue, expireTime)) {            // 获取分布式锁成功后,继续执行业务逻辑            try {                return (boolean) joinPoint.proceed();            } catch (Throwable throwable) {                logger.error("业务逻辑执行失败。", throwable);            } finally {                // 最终保证分布式锁的释放                redisService.releaseDistributedLock(lockKey, LockValue);            }        }        return false;    }}

③、最后,在 Controller 中的方法上使用 @GetDistributedLock 自定义注解即可;当某个方法上使用了 自定义注解,那么这个方法就相当于一个切点,那么就会对这个方法做环绕(方法执行前和方法执行后)增强处理;

在浏览器中输入网址: http://127.0.0.1:8080/v1/api/user/getDistributedLock 回车后触发方法执行:

// 自定义注解的使用@GetDistributedLock(lockKey = "userLock")@GetMapping("/user/getDistributedLock")public boolean getUserDistributedLock() {    logger.info("获取分布式锁...");    // 写具体的业务逻辑    return true;}

通过自定义注解的方式,可以看到代码变得更加简洁、优雅。

场景三:自定义注解 + AOP = 实现日志的打印

先看看最为常见的日志打印的方式,然后再聊聊自定义注解怎么优雅的实现日志的打印。

普通日志的打印方式:

怎么在Java中实现自定义注解

通过看上面的代码可以知道,如果每个方法都需要打印下日志,那将会存在大量的冗余代码;

自定义注解实现日志打印:
①、首先创建一个标记日志打印的自定义注解:

@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface PrintLog {}

②、定义一个切面,在切面中对使用了 @PrintLog 自定义注解的方法进行环绕增强通知:

@Component@Aspectpublic class PrintLogAspect {    private final Logger logger = LoggerFactory.getLogger(this.getClass());        @Around(value = "@annotation(com.lyl.annotation.PrintLog)")    public Object handlerPrintLog(ProceedingJoinPoint joinPoint) {        // 获取方法的名称        String methodName = joinPoint.getSignature().getName();        // 获取方法入参        Object[] param = joinPoint.getArgs();        StringBuilder sb = new StringBuilder();        for (Object o : param) {            sb.append(o + "; ");        }        logger.info("进入《{}》方法, 参数为: {}", methodName, sb.toString());        Object object = null;        // 继续执行方法        try {            object = joinPoint.proceed();        } catch (Throwable throwable) {            logger.error("打印日志处理error。。", throwable);        }        logger.info("{} 方法执行结束。。", methodName);        return object;    }}

③、最后,在 Controller 中的方法上使用 @PrintLog 自定义注解即可;当某个方法上使用了 自定义注解,那么这个方法就相当于一个切点,那么就会对这个方法做环绕(方法执行前和方法执行后)增强处理;

@PrintLog@GetMapping(value = "/user/findUserNameById/{id}", produces = "application/json;charset=utf-8")public String findUserNameById(@PathVariable("id") int id) {    // 模拟根据id查询用户名    String userName = "木子雷 公众号";    return userName;}

④、在浏览器中输入网址: http://127.0.0.1:8080/v1/api/user/findUserNameById/66 回车后触发方法执行,发现控制台打印了日志:

进入《findUserNameById》方法, 参数为: 66; 
findUserNameById 方法执行结束。。

看完上述内容,你们掌握怎么在Java中实现自定义注解的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注编程网行业资讯频道,感谢各位的阅读!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

怎么在Java中实现自定义注解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

怎么在Java中实现自定义注解

怎么在Java中实现自定义注解?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。注解是什么?①、引用自维基百科的内容:Java注解又称Java标注,是JDK5.0版本开始支持加入源
2023-06-14

怎么在java中自定义注解

这篇文章给大家介绍怎么在java中自定义注解,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java是什么Java是一门面向对象编程语言,可以编写桌面应用程序、Web应用程序、分布式系统和嵌入式系统应用程序。1、@Val
2023-06-07

Java怎么实现自定义注解

本文小编为大家详细介绍“Java怎么实现自定义注解”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java怎么实现自定义注解”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。概念概念:说明程序的。给计算机看的注释:用
2023-07-02

OutputFormat在java 中怎么实现自定义

本篇文章给大家分享的是有关OutputFormat在java 中怎么实现自定义,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。java 中 自定义OutputFormat实例代码
2023-05-31

怎么在java项目中对注解自定义

这期内容当中小编将会给大家带来有关怎么在java项目中对注解自定义,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。java 自定义注解的实例详解Java的Annotation是在5.0版本之后引入的,可以
2023-05-31

Java怎么自定义注解

这篇文章主要介绍“Java怎么自定义注解”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java怎么自定义注解”文章能帮助大家解决问题。注解注解为我们在代码中添加信息提供一种形式化的方法,使我们可以在
2023-07-05

如何在Java中自定义注解

这篇文章给大家介绍如何在Java中自定义注解,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一、自定义注解格式分析 Java 中自带的 @Override 注解 , 源码如下 :@Target(ElementType.M
2023-06-15

java中怎么自定义注解详解

在Java中,可以使用`@interface`关键字来定义注解。自定义注解的语法如下:public @interface CustomAnnotation {String value() default "";int number() d
java中怎么自定义注解详解
2023-10-28

java中什么是自定义注解

今天就跟大家聊聊有关java中什么是自定义注解,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;3.
2023-06-14

Java注解怎么自定义使用

这篇文章主要介绍了Java注解怎么自定义使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java注解怎么自定义使用文章都会有所收获,下面我们一起来看看吧。注解注解基本介绍注解概述:Java 注解(Annota
2023-07-05

Redis+AOP怎么自定义注解实现限流

今天小编给大家分享一下Redis+AOP怎么自定义注解实现限流的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。下载1,下载页面
2023-07-02

Feign怎么利用自定义注解实现路径转义

本篇内容主要讲解“Feign怎么利用自定义注解实现路径转义”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Feign怎么利用自定义注解实现路径转义”吧!背景近期由于项目中需要,所以需要通过Feig
2023-07-02

怎么用自定义注解

本篇内容介绍了“怎么用自定义注解”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!基本知识在Java中,注解分为两种,元注解和自定义注解。很多人
2023-06-16

Java自定义注解实现Redis自动缓存的方法

在实际开发中,可能经常会有这样的需要:从MySQL中查询一条数据(比如用户信息),此时需要将用户信息保存至Redis。刚开始我们可能会在查询的业务逻辑之后再写一段Redis相关操作的代码,时间长了后发现这部分代码实际上仅仅做了Redis的写
2023-05-31

怎么在Vue3中实现自定义指令

这篇文章主要介绍“怎么在Vue3中实现自定义指令”,在日常操作中,相信很多人在怎么在Vue3中实现自定义指令问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么在Vue3中实现自定义指令”的疑惑有所帮助!接下来
2023-07-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录