Groovy系列一 Groovy基础语法

news2024/11/24 11:06:02

目录

为什么要学习Groovy

Groovy 介绍

Groovy 特点

Groovy 实战

动态类型

简单明了的list,map类型

在groovy世界任何东西都是对象

属性操作变得更容易

GString

闭包

委派:delegate

Switch变得更简洁

元编程

强制类型检查

Elvis Operator

安全访问


为什么要学习Groovy

        作为Java为主语言的程序员,Groovy作为Java平台上的脚本语言(动态语言)在兼容方面是有得天独厚的优势。接触过Aandroid开发的同学大家都已经接触到啦Gradle构建项目(Gradle构建也支持Java项目的构建),Gradle就是采用Groovy实现的用于构建项目的框架。还有就是新版本的idea推荐使用Gradle进行插件开发。当然以上只是从客观方面阐述Groovy技术的发展趋势(包含但不仅限于这些方面)来说明我们需要学习,从自身层面来说多学多益,或者说有技术追求或者技术理想的同学,多学一门技术也是对自己的提升。接下来就让我们一起走进Groovy的世界吧。

Groovy 介绍

        Apache Groovy是一种强大的、可选的类型和动态语言,具有静态类型和静态编译功能,用于Java平台,旨在通过简洁、熟悉和易于学习的语法提高开发人员的生产力。它可以与任何Java程序顺利集成,并立即为您的应用程序提供强大的功能,包括脚本功能、领域特定语言创作、运行时和编译时元编程以及函数式编程。

                                                                                                                           -- groovy 官网

        以上是官网介绍,我个人的通俗理解如下,Groovy是用于Java虚拟机的一种敏捷的动态语言,是一种成熟的面向对象编程语言,又是一种纯粹的脚本语言。Groovy运行在JVM环境上,在语法上兼具java 语言和脚本语言特点,大大简化了语法。同时又具有闭包和动态语言中的其他特性,弥补了单纯的Java代码所不具备的动态性,我们可以在程序运行时任意修改代码逻辑,不需要重新发布。

        Groovy 1.0于2007年1月2日发布。 Groovy通过Apache License v 2.0发布。Groovy的官方网站是The Apache Groovy programming language

Groovy 特点

        Groovy中有以下特点(官网API文档:The Apache Groovy programming language - Groovy Development Kit):

  1. Groovy是基于Java虚拟机的一种面向对象的脚本语言,同时支持静态和动态类型。
  2. Groovy对于Java开发人员来说很简单,因为Java和Groovy的语法非常相似,可以使用现有的Java库。
  3. 支持运算符重载。
  4. 本地语法列表和关联数组。
  5. 对正则表达式的本地支持。
  6. 各种标记语言,如XML和HTML原生支持。
  7. Groovy扩展了java.lang.Object。

Groovy 实战

动态类型

        groovy的第一个特性就是动态特性,定义一个变量以后,可以对这个变量赋与任何类型的值;groovy提供了这种支持方式但是出于好的编码习惯而言,我们是杜绝这种开发方式的。

def v = 0;//定义了数值类型的变量v 
v = new Date();//然后又给它赋与Date类型的值. 
println(v);

简单明了的list,map类型

        我们可以使用groovy简单明了的定义list和map数据结构。不像在Java中国我们要先定义类型才能赋值,Groovy直接使用语法糖就可以快速赋值。具体看如下的代码事例,只列举了其中典型的API进行概念说明,剩余的API如果同学们感兴趣可以自行尝试练习。

  • 使用"[]"定义list列表,并对其直接操作
// list 可以看到的是编译器帮我们作何类型识别
def list = [1, 2]
// 这个是groovy的list的API从左边插入一个记录
list.leftShift(3)
// push是从插入到列表前面
list.push(0)
// add和我们java的List一样是追加到最后
list.add("abc")
// groovy中的<<可以对list数据类型的作用添加值,也是追加到最后面,需要注意的是在数字类型时是位运算操作
list << "<<号";
// "+"在groovy的语法中也是追加元素到集合中,并且是追加到最后
list += "加上+="

// 查看打印结果可以看到我在每个API的功能描述上的说明
// [0, 1, 2, 3, abc, <<号, 加上+=]
println(list)

// 也可以采用 lambda 表达式遍历集合
list.forEach({ println(it)})
  • 使用":"分隔key和value来定义map数据,key不需要用引号引起来,并且能使用key直接对map进行读写操作
def map = [a: 2, b: new Date()];

// 写入 直接追加到map集合中
map.put("aaaa","bbb")

println(map)
// 写入 直接采用Map的key进行操作
map.a = "a value change"

println(map)

// 结果如下 可以看到 key 为 "a" 的value被改变
[a:2, b:Sun May 28 11:19:56 CST 2023, aaaa:bbb]
[a:a value change, b:Sun May 28 11:19:56 CST 2023, aaaa:bbb]

在groovy世界任何东西都是对象

        怎么理解上面的概念呢,对照着Java语言来说,我们都知道在Java语言中有是区分对待基本类型和引用类型的。基本类型是没有方法调用的因为基本类型是按值访问的,在这个方面Groovy比Java更像是一个面向对象语言。

// 在groovy中,一切都是对象。一切都可以当成对象来使用;
// 即使在Java中的基本类型也更像是一个对象
100.times {println("hello world")}

属性操作变得更容易

        不同于Java中要操作一个对象中的属性必须通过get,set方法,即使是将属性定义成private。在groovy中,定义一个javabean,不再需要写getter,setter方法。读写属性直接用"."号操作即可。

class JavaBeans {
    String a
    // 即使是定义成private 也可以直接访问
    //    private String a
}

def beans = new JavaBeans();

beans.a = "object set property value"

println(beans.a)

GString

        GString是Groovy提供的快速处理字符串的工具。双引号""号的下字符串,可以直接通过 $var 或者 ${var}方式简单明了的嵌入变量。并且GStrng会调用包含的闭包 无参GString会调用闭包并打印我们期待的结果到Writer。

// case1
def a = "a"
def b = "b"
// a=a,b=b 快速替换字符串变量
println("a=${a},b=$b")

// case2
def name = 'lly'
def hello = "hello $name !"
// 结果:hello lly ! 快速替换字符串变量
println hello
// 注意这一行
name = '阳仔'
// 结果:hello lly !
println hello

// case3 :GStrng会调用包含的闭包 无参GString会调用闭包并打印我们期待的结果到Writer
def quote = "${-> name} , welcome"
// 结果:阳仔 , welcome
println quote
// 注意这一行
name = 'lisi'
// 结果:lisi , welcome
println quote

闭包

        在groovy中的闭包可以看成是一个代码块,它可以没有参数和返回值;它像java8以上的lambda语法或者像一个有一个方法的java内部类(这里类似于函数式编程规范定义)。在groovy闭包中的隐含内置变量,通过哪住的变量我们可以做一些获取单独元素操作的事情。比如迭代、判断等等其他业务。

it:当闭包的方法中,没有定义任何参数时,也可以直接用it变量;

this:跟java中的this一样;

owner:基本上跟this一样,只是除了一种情况,如果该闭包是在其他的闭包中定义的,owner就指向定义它的闭包对象。

// 闭包中内置了很多迭代方法,如find、findAll、collect等等。这里需要注意的是内置对象 it 的使用
def list = ['foo', 'bar']
def newList = []
list.collect(newList) {
    it.toUpperCase()
}
// 结果:[FOO, BAR]
println newList

//或者
list = ['foo', 'bar']
newList = []
list.collect(newList, {
    it.toUpperCase()
});
// 结果:[FOO, BAR]
println newList


// 在groovy闭包中的隐含内置变量

def a = {
    println "a this:" + this
    println "a owner:" + owner // ower指向b

    def b = {
        println "b this:" + this
        println "b owner:" + owner // ower指向b
    }
    b.call()

}
a.call()

// 结果:这里可以看到 this 和 owner 异同
a this:script.GroovyDemo1@78b236a0
a owner:script.GroovyDemo1@78b236a0
b this:script.GroovyDemo1@78b236a0
b owner:script.GroovyDemo1$_run_closure3@304b9f1a

委派:delegate

        Groovy中的delegate基本上是跟owner的一样,除非它通过Closure.setDelegate()委派指向新的对象。

def scriptClosure = {
    println "scriptClosure this:" + this
    println "scriptClosure owner:" + owner
    println "scriptClosure delegate:" + delegate
}
println "before setDelegate()"
scriptClosure.call()
scriptClosure.setDelegate("lly")
println "after setDelegate()"
scriptClosure.call()

// 结果:
scriptClosure this:script.GroovyDemo1@5c371e13
scriptClosure owner:script.GroovyDemo1@5c371e13
scriptClosure delegate:script.GroovyDemo1@5c371e13
after setDelegate()
scriptClosure this:script.GroovyDemo1@5c371e13
scriptClosure owner:script.GroovyDemo1@5c371e13
scriptClosure delegate:lly

Switch变得更简洁

        Groovy中的switch可以允许专入list,对象,范围...等表达式作为case依据

def x = 20;
switch (x) {
    case [1, 2, 3, 4, 5]:
        println("aaaaaa")
        break;
    case "foo":
        println("bbbbb")
    case 10..1000:
        println("ccccc")
        break;
    case Date:
        println("dddddd")
        break;
}

// 结果: 因为 20 在 10到1000 的范围内
ccccc

元编程

        在Groovy中,可以使用 metaClass类来,对元对象添加属性和方法。原因是由于在Groovy中,如果你的对象使用了一个它不曾定义过的方法不会报错,至少编译的时候不会报错,这就是因为Groovy特殊的运行时机制。你调用了一个对象的方法,他会先去检查这个类里面是否定义了这个属性和方法。如果没有的话会去MeteClass中去寻找,在Java中所有的类都会集成Object,而在Groovy中,所有的类都会继承GroovyObject,这个类里面就有获取MeteClass的方法。这样,就可以在运行中去动态的给对象添加属性和方法,甚至是静态的方法。如果MeteClass中也没有定义这个方法的话,就要去依次去寻找实体类里面是否实现了methodMissing和invokeMethod方法了,在这两个方法里面可以做特殊的处理。

// 基本元编程
// 对String类,添加一个uppers方法
String.metaClass.uppers = { -> toUpperCase() };
println "aaa".uppers()
//结果:AAA

// 对Interger类,添加一个say方法
Integer.metaClass.say = { -> "I am Interger" }
def i = new Integer(100);
println i.say()
// 结果:I am Interger

// 给对象添加属性、方法、静态方法
class Person {
    String name;
    Integer age;

    String eat() {
        println(name + "like eat")
    }
}
def p = new Person(name: "阳仔", age: 18)
p.eat()

// 给对象添加属性
p.metaClass.sex = '男'
// 给对象添加方法
p.metaClass.drink = {
    -> println "喜欢喝饮料"
}
// 给对象添加静态方法
p.metaClass.static.play = {
    -> println "喜欢玩游戏"
}

println "${p.name},今年${p.age}岁了,性別是${p.sex}"
p.drink()
p.play()

// 结果
阳仔like eat
阳仔,今年18岁了,性別是男
喜欢喝饮料
喜欢玩游戏

强制类型检查

        在之前的咱们的学习过程中发现,Groovy都是弱类型检查,不管是变量还是集合都可以接收任何类型的数据,如果我们想要在某些场景中限制某种类型就要用到@TypeChecked注解尽兴强制类型检查。在闭包中,如果需要强类型检查,可以像java语法一样定义参数类型。

class Foos{
    int  i = 42.0; //编译通过
}

@TypeChecked
class Foo{
    int  i = 42.0; //编译不通过
}

//在闭包中,如果需要强类型检查,可以像java语法一样定义参数类型
def list = ["a","b","c"]
// 编译不通过
//def list = ["a","b","c", 1]
list.collect {
    String it -> it.toUpperCase()
}
println(list)

Elvis Operator

        Groovy中采用Elvis Operator 运算,进一步简化三元运算符。

def a = null;
// 在定义b变量时,先判断a是否有值,如果有,就取a的值作为b的值,否则就取值"b"
def b = a ?: "b"; 
println(b)

安全访问

        跟我们Java中判空一样,只不过更加简洁而已。

def person;
//先判断person是否为不为null,然后,再调用getName方法
String name = person?.getName();
println(name)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/579473.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【五】设计模式~~~创建型模式~~~单例模式(Java)

【学习难度&#xff1a;★☆☆☆☆&#xff0c;使用频率&#xff1a;★★★★☆】 5.1. 模式动机 对于系统中的某些类来说&#xff0c;只有一个实例很重要&#xff0c;例如&#xff0c;一个系统中可以存在多个打印任务&#xff0c;但是只能有一个正在工作的任务&#xff1b;一…

一波三折,终于找到 src 漏洞挖掘的方法了【建议收藏】

0x01 信息收集 1、Google Hack 实用语法 迅速查找信息泄露、管理后台暴露等漏洞语法&#xff0c;例如&#xff1a; filetype:txt 登录 filetype:xls 登录 filetype:doc 登录 intitle:后台管理 intitle:login intitle:后台管理 inurl:admin intitle:index of /查找指定网站&…

C++:征服C指针:指针(二)

指针二 1. 指向数组的指针2. 多维数组三级目录 上一篇文章我们介绍了&#xff1a;什么是指针&#xff0c;指针常见的问题&#xff0c;本篇我们主要介绍 &#xff1a;指针与数组。 1. 指向数组的指针 int *p[n] : 指针数组&#xff0c; 它包括 n 个成员&#xff0c;每个成员都是…

探索Maven创建项目全过程(超详细~~~)

文章目录 1.Maven介绍2.Servlet介绍2.1 Servlet定义2.2 Servlet的主要任务 3.创建Servlet程序步骤3.1 创建项目3.2 引入依赖3.3 创建目录3.4编写代码3.5 打包程序3.6 部署程序3.7 验证结果 4.更方便的部署方式4.1.下载Tomcat插件4.2 配置Tomcat插件4.3运行项目 1.Maven介绍 Ma…

认识Tomcat

hi,大家好,今天为大家带来Tomcat的相关知识 &#x1f36d;1.Tomcat是什么 &#x1f36d;2.Tomcat的下载安装 &#x1f36d;3.Tomcat的目录结构 &#x1f36d;4.启动Tomcat &#x1f36d;5.部署博客系统到Tomcat &#x1f349;1.Tomcat是什么 我们之前也已经学了http,http…

【JAVAWEB】HTML的常见标签

目录 1.HTML结构 1.1认识HTML标签 1.2HTML文件基本结构 1.3标签层次结构 1.4快速生成代码框架 2.HTML常见标签 注释标签 标题标签&#xff1a;h1-h6 段落标签:p 换行标签&#xff1a;br 格式化标签 图片标签 超链接标签&#xff1a;a 表格标签 列表标签 表单标…

Windows 同时安装 MySQL5 和 MySQL8 版本

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是Rockey&#xff0c;不知名企业的不知名开发着 &#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&#x1f44d;一下博主哦 &#x1f4dd;联系方式&#xff1a;he18339193956&#xff0c;…

MySQL 00 : MySQL_数据库shell登录时遇到的问题

问题1描述&#xff1a;输入链接数据块的命令提示 sh:mysgl:command not found 解决:第一步配置环境变量来解决 1、vim /etc/profile 2、末尾写入export PATH$PATH:/usr/local/mysql/bin 3、保存 4、执行 source /etc/profile 第二部 问题描述 Mac通过MAMP安装MySQL时&#…

K8s之Replicaset控制器详解

文章目录 一、ReplicaSet 控制器介绍二、ReplicaSet案例1、Pod副本扩缩容案例2、Pod更新版本案例 一、ReplicaSet 控制器介绍 官方中文参考文档&#xff1a; ReplicaSet是k8s中一种资源对象&#xff0c;简写 rs&#xff0c;用于管理Pod副本数量和健康状态&#xff0c;在spec.…

MySQL如何导入大量数据?

有时我们会遇到需要将大量数据导入MySQL的需求&#xff0c;一般数据存储在csv或者txt中&#xff0c;数据由","分隔。这里提供两种方案供大家选择。 一、创建测试表 为了测试&#xff0c;我们先创建数据库和表&#xff0c;并创建一个用户。 create database loadda…

js的BOM对象中的window、location使用

说明&#xff1a;BOM的全称是Browser Object Model&#xff0c;浏览器对象模型&#xff0c;有Window&#xff08;浏览器窗口&#xff09;、Navigator&#xff08;浏览器&#xff09;、Screen&#xff08;屏幕&#xff09;、History&#xff08;历史记录&#xff09;和Location&…

js函数、Array和String的定义和使用

函数 js中的函数&#xff0c;类似java中的方法 <script>// 定义函数方式一function addA(x, y) {return x y;}// 定义函数方式二&#xff0c;注意addB是函数的名称&#xff0c;不是返回值let addB function(x, y) {return x * y;}</script><script>docume…

vscode语音插件开发-在nodejs里面转换音频文件格式并压缩导出zip格式

继我的上一篇文章&#xff1a;&#xff01;vscode录音及语音实时转写插件开发并在工作区生成本地mp3文件 踩坑日记 音频格式转换这里我使用的是ffmpeg第三方库&#xff0c;官网文档。 ffmpeg简介 FFmpeg是一个跨平台的自由软件&#xff0c;用于实时音频、视频流以及音频、视频…

Vue电商项目--详情页面--产品售卖属性

滚动行为 开发某一个产品的详情页面&#xff1f; 1.静态组件 但是这个详情页&#xff0c;我们还没有注册为路由 当点击商品图片的时候&#xff0c;跳转到详情页面&#xff0c;在路由跳转的时候需要带上产品的ID给详情页面 干掉a标签换成router-link标签 这里要注意的一点是…

MybatisPlus处理业务数据新思路

文章目录 [toc] 1.问题2.新姿势3.总结 1.问题 由于之前搞了一个停车场的项目&#xff0c;有一个区域的停车场收费被投诉了被举报没有按照政府要求来计费&#xff0c;所以需要将之前的那些停车计费的数据重新计算&#xff0c;将多缴费的钱拿去交罚款&#xff0c;这个就很坑&…

MongoDB—Indexes

一、Indexes 索引支持在MongoDB中高效执行查询。没有 索引&#xff0c;MongoDB必须执行集合扫描&#xff0c;即扫描每个 集合中的文档&#xff0c;以选择与 查询语句。如果查询存在适当的索引&#xff0c; MongoDB可以使用索引来限制它必须的文档数量 检查。 索引是特殊的数据…

近期群晖容器查询注册表失败,无法下载镜像

前言 本以为是群晖的问题&#xff0c;后来发现大家也都遇到了类似的问题&#xff0c;查询注册表出现错误&#xff0c;无法搜索和下载docker镜像&#xff0c;本期就来介绍几个可平替的方案。 方案一&#xff1a;SSH拉取 使用SSH拉取首先要打开群晖的SSH功能&#xff1a;在【…

数据结构(C语言):顺序循环队列的基本操作

一、题目 设队列的元素类型为char&#xff0c;实现顺序循环队列的各种基本操作的程序&#xff1a; ① 初始化队列Q&#xff1b; ② 判断队列Q是否为空&#xff1b; ③ 入队操作。循环调用入队操作&#xff0c;将若干元素&#xff08;不少于10个&#xff09;入队&#xff1b…

LeetCode 49 字母异位词分组

LeetCode 49 字母异位词分组 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/group-anagrams/description/ 博主Github&#xff1a;https://github.com/GDUT-Rp/LeetCode 题目&#xff1a; 给你一个字符串数组&#x…

如何使用ChatGPT?初学者简单指南

Chat GPT 风靡全球&#xff0c;并在短时间内成为世界上增长最快的应用程序。随着所有关于其潜在影响的讨论&#xff0c;你可能会想知道&#xff1a;如何使用 ChatGPT&#xff1f; 使用 ChatGPT需要准备&#xff1a;一个 OpenAI 帐户和一个网络浏览器。 访问chat.openai.com并…