Java SE第7章 Java基础类库

Java SE第7章 Java基础类库

1. Java 程序的参数

先来分析一下java的**main()**方法~~~

public static void main(String[] args){}

首先,我们知道该方法是由 JVM 调用,为了让 JVM 自由调用此方法,所以应使用 public 修饰

JVM调用该方法时,不是先创建这个类的对象再调用main()方法,而是直接通过该类来调用该方法,所以应该用static修饰

因为方法的返回值会返回给调用此方法的对象,所以返回值将返回给 JVM ,没有什么意义,所以使用 void

对于形参列表String[] args ,则应由 JVM 负责赋值

赋值方法是

java ArgsTest Java Spring

ArgsTest 为类名,后面的两个参数 java,Spring 会以字符串形式加入到 args ,参数以空格为间隔,其中 args[0] = “java”args[1] = “Spring”

如果想加上空格,应

java ArgsTest "Java Spring"

2. 程序运行过程中接收用户输入

Scanner可以快速获取用户的键盘输入,它可以从文件,输入流,字符串中解析除基本类型值和字符串值

Scanner 主要提供两个方法扫描输入

  • hasNextXxx() : 用于判断是否有下一个输入项,Xxx可以是基本数据类型,如果判断下一个输入项是不是字符串,可以直接使用 hasNext()
  • nextXxx(): 获取下一个输入项

Scanner 默认使用空白(空格,tab,回车)为分隔符,同时可以使用 **useDelimiter(String pattern)**设置分隔符,该方法参数应该是一个正则表达式(之后再谈)

使用方法如下,将回车设置为分隔符

Scanner sc = new Scanner(System.in); //Systerm.in 表示标准输入,即键盘输入
sc.useDelimiter("\n");

再介绍两个逐行读取的方法,这与集合使用迭代器进行遍历形式十分相似

  • boolean hasNextLine(): 判断输入源是否有下一行
  • String nextLine(): 返回输入源下一行的字符串
while(sc.hasNextLine()){	//如果输入源有下一行
    System.out.println(sc.nextLine());	//输出下一行
}

前面创建Scanner对象时形参为System.in ,表示接收键盘输入,如果想读取文件,就应该传入一个文件对象

Scanner scFile = new Scanner(new File("ScannerFile.java"));//读取创建的文件对象

之后使用时就和前面一样,只是因为涉及文件,可能会有异常,则可以将异常抛出或抓取处理

3. System类相关用法

System类代表当前Java程序的运行平台,程序不能创建System类的对象,允许调用类变量和类方法

System类提供了代表标准输入(通常键盘),标准输出(通常显示器),错误输出的类变量(流形式),以可以通过对应set方法改变标准,并且提供了一些访问环境变量,系统属性的静态方法,还提供 加载文件和动态链接库的方法(主要对native方法有用,因为一些的底层的 功能需要使用c语言来实现),实现步骤如下

  1. native修饰的方法类似抽象方法,没有实现当使用带 -h 选项(Java 10 用-h 替换了 javah)的 javac 命令编译Java程序,会生成一个**.class文件和.h**文件
  2. 用c语言实现该native方法,要引入上一步生成的**.h**头文件
  3. 将第二步的文件编译成动态链接库文件
  4. 在Java中用System类的loadLibrary…()Runtime类的loadLibrary()加载第三步的动态链接库文件,就可以调用这个native方法
/*访问环境变量和系统属性*/
        Map<String, String> env = System.getenv();
        for(String name: env.keySet()){
            System.out.println(name + env.get(name));
        }
        //获取所有系统属性
        Properties properties = System.getProperties();
        //输出特定系统属性
        System.out.println(System.getProperty("os.name"));

System类还提供获取时间的方法

  • Long currentTimeMillis():返回当前时间到UTC 1970年1月1日午夜到现在的毫秒数
  • Long nanoTime(): 返回当前时间到UTC 1970年1月1日午夜到现在的纳秒数

System还提供 **identityHashCode(Object x)方法,返回指定对象精确的hashCode值,因为可能会重写hashCode()方法,所以该对象的hashCode()**不能唯一标识该对象,但使用 **identityHashCode(Object x)**方法比较,可以保证两个对象一定是同一个对象,即 任何两个对象的 identityHashCode值都不同

4. Runtime,ProcessHandle类的相关用法

Runtime 类代表Java程序当前运行环境,每个Java程序都有一个与之对应的 Runtime 实例,但程序不能创建该实例,但可以通过 **getRuntime()**方法获取与之关联的 Runtime对象

Runtime类也提供了 gc()方法和runFinalization()方法通知系统进行垃圾回收,清理系统资源,并提供了 **load(String filename)**和 **loadLibrary(String libname()**方法加载文件和动态链接库

Runtime能访问Jvm的相关信息

//获取java程序关联的运行时对象
Runtime runtime = Runtime.getRuntime();
System.out.println("处理器数量" + runtime.availableProcessors());
System.out.println("空闲内存数 " + runtime.freeMemory());
System.out.println("总内存数 " + runtime.totalMemory());
System.out.println("可用最大内存数 " + runtime.maxMemory());

Runtime还可以直接单独启用一个进程来运行操作系统的命令

//Process 代表进程
Process p = runtime.exec("notepad.exe");//单独启用笔记本线程

Java 使用 Process代表进程,Java 9 还新增一个 ProcessHandle接口,通过该接口可获取进程的各项信息,通过该接口的 onExit() 方法可以在进程结束时完成某些行为

ProcessHandle还提供一个 ProcessHandle.Info类,用于获取进程的命令,参数,启动时间,累计运行时间,用户等信息

Runtime runtime = Runtime.getRuntime();
Process p = runtime.exec("notepad.exe");
ProcessHandle ph = p.toHandle();
//ProcessHandle.Info 
ProcessHandle.Info info = ph.info();
System.out.println("进程相关参数"+ info.command() + info.arguments() + info.startInstant + info.totalCpuDuration);
//通过CompletableFuture在进程结束时运行某个任务
CompletableFuture<ProcessHandle> cf = ph.onExit();
cf.thenRunAsync(() -> {
     System.out.println("程序退出");
});

注意上述类的不同分工,主要就是通过 ProcessHandle对象来获取进程信息

5. Object与Objects类

Object类是所有类,数组,枚举类的父类,任何对象都可以赋值给 Object类型的变量,因为所有对象默认继承 Object类,所以它们都可以使用 Object类的方法

  • boolean equals(Object obj): 判断指定对象与该对象是否相等,只判断是不是一个对象,所以一般会重写此方法
  • protected void finalize(): 当系统中没有引用变量引用到该对象时,垃圾回收器调用此方法清理该对象的资源
  • Class<?> getClass(): 返回该对象的运行类型
  • **int hashCode(): **返回该对象的 hashCode值,默认情况是根据对象的地址计算的
  • String toString(): 返回该对象的字符串表示,默认格式为 运行时类名@十六进制hashCode值 ,当直接输出一个对象时,其实是调用该对象的 toString()方法

Java克隆方法,自我克隆:得到一个该对象的副本,二者之间完全隔离

但克隆机制只是对对象各实例变量进行复制值,但如果有引用变量,克隆之后依旧指向原来实例

实现 User类的克隆

class User implements Cloneable{	//需要先实现 Cloneable 接口,接口里没有定义方法
    public User clone() throws CloneNotSupportedException{	//实现克隆方法
        return (User) super.clone();
    }
}

Java 7 新增一个 Object工具类,它里面提供了一些工具方法来操作对象,这些方法大多是空指针安全的,不会引发空指针异常

public class Objects_ {
    //定义一个 obj 变量,默认null
    static Objects_ obj;
    public static void main(String[] args) {
        System.out.println(Objects.hashCode(obj));//输出 0
        System.out.println(Objects.toString(obj));//输出 null
        System.out.println(Objects.requireNonNull(obj));//要求 obj 不能为空,否则会引发异常,一般用于检验参数是否为空
    }
}

6. 使用String,StringBuffer,StringBuilder类

String类是不可变类,一旦创建一个 String 对象,这个对象的字符序列就不可变,直到这个对象销毁

StringBuffer 对象代表一个字符序列可变的字符串,通过其方法可以改变字符序列,再通过 toString方法转成String对象

StringBuilder是JDK 1.5新增的类,也代表字符序列可变的字符串,它与 StringBuffer基本相似,只是StringBuffer是线程安全的,所以 StringBuilder效率更高,应优先考虑

三个类都实现了 CharSequence接口,可认为是一个字符串的协议接口

java 9 改进了字符串的实现,java 9之前,字符串采用 **char[]**保存字符,所以每个字符占2字节,之后 字符串采用 **byte[]**外加一个 encoding-flag字段来保存字符,因此每个字符只占一个字节

通过 String构造器可以将 **byte[] 或 char[]**转成字符串

  • int compareTo(String anotherString): 比较两个字符串的大小,相等返回0,否则返回长度差
  • String concat(String str): 将两个字符串连接起来,和’+'相同
  • boolean contentEquals(StirngBuffer sb): 将 String对象和StringBuffer对象比较,字符序列相同时返回true
  • static String copyValueOf(char[] data): 将字符数组转为字符串,和构造器转换相同
  • **boolean endsWith(String suffix)😗*返回该String 对象是否以 suffix结尾,对应还有 startWith方法
  • boolean equalsIgnoreCase(String str): 和equals() 相同,只是忽略大小写
  • **byte[] getBytes()😗*将String对象转为 byte[]
  • **char[] toCharArray()😗*将String对象转为 char[]
  • **void getChars(int srcBegin,int srcEnd,char[] dst, int dstBegin)😗*将字符串从 srcBegin开始到srcEnd结束的字符复制到 dst字符数组中,dstBegin为起始复制位置
  • **String replace(char oldChar ,char newChar)😗*将字符串里的第一个 oldChar替换成 newChar
  • **String toLowerCase()😗*将字符串转换为小写
  • **String toUpperCase()😗*将字符串转换为大写
  • **static String valueOf(X x)😗*将一切基本类型值转为String对象

StringBuilder使用

public class StringBuilder_ {
    public static void main(String[] args) {
        StringBuilder sb = new StringBuilder();
        sb.append("asd");//追加
        sb.insert(0, "sda");//插入
        sb.replace(5,6,",");//替换
        sb.delete(5,6);//删除
        sb.reverse();//翻转
        sb.setLength(10);//改变长度,保留
    }
}

StringBuffer和StringBuilder有两个属性,length,capacity,容量通常比length大

7. 使用Math类进行数字计算

Math类是一个工具类,构造器私有,所以无法创建对象,都是类方法,还有两个类变量:PI E

可以查看对应 API熟悉这些类方法

8. 使用BigDecimal保存精确浮点数

float和double容易造成精度丢失,BigDecimal提供大量构造器创建对象,其中 **BigDecimal(double val)**构造器不建议使用,因为有一定的不可预知性,实际值并不会是val,而是一个近似值,应该使用 **BigDecinmal(String val)**构造器,这样结果可预知

如果必须使用double作为构造器参数时,应该通过 **BigDecimal.valueOf(double value)**静态方法创建对象

BigDecimal提供 **add(), subtract(), multiply(), divide(), pow()**等方法对浮点数进行常规运算

9. 使用Random类生成各种伪随机数

Random类用于生成一个伪随机数,一个构造器使用默认时间作为种子,另一个传入一个long型整数的种子

ThreadLocalRandom类是 java 7 新增的一个类,是 Random的加强版,在并发访问下,可以减少多线程资源竞争,保证系统具有良好的线程安全性,提供了一个静态的 **current()**方法获取其对象,之后调用 **nextXxx()**方法就可以获得伪随机数

public class Random_ {
    public static void main(String[] args) {
        Random random = new Random();
        System.out.println(random.nextBoolean());
        byte[] bytes = new byte[16];
        random.nextBytes(bytes);
        System.out.println(Arrays.toString(bytes));
        //种子相同时,生成的随机数也相同,所以是伪随机
        Random r1 = new Random(50);
        System.out.println(r1.nextInt());//-1160871061
        Random r2 = new Random(50);
        System.out.println(r2.nextInt());//-1160871061
    }
}

BigDecimal提供 **add(), subtract(), multiply(), divide(), pow()**等方法对浮点数进行常规运算

10. Date,Calendear 的用法及之间的联系

Date类用于处理时间,历史悠久,所以有很多过时的方法和构造器
剩下两个构造器

  • Date(): 生成一个代表当前日期时间的 Date对象,底层调用 **System.currentTimeMills()**获得 long整数作为日期参数
  • Date(long date): 参数创建的对象和 GMT 1970年1月1日 00:00:00之间的时间差,以毫秒为计时单位

现在一般使用 Calendar工具类
Calendar是一个抽象类,用于表示日历,java 本身提供了一个 GregorianCalendar类代表公历

创建Calendar对象

Calendar cal = Calendar.getInstance();

同时 Calendar对象和 Date对象可以相互转化

主要注意月份是以0为起始点,同时注意 add()和 roll()

Calendar call = Calendar.getInstance();
call.set(2003, 7 ,23, 0, 0 ,0);//2003-8-23
//call.add(MONTH, 6);//2004-2-23
//call.roll(MONTH, 6);//2003-2-23	不会进位

Calendar默认支持较好的容错性,可以传入超范围的参数,会取余,也可以关闭容错性,此时传入超范围参数时会报错

11. Java 8 新增的日期,时间的API的功能和用法

java 新增了 java.time包,包含很多常用类,可以查阅对应API熟悉使用方法

12. 创建正则表达式

正则表达式可以对字符串进行查找,提取,分割,替换等操作

**String **类的几个特殊方法

  • **booelan matches(String regex)😗*用于判断该字符串是否匹配指定的正则表达式
  • String replaceAll(String regex, String replacement): 将该字符串所有匹配正则表达式的子串替换为 replacement
  • String replaceFirst(String regex, String replacement): 将该字符串第一个匹配正则表达式的子串替换为 replacement
  • String[] split(String regex): 以 regex 为分隔符,分割字符串

特殊字符

$:匹配一行开头

^:匹配一行结尾

[]:匹配其中任意一个字符

{}:标记前面子表达式出现的频度,{n , m}表示前面出现次数 >= n 同时 <= m

*:前面子表达式可以出现任意次

+:前面子表达式出现 >= 1次

?:前面子表达式出现 0/1 次

. :匹配除 \n外任何单字符

\:转义

“\u0041\\\\" //匹配 A\

两个反斜杠相当于一个

通配符

\d:0-9数字

\D:除\d

\s:所有空白字符

\S:除\s

\w:匹配所有单词字符,数字,英文字母,_

\W:除\w

-:表示范围

^:表示求否

正则表达式的数量标识符有三种模式,默认贪婪模式(一直匹配)勉强模式(以?结尾,匹配最少字符) 占有模式(以 + 结尾)

13. 通过Pattern和Matcher使用正则表达式

Pattern对象是正则表达式编译后在内存中的表示形式,因此,正则表达式字符串应该先被编译成Pattern对象,再利用 Pattern对象创建对应的 Matcher对象,多个 Matcher对象共享一个 Pattern对象

//将一个字符串编译成pattern对象
Pattern p = Pattern.compile("a*b");
//使用Pattern对象创建Matcher对象
Matcher m = p.matcher("aaaaab");
boolean b = m.matches();//true

如果只使用一次 Pattern对象,可以

boolean b = Pattern.matches("a*b", "aaaaab");//true

**Pattern **是不可变类,可以供多个并发线程安全使用
Matcher类常用方法

  • **find()😗*返回目标字符串是否包含于 Pattern 匹配的子串
  • **matches()😗*返回整个目标字符串与 Pattern 是否匹配
  • group(): 返回上一次与 Pattern匹配的子串
  • **reset()😗*将现有的matcher对象应用于一个新的字符序列
public class FindGroup_ {
    public static void main(String[] args) {
        String str = "电话1:13500006666,电话2:18292900758,电话3:13611125565,电话4:15899903312";
        //创建一个Pattern对象,建立一个 Matcher对象
        //抓取 13X和15X的电话号
        Matcher matcher = Pattern.compile("((13\\d)|(15\\d))\\d{8}").matcher(str);
        //输出所有符合正则表达式的子串
        while(matcher.find()){
            System.out.println(matcher.group());
        }
    }
}

String类也提供 **matches()**方法,用于返回该字符串是否匹配指定的正则表达式

"kongyeeku@163.com".matches("\\w{3,20}@\\w+\\.(com|org|cn|net|gov)");//返回true

14. 程序国际化的思路

获取java所支持的国家和语言

        Locale[] availableLocales = Locale.getAvailableLocales();
        for(int i = 0;i < availableLocales.length;i++){
            System.out.println(availableLocales[i].getDisplayCountry() + "=" + 											availableLocales[i].getCountry() + " "
                    			+ availableLocales[i].getDisplayLanguage() + "=" + 										availableLocales[i].getLanguage());
        }

15. 程序国际化

Java 9 开始,Java支持使用 UTF-8字符集保存属性文件,注意属性文件必须显式保存为 UTF-8字符集

ResourceBundle类有一个静态方法 getBundle(String baseName, Locale locale):,该方法将根据 Locale加载资源文件, Locale 封装了一个国家,语言,可以通过 **Locale.getDefault(Locale.Category.FORMAT)**获取系统默认的国家/语言环境,而文件的 baseName都相同,加载时环境取决于 Locale,加载了该文件之后,该资源文件对应的内容就是多个 key-value,程序根据 key获取 value信息,就可以达到转换语言的目的

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值