1 理解生产者与消费者
生产者生产产品,消费者消费产品。当消费者发现没有产品可以消费时,会唤醒生产者,然后自己等待;当生产者生产产品后,会唤醒消费者,然后自己等待。
例如:厨师与顾客,当顾客要吃包子时,发现没有包子,所以唤醒厨师,然后自己开始等待。当厨师做好包子后,会唤醒顾客,然后自己等待。
顾客查看包子柜上的包子数,如果大于0,那就吃包子,吃完了告诉厨师没包子了;如果等于0,那就等待;
厨师查看包子柜上的包子数,如果等于0,那就做包子,做好了告诉顾客包子好了;如果大于0,那就休息去;
2 生产者与消费者例子
包子柜:
属性:int count,包子数;
方法:int getCount(),查看包子数;
方法:void chi(),吃包子。把count—即可;
方法:void zuo(),做包子。把count++即可。
厨师(线程类):
属性:包子柜
方法:run()
死循环:while(true)。只要没包子就做,不怕累死;
查看包子数,如果包子数大于0,说明有包子,那么就wait(),一直到包子数小于等于0才不wait()。这里需要使用while(包子数 > 0) {wait()}来处理;
做包子;
唤醒顾客,已有包子了。
顾客(线程类)
属性:包子柜;
方法:run()
死循环:while(true)。只要有包子就吃,不怕撑死;
查看包子数,如果包子数小于等于0,说明没包子,那么就wait(),一直到包子数大于0才不wait()。这里需要使用while(包子数 <= 0){wait()}来处理;
吃包子;
唤醒厨师,包子没了。
3 多个生产者和多个消费者的问题
创建两个顾客,以及两个厨师。
可能死锁:可能会出现厨师唤醒的不是顾客,而是厨师,最后造成四个线程都休眠,就死锁了
Scanner类
使用Scanner类可以用来获取输入。
通常输入源是控制台,也就是说用它可以获取键盘输入!
常用方法介绍
构造器:Scanner in = new Scanner(System.in);
String next():获取一个单词;
String nextLine():获取一行输入;
int nextInt():获取一个整数。
2 先nextInt()再nextLine()的问题
每次输入都需要给出回车,表示输入结束。
但回车也一个字符串。
先nextInt(),获取一个整数,再nextLine()获取一行字符串。
因为nextInt()时,输入结束时会输入一个回车符,这个回车符会被nextLine()获取,所以会出问题。
public static void fun1() {
Scanner in = new Scanner(System.in);
int i = in.nextInt();
Scanner in1 = new Scanner(System.in);
String s = in1.nextLine();
System.out.println("i = " + i);
System.out.println("s = " + s);
}
处理的方法:让两个输入使用不同的Scanner对象!
Random
获取生成伪随机数;获取的数非常精确,导致不能确定什么时候,你能给出结果,所以就像随机数一样。
Random r = new Random();
int n = r.nextInt(100);//生成的一个随机数,范围是0~100,但包含0,却不包含100。有头没尾。
int n = new Random().nextInt(100);生成的一个0~100随机数,但包含0,却不包含100。
String类构造器
描述一个字符串。
编码:用记事本打开a.txt,显示内容
记事本把a.txt文件中的1001样式的二进制数据读取出来,然后通过编码表转换成字符,显示在记事本的窗口中。
当使用记事本向a.txt文件写入字符数据时,记事本会把输入输入的字符数据,通过编码表转换成二进制数据,然后再保存到a.txt中。
编码表是一个两列的表格,第一列是字符本身,第二列这个字符对应的值(二进制数值)
天下编码表太多了!
ASCII – 只有英文字母,只能用上7位,用不上8位。
ISO-8859-1 – 包含了一些欧州人用的一些字符。8位了!
GB2312 – 都是汉字,也包含了ASCII
GBK – 比GB2312多点
GBK18030 – 比GBK 更全了!
Unicode – 包含天下所有的字符。
UTF-8 – 它是Unicode的一种,这最为常用!
“崔”字个汉字,在GBK和UTF-8两个表中,对应的编码是不同的!
String()
String s = new String();//0长,不同于String s = null;
2 String(String)
使用字符串来构造字符串
3 String(byte[])
把byte[]转换成字符串。
4 String(byte[], int, int)
同上,上面是使用整个byte[]来创建字符串,而这个是用byte[]的一部分来创建字符串。
int offset:从哪一个下标位置开始;
int length:一共需要几个字节。
例如:byte[] b = {1,2,3,4,5}; new String(b, 1, 3);表示使用b数组下标1开始,一共要3个字节,即2,3,4。
5 String(char[])
把字符数组转换成字符串
6 String(char[], int, int)
不解释了!
String实体不可变
String s1 = “a”;
s1 = “b”;
1 实体不可变
实体不可变,所有方法(例如替换)只是返回替换后的新串,而原串没有改变!
2 String常量池
所有字符串常量都在常量池中,常量池在方法区中。
当不使用构造器来创建字符串时,
查看池中是否存在这个字符串
如果存在,那么直接返回这个字符串常量的引用;
如果不存在,在池中创建这现代战争字符串常量,再返回其引用。
String s1 = “hello”;
String s2 = “hello”;
System.out.println(s1 == s2);
String方法:判断方法
判断相关方法返回值都是boolean
1 boolean equals(Object)
2 boolean equalsIgnoreCase(String)
3 boolean contains(String)
4 boolean startsWith(String)
5 boolean endsWith(String)
6 boolean isEmpty()
控制台登录程序练习
固定一个用户:qdmmy6、123
让用户输入用户名
让用户输入密码
比较用户名与qdmmy6是否相等,并且密码是否与123相等
如果比较成功,那么输出欢迎XX登录成功
对不起!您的用户名或密码错误!
String方法:获取方法
1 int length()
返回当前字符串的字符个数
2 char charAt(int)
返回指定下标位置上的字符。”abc”.charAt(0);
3 int indexOf(int)
你给我一个字符,我给你下标位置。
“abcabc”.indexOf(‘b’);
4 int indexOf(String)
你给我一个子串,我返回子串在当前字符串中的位置。
5 int index(int ch, int fromIndex)
返回指定字符在当前字符串中的下标位置,从指定位置开始查找。
“abcabc”.indexOf(‘b’, 2);
6 int index(String s, int fromIndex)
自己想去
7 String substring(int)
截取子串,从指定位置到当前字符串末尾。
“abcdefg”.substring(3);
8 String substring(int, int)
截取子串,从指定开始位置到指定结束位置。有头没尾!
“abcdefg”.substring(2, 5);//cde
练习:统计字符串中大写小写数字分别出现的个数
String方法:转换方法
1 byte[] getBytes()
把字符串转换成byte[]返回
2 char[] toCharArray()
把字符串转换成字符数组
3 static String copyValueOf(char[] chs)
把字符数组转换成字符串
4 static String valueOf(char[] chs)
把字符数组转换成字符串
5 static String valueOf(int i)//所有基本类型
static String valueOf(byte b)
把基本类型转换成字符串
int n = 10;
String s = String.valueOf(n);
String s = “” + n;
6 String toUpperCase()
把当前字符串转换成都是大写
7 String toLowerCase()
把当前字符串转换成都是小写
8 String concat(String str)
连接字符串。
等同与“+”
String s = “ab” + “cd”;
String s = “ab”.concat(“cd”);
练习
1 获取文件的扩展名
2 忽略大小写比较前缀和后缀
3 判断文件名是否是指定扩展名
String方法:其他功能
1 替换功能
String replace(char old, char new)
String replace(String old, String new)
2 String[] split(String)
这个方法只学一半!
使用参数作为分隔符,把当前字符串分隔成字符串数组。
其实参数是一个正则表达式regex
3 String trim()
去掉当前字符串的前导和后导空格,但中间空格不会去掉
String方法:比较方法
1 int compareTo(String)
这个方法是比较参数字符串和当前字符串谁大谁小:
当前>参数,返回正数;
当前<参数,返回负数
当前=参数,返回0
字典顺序!
Zhang
Zhan
Zhan在前Zhang在后
aaa
z aaa在前,z在后
zhao zhan zhan在前 zhao在后
zhan Zhao Zhao在前 zhan在后 Z小 ASCII
按字典顺序排列,几个字符串从前往后依次比较知道比较出大小,在后的,字符串大,如果有大小写的,小写的要在后。
2 int compareToIgnoreCase(String)
忽略大小写比较,Java和JAVA一样大。
3 给字符串数组排序
4 查找指定字符串在大串中出现的次数!
String s = “fkdhellolsjahellof;ljhellodshello;kajfld;sjal;kfjdhellol;sajf”;
String s1 = “hello”;
eclipse的debug
1 设置断点
双击即可!
2 debug启动方法
选择debug as
3 debug试图介绍
4 F5(进入方法)
5 F6(跳过方法)
6 F7(结束方法)
7 去除断点
8 重看当前方法(drop to frame)
9 查看表示式的值(Watch)
选中表达式,右键选择Watch