Day 07 debug调试 junit单元测试 jdk5.0新特性 泛型,枚举,静态导入,拆装箱,增强for,可变参数 反射

时间:2022-10-18 15:22:53
day07
1.myeclipse的安装和使用
*eclipse:是一个免费的开发工具
*myeclipse:是一个收费的插件,破解myeclipse
**安装目录的要求,不能有中文和空格
**安装完成之后,选择一个工作空间,这个工作空间不能有中文和空格
**破解myeclipse
**运行run.bat文件,但是运行之前,必须安装jdk,通过配置环境变量


*myeclipse的使用
*创建一个工程
-类型java project  web project
-选择依赖的jdk,可以使用myeclipse自带的jdk,或者使用安装的jdk


*创建包 package
-cn.itcast.test  xx.xx.xx


*在包里面创建类
-命名规范:
**首字母大写
比如:TestDemo1 UserManager
*在类里面创建方法
public void test1(参数列表){
方法的返回值;
}
-方法的命名规范 
首字母小写  比如:addNum();


*定义变量
变量的命名规范
**首字母小写,第二个单词的首字母大写,比如userName
*这些命名还有一种方式
**使用汉语拼音命名  yonghuming
**不能把汉语拼音和英文字母混合使用
userMing
*命名的最基本的原则:看到名字知道是什么含义

*代码需要有缩进


*运行程序 run as java application
debug as java application


2.debug的调试模式(断电调试模式)
*使用这种模式,调试程序(看到程序里面数据的变化)


*使用debug第一步需要设置断点(让程序运行停止在这一行)
-显示出来行号
-双击左边,出现一个圆点,表示设置了一个断点
*使用debug方式,运行程序
-提示是否进入到调试界面,yes
-在断点那一个,有一个绿色条,表示程序停止在这一行,没有向下运行
*可以让程序向下执行
-使用step over 快捷键是F6(单步执行)
-resume F8 表示调试结束,执行直接向下运行
**比如当前的断点,之后还有断点,跳到下一个断点
**如果当前断点后面没有断点,程序直接运行结束


*debug另外一个用途
**查看程序的源代码
**F7 step into 进入到方法
**step return :返回


3.myeclipse的快捷键的使用
*生成main方法  直接写mian 然后按ALT+/
*快速导入包  ctrl+shift+O   
*Ctrl+/   (取消)单行注释    Ctrl+Shift+/  多行注释   取消多行注释 Ctrl+Shift+\
*快速生成get set方法  选中属性,右键 source  generate getters and setters
*Ctrl+i  格式化                         trl+shift+f  有時候不好使,选中,点source然后点format


*Ctrl+1 代码修复   alt+/也可以
*Ctrl+f   查找


4.junit的使用
*单元测试


*测试的对象是一个类中的方法


*junit不是javase的一部分,想要使用导入jar包
**但是,在myeclipse中自带了junit的jar包


*首先junit版本  3.x 4.x
* 单元测试方法的时候,方法命名规则  public void 方法名(){}

*使用注解的方法运行测试,在方法的上面
**@Test:表示方法进行单元测试


@Test
public void testAdd1(){
TestJunit test01 = new TestJunit();
test01.testAdd(1, 2);
}
-选中方法名称,右键运行 点击run as --- junit test
-当出现绿色条,表示方法测试通过
-当出现了红棕色条的时候表示测试方法不通过


---要运行类中的多个测试方法,点击类中的其他位置  run as -----jnuit test


** @Ignore:表示这个方法不进行单元测试


** @Before:在每个方法之前运行
** @After:在每个方法之后运行


**断言(了解)


jdk5.0新特性
jdk 1.1   1.2   1.4  5.0
**泛型,枚举,静态导入,自动拆装箱,增强for循环,可变参数
**反射

5.泛型的简介
*为什么要使用泛型?
-一般使用在集合上
**比如现在把字符串类型的值放入到集合中,这个时候,这个值放入到集合中之后,失去本身的类型,只能是object类型,这个时候,比如想对这个值进行类型转换,很容易出现类型转换的错误,怎么解决这个问题,可以使用泛型来解决


*在集合上如何使用泛型
-常用集合 list set map
-泛型语法 集合<String>比如 List<String>


*在泛型里面写的是一个对象,String,不能写基本的数据类型,比如int
**写基本的数据类型队形的包装类
int --- Integer
byte --- Byte
short --- Short
long --- Long


float ----Float
double ----Double
char --- Character
boolean --- Boolean


*在list上使用泛型
list的三种实现  ArrayList linkList Vector
public void testList(){
List<String> list = new ArrayList<String>();
list.add("aaa");
list.add("bbb");
list.add("bbb");

//遍历list集合有几种方式
//普通for循环  迭代器  增强for

//普通for循环
for(int i=0;i<list.size();i++){
String s = list.get(i);
System.out.println(s);

}
System.out.println("==========================");
//使用增强for循环
for (String s1 : list) {
System.out.println(s1);
}
System.out.println("==========================");
//使用迭代器
Iterator<String> it = list.iterator();
while(it.hasNext()){
System.out.println(it.next());

}
}


*ArrayList linkList Vector 的区别


*在set上使用泛型
//泛型使用在set集合上
@Test
public void testSet(){
Set<String> set = new HashSet<String>();
set.add("www");
set.add("qqq");
set.add("zzz");
set.add("qqq");
//遍历set  有几种  两种
//迭代器  增强for
//使用增强for遍历
for (String s2 : set) {
System.out.println(s2);
}
System.out.println("============================");

//使用迭代器遍历
Iterator<String> it1 = set.iterator();
while(it1.hasNext()){
System.out.println(it1.next());
}

}


*在map上使用泛型
-map结构 key-value形式


6.泛型使用在方法上
*定义一个数组                            ,<font size="" color=""></font>
实现指定位置上数组元素的交换
*方法逻辑相同,只是数据类型不同,这个时候使用泛型方法
/*
* 使用泛型方法需要定义一个类型 使用大写字母来表示T :这个T表示
* 任意的类型,写在返回值void 之前  <T>
* =============表示定义了一个类型这个类型是T
* 在下面就可以使用这个类型了
*/
public static <T> void swap1(T[] arr,int a,int b){
T temp = arr[a];
arr[a] = arr[b];
arr[b] = temp;
}

**作业2:实现一个泛型方法,接受任意一个数组,颠倒数组中所有元素


7.泛型在类上的使用(了解)
*在一个类上面定义一个类型,这个类型可以在类面直接使用
//在类里面可以直接使用T的类型
T aa;
public void test11(T bb){}

//写一个静态方法在类上面定义的泛型,不能在静态方法里面使用
public static <A> void test12(A cc){}


8.枚举的简介
*什么是枚举?
**需要在一定的范围内取值,这个值只能是这个范围中的任意一个
**现实场景:交通信号灯,有三种颜色,但是每次只能亮三种颜色中的任意一个


*使用一个关键字 enum
**enum Color3{
RED,GREEN,YELLOW;
}
*枚举的构造方法是私有的


*特殊枚举的操作(了解)
**在枚举里面有构造方法
**构造方法里面有参数,需要在每个实例上面都写参数
**在枚举的类里面有抽象的方法
**在枚举的每个实例里面都重写这个抽象方法


9.枚举的api操作
**name() :返回枚举的名称
**ordinal:获取枚举的下标  下标从零开始
**valueOf(Class<T> enumType, String name)  :得到枚举的对象


**还有两个方法,但是这两个方法不在api里面,编译的时候生成两个方法
***valueof(String name)    转换枚举对象
***values():获取所有枚举对象数组


*练习:枚举对象,枚举下标,枚举对象名称表示之间的转换

public class TestEnum3 {

//知道枚举的下标,得到枚举对象和名称
@Test
public void test3(){
int idx2 = 2;
//得到枚举对象数组
Color100[] cs = Color100.values();
//根据下标来得到对象
Color100 c12 = cs[idx2];
//得到枚举的名称
String name = c12.name();
System.out.println(name);
}

//知道枚举的名称,怎么得到枚举的对象的下标
@Test
public void test2(){
String name1 = "GREEN";
//得到对象
Color100 c1 = Color100.valueOf(name1);
//枚举下标
int idx1 = c1.ordinal();
System.out.println(idx1);
}

//知道枚举的对象,得到枚举名称和下标
@Test
public void test1(){
//得到枚举对象
Color100 c100 = Color100.RED;
//枚举的名称
String name = c100.name();
//枚举的下标
int idx = c100.ordinal();
System.out.println(name+" "+idx);

}

}


enum Color100{
RED,GREEN,YELLOW;

}


10.静态导入(了解)
*可以在代码里面直接使用静态导入的方式,导入静态方法或者常量
*import static xx.xx.xxx
*******
import static java.lang.System.out;
import static java.util.Arrays.sort;
import java.util.Arrays;
public class TestDemo1 {
public static void main(String[] args) {
out.println("hello");

int[] arr1 = {10,3,5,20,15};
sort(arr1);
System.out.println(Arrays.toString(arr1));
}
}


**比如实现计算器 在Math类里面


11.自动拆装箱
* 装箱
**把基本的数据类型转换成包装类
* 拆箱
**把包装类转成基本的数据类型
** //自动装箱
Integer i = 10;
//自动拆箱
int m = i;


**在jdk1.4里面如何实现装箱和拆箱
//装箱
Integer m = new Integer(10);
//拆箱
int a = m.intValue();


**jdk是会向下兼容的
-比如jdk1.4里面写的代码,这个时候到5.0里面也可以运行


**练习:向下兼容
==执行的结果是会调用 doSomething(double m)
==首先在jdk1.4里面肯定是调用这个方法,如果调用下面的方法需要类型转换,但是jdk1.4不能实现自动拆装箱
==由于jdk是向下兼容的,所以jdk1.4调用这个方法,在jdk5.0还是会调用这个方法
public static void main(String[] args) {
doSomething(10);
}
public static void doSomething(double m){
System.out.println("double。。。");
}
public static void doSomething(Integer a){
System.out.println("integer。。。");
}
**记住:基本的数据类型对应的包装类
*int----Integer
*char---Character


12.增强for循环
*语法for(遍历出来的值:要遍历的集合){}
List<String> list = new ArrayList<String>();
list.add("aaa");
list.add("bbb");
//使用增强for循环
for(String s:list){
System.out.println(s);
}
*使用场景:数组;    实现Iterable接口的集合可以使用for循环


*在集合上可以使用增强for循环遍历
list set 实现了Iterator接口,所以可以使用增强for循环
map不能使用增强for循环,没有实现Iterable接口,所以不能使用增强for循环


*增强for循环出现的目的:为了替代迭代器
**增强for的底层就是迭代器


13.内容补充
(1)泛型擦除
*首先泛型只是出现在源代码阶段,当编译之后泛型就不存在了


(2)练习:实现一个泛型方法,接受任意一个数组,颠倒数组中所有元素
代码 public static <T> void reverse(T[] arr1) {
/*
* 基本思想:把第一个元素和最后一个元素交换位置,把第二个元素和倒数第二个元素交换位置
* 交换  长度/2次
*/
//遍历数组
for(int i = 0;i<arr1.length/2;i++){
/*int temp = arr1[0];
arr1[0] = arr1[arr1.length-1];*/
T temp = arr1[i];
arr1[i] = arr1[arr1.length-i-1];
arr1[arr1.length-i-1] = temp;
}
}


14.可变参数
*可变参数可以                               <font size="" color=""></font>
应用在什么场景下
**实现两个数相加,实现三个数的相加,实现四个数的相加
--如果实现的多个方法,这些方法里面逻辑基本相同,唯一不同的是传递参数的个数,可以使用可变参数
*可变参数的定义方法  数据类型...
--代码
public static void add1(int...nums){
//nums理解为一个数组,这个数组存储传递过来的参数
//System.out.println(nums.length);
int sum = 0;
//遍历数组
for(int i=0;i<nums.length;i++){
sum+=nums[i];
}
System.out.println(sum);
}

*注意的地方
(1)可变参数需要写在方法的参数列表中
不能单独定义
(2)在方法的参数列表中只能有一个可变参数
(3)方法的参数列表中的可变参数必须放在参数的最后
-add1(int a,int...nums)


15,反射的原理(********理解**********)
*应用在通用性比较高得代码中
*后面学到得框架,大多数都是使用反射来实现的


*在框架开发中,都是基于配置文件开发
**在配置文件中配置了类,可以通过反射得到类中的所有内容,可以让类中的某个方法来执行


*类中的所有内容:属性,没有参数的构造方法,有参数的构造方法,普通方法


*画图分析反射的原理
*首先需要把java文件保存到本地硬盘  .java
*编译java文件,生成 .class文件
*使用JVM,把class文件通过类加载器加载到内存中
*万事万物都是对象,class文件在内存中使用Class类表示


*当使用反射的时候,首先需要获取到class类,得到了这个类之后,就可以得到class文件中的所有内容
-包含属性   构造方法  普通方法
*属性通过一个类  Field
*构造方法通过一个类  Constructor
*普通方法通过一个类  Method


16.使用反射操作类里面的无参数的构造方法
*首先获取到Class类
-// 获取Class类
Class classzz1 = Person.class;
Class classzz2 = new Person().getClass();
Class classzz3 = Class.forName("cn.itcast.test09.Person");


*比如:要对一个类进行实例化,可以new,不使用new ,怎么获取?
//得到Class
Class c3 = Class.forName("cn.itcast.test09.Person");
//得到Person类的实例
Person p = (Person) c3.newInstance();
*代码
//操作无参数的构造方法
@Test
public void test1() throws Exception{
//得到Class
Class c3 = Class.forName("cn.itcast.test09.Person");
//得到Person类的实例
Person p = (Person) c3.newInstance();
//设置值
p.setName("zhangsan");
System.out.println(p.getName());
}


17.使用反射操作有参数的构造方法
//操作有参数的构造方法
@Test
public void test2() throws Exception{
//得到Class
Class c1 = Class.forName("cn.itcast.test09.Person");
//使用有参数的构造方法
//c1.getConstructors();  //获取所有的构造方法
//传递的是有参数的构造方法里面参数类型,类型使用class形式传递
Constructor cs = c1.getConstructor(String.class,String.class);
//通过有参数的构造方法设置值
//通过有参数的构造方法创建了Person实例
Person p1 = (Person) cs.newInstance("lisi","100");
System.out.println(p1.getId()+" "+p1.getName());
}


18.使用反射操作里面的属性
//操作name属性
@Test
public void test3(){
//得到Class类
try {
Class c2 = Class.forName("cn.itcast.test09.Person");
//得到name属性
//c2.getDeclaredFields();//得到所有的属性
//通过这个方法得到属性,参数是属性的名称
Field f1 = c2.getDeclaredField("name");
//设置可以操纵私有属性
//操作的是私有的属性,不让座操,需要设置可以操作私有属性setAccessible(true),可以操作私有属性
f1.setAccessible(true);
//设置name值
Person p11 = (Person) c2.newInstance();
//设置name值  set方法   两个参数; 第一个参数是类的实例,第二个参数是设置的值
f1.set(p11, "wangwu");   //相当于p.name = "wangwu"
System.out.println(f1.get(p11));   //相当于p.name
} catch (Exception e) {
e.printStackTrace();
}


}


19.使用反射操作普通方法
*使用  Method类表示普通方法
*代码
//操作普通方法  操作setName
@Test
public void test4() throws Exception{
//得到Class类
Class c5 = Class.forName("cn.itcast.test09.Person");
//得到person的实例
Person p5 = (Person)c5.newInstance();
//得到普通的方法
//c5.getDeclaredMethods();   //得到所有的方法
//传递两个参数,第一个参数是方法名称,第二个参数是通过放啊发设置的值
Method m1 = c5.getDeclaredMethod("setName", String.class);
//让setName方法执行,执行设置值
//使用invoke(p5, "niu7");传递两个参数:第一个参数:person实例,第二个参数是设置的值
//执行了invoke方法之后,先当于执行了setName方法,同时通过这个方法设置了一个值niu7
m1.invoke(p5, "niu7");
System.out.println(p5.getName());
}
*//操作的是私有的方法,需要设置值是true
*//m1.setAccessible(true);


*当操作的方法是静态的方法的时候,因为静态方法调用方式是 类名.方法名 ,不需要实例
*使用反射操作静态方法的时候,也是不需要实例
*在invoke方法的第一个参数里面,写一个 null
-m1.invoke(null,"niu7");