多线程
一、判断题(T为正确,F为错误),每题1分
1.如果线程死亡,它便不能运行。(T)
2.在Java中,高优先级的可运行线程会抢占低优先级线程。( T)
3.线程可以用yield方法使低优先级的线程运行。(F)
4…程序开发者必须创建一个线程去管理内存的分配。(T)
5.一个线程在调用它的start方法,之前,该线程将一直处于出生期。( )
6.当调用一个正在进行线程的stop( )方法时,该线程便会进入休眠状态。(F)
7.一个线程可以调用yield方法使其他线程有机会运行。(T)
8. 多线程没有安全问题(F)
9. 多线程安全问题的解决方案可以使用Lock提供的具体的锁对象操作(T)
10. Stop()方法是终止当前线程的一种状态()

二、选择题(不定项选择题),每题2分
1.Java语言中提供了一个▁线程,自动回收动态分配的内存。ACD
A.异步
B.消费者
C.守护
D.垃圾收集
2.Java语言避免了大多数的▁错误。ACD
A.数组下标越界
B.算术溢出
C.内存泄露
D.非法的方法参数
3.有三种原因可以导致线程不能运行,它们是▁。BCD
A.等待
B.阻塞
C.休眠
D.挂起及由于I/O操作而阻塞
4.当▁方法终止时,能使线程进入死亡状态。A
A.run
B.setPrority//更改线程优先级
C.yield//暂停当前线程的执行 执行其他线程
D.sleep//线程休眠
5.用▁方法可以改变线程的优先级。B
A.run
B.setPrority
C.yield
D.sleep
6.线程通过▁▁方法可以使具有相同优先级线程获得处理器。C
A.run
B.setPrority
C.yield
D.sleep
7.线程通过▁▁方法可以休眠一段时间,然后恢复运行。D
A.run
B.setPrority
C.yield
D.sleep
8.方法resume( )负责重新开始▁▁线程的执行。D
A.被stop( )方法停止
B.被sleep( )方法停止
C.被wait( )方法停止
D.被suspend( )方法停止
9.▁▁方法可以用来暂时停止当前线程的运行。BCD
A.stop( )
B.sleep( )
C.wait( )
D.suspend( )
10. 请问下列哪些类是定义在java.io包中的抽象类()ABEF
A. InputStream
B. OutputStream
C. PrintStream
D. Reader
E. FileInputStream
F. FileWriter

三、简述题,每题5分
1.简述程序、进程和线程之间的关系?什么是多线程程序?
程序:具有能够正常运行的特性的代码
进程:运行在内存中的程序,活跃的程序
线程:线程是进程的独立运行单元

3.什么是线程调度?Java的线程调度采用什么策略?
线程调度是系统调节CPU,提高CPU的使用率,且能实现多个任务同时运行的一种机制。
Java线程调节是抢占式线程调节,即每个任务的每个线程都可以抢占处理器的使用权

4.如何在Java程序中实现多线程?
一、 对象类继承Thread类,重写run方法。测试类新建继承Thread的对象类。
二、 对象类实现Runnable接口,重写run方法。测试类新建对象类,把对象类作为新建Thread类的target。
三、 对象类实现Callable接口,重写call方法。测试类新建对象类,吧对象类作为fatureTask类的target。

5.试简述Thread类的子类或实现Runnable接口两种方法的异同?
两者都可以创建多线程。
Thread类的子类不能继承别的类,实现Runnable接口可以继承别的类;
Thread创建的类就是线程对象,实现Runnable接口创建的类是Thread类的target;
Thread类的子类的线程对象用this获取,实现Runnable接口的线程对象用Thread.currentThread获取。

6.说明缓冲流的优点和原理

8:在Java中wait()和sleep()方法的不同?
1、sleep()方法使当前线程暂时停止一段时间,但是不释放同步锁对象,一直持有同步锁对象,暂停时间过了以后继续运行。
wait()方法暂时停止运行,释放同步锁对象,等待用notify()方法唤醒。
2、sleep()方法是Thread类对象,wait()是Object类对象

9:Java中Runnable和Callable有什么不同?
.实现Runnable接口的方法,然后重写run方法,创建的类是Thread的target
实现Callable接口的方法,然后重写call方法,创建的类是FatureTask对象的target

四、程序设计题
1.编写一个应用程序,在线程同步的情况下来实现“生产者―消费者”问题。

star类:
public class Star {
    private String name;
    private int age;
    private boolean flag= false;//false为没有值

    public boolean getflag(){
        return flag;
    }
    public boolean setflag(boolean flag1){
        return this.flag=flag1;
    }
    public Star() {
        super();
    }
    public Star(String name, int age) {
        if(name!=null){
            this.name = name;
        }else{
            System.out.println("姓名格式有误,请输入姓名!");
            System.exit(0);
        }
        if(age>0){
            this.age = age;

        }else{
            System.out.println("年龄格式有误,输入正确年龄!");
            System.exit(0);
        }
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
    }

}

setstar类:
public class SetStar implements Runnable {

    private Star s;
    public  SetStar(Star s){
        this.s = s;
    }


    @Override
    public void run() {
        //加入循环,多线程出现数据错位,应使用同步代码块
        while(true){
            synchronized (s) {
                if(s.getflag()==true){
                    try {
                        s.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("set线程名:" + Thread.currentThread().getName() + "\t姓名:"
                        + s.getName() + "\t年龄:" + s.getAge());
                //如果生产者有数据,唤醒消费者线程拿数据
                s.setflag(true);
                s.notify();
            }
        }


    }

}

getstar类:
public class GetStar implements Runnable {

    private Star s;

    public GetStar(Star s) {
        this.s = s;
    }

    @Override
    public void run() {
        while (true) {
            synchronized (s) {
                if (s.getflag()==false) {
                    try {
                        s.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                StringBuffer sb = new StringBuffer();
                sb.append("get线程名:")
                .append(Thread.currentThread().getName())
                .append("\t姓名:").append(s.getName())
                .append("\t年龄:").append(s.getAge());
                System.out.println(sb);
                //消费者线程没有数据,唤醒生产者线程生产数据
                s.setflag(false);
                s.notify();
        }
        }

    }

}

测试类:
package Day21_1;

import java.util.Scanner;

/**
 *屏幕录入人物信息,然后多线程实现设置人物信息,获取人物信息
 * @author Aoman_Hao
 */
public class StarTest {

    public static void main(String[] args) {
        //屏幕录入
        Scanner sc_name = new Scanner(System.in);
        System.out.println("请输入人物姓名");
        String name = sc_name.next();

        Scanner sc_age = new Scanner(System.in);
        System.out.println("请输入人物年龄");
        int age = sc_age.nextInt();
        //创建资源对象
        Star s1 = new Star(name,age);
        SetStar ss1 = new SetStar(s1);

        GetStar gs1 = new GetStar(s1);
        //创建线程对象
        Thread sT1 = new Thread(ss1);
        Thread gT1 = new Thread(gs1);

        //启动线程对象
        sT1.start();
        gT1.start();

    }

}

输出(节选):
请输入人物姓名
刘诗诗
请输入人物年龄
23
set线程名:Thread-0 姓名:刘诗诗  年龄:23
get线程名:Thread-1 姓名:刘诗诗  年龄:23
set线程名:Thread-0 姓名:刘诗诗  年龄:23
get线程名:Thread-1 姓名:刘诗诗  年龄:23
set线程名:Thread-0 姓名:刘诗诗  年龄:23

2.修改上题,由于各个线程是异步运行的,因此无法预计其相对速度,为了使生产者能够不断地生产,可以使用循环缓冲区,保证有足够多的内存区保存更多的产品。(生产者——仓库——消费者)

3 :
1)将若干个Student对象;若干个Teacher对象,写出到d:/0404/a.txt中,
2)将该文件中所有的Student对象反序列化回来,装入List,所有的Teacher对象反序列化回来装入另一个List
1):

测试类
package Day21_3;

import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.util.HashMap;
import java.util.Set;

/**
 * @author Aoman_Hao
 */
public class Demo {

    public static void main(String[] args) throws IOException {
        // 创建文件d:/0404/a.txt
        File file1 = new File("d://0404");
        file1.mkdir();
        File file = new File("d://0404//a.txt");
        file.createNewFile();
        FileWriter FW = new FileWriter(file);

        // 创建学生老师资源对象,HashSet
        HashMap<String, Object> S = new HashMap<>();
        S.put("1", new Student("Aoman", 30));
        S.put("2", new Student("蓉蓉", 23));
        S.put("001", new Teacher("一老师", "数学"));
        S.put("002", new Teacher("二老师", "语文"));

        Set<String> S_Keys = S.keySet();
        StringBuffer sb = new StringBuffer();
        for(String key : S_Keys){
            sb.append("编号:").append(key).append(";信息:").append(S.get(key)).append("\r\n");
        }
        System.out.println(sb);

        FW.write(sb.toString());
        FW.flush();
        FW.close();
    }

}
student,teacher普通的对象类。
学生资源类:
public class Student {
    private String name;
    private int age;
    public Student(String name, int age) {
        super();
        this.name = name;
        this.age = age;
    }
    public Student() {
        super();
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
    }
    @Override
    public String toString() {
        return "Student [name=" + name + ", age=" + age + "]";
    }

}
教师资源类:
public class Teacher {
    private String name;
    private String subject;
    public Teacher() {
        super();
    }
    public Teacher(String name, String subject) {
        super();
        this.name = name;
        this.subject = subject;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
    }
    public String getSubject() {
        return subject;
    }
    public void setSubject(String subject) {
    }
    @Override
    public String toString() {
        return "Teacher [name=" + name + ", subject=" + subject + "]";
    }

}

4:实现字符串和字节数组之间的相互转换,比如:将字符串”西部开源技术中心xbkyjszx”转换为字节数组,并将字节数组再转换回字符串!

package Day21_2;
/**
 * @author Aoman_Hao
 */
public class Test {

    public static void main(String[] args) {

        String s = new String("西部开源技术中心xbkyjszx");
        //字符串--字节数组
        byte[] s_by = s.getBytes();
        System.out.println("字节数组:"+s_by);
        //字节数组--字符串
        String s2 = new String(s_by);
        System.out.println(s2);


        //字符串--字符数组
        char[] s_ch= s.toCharArray();
        System.out.println("字符数组:"+s_ch);

        //字符数组--字符串
        StringBuffer sb = new StringBuffer();
        for(char ch : s_ch){
            sb.append(ch);
        }
        System.out.println(sb);
    }

}

输出:
字节数组:[B@1afb7ac7
西部开源技术中心xbkyjszx
字符数组:[C@7439aca7
西部开源技术中心xbkyjszx

5:用Java编程一个会导致死锁的程序,你将怎么解决?请你设计
资源对象A类:

public class DeadClassA {
    public void FirstClassA(){
        try {
            Thread.sleep(100);
        } catch (Exception e) {
            System.out.println("进程名:"+Thread.currentThread().getName()+"调用A的First方法内部");
        }

    }
    public void LastClassA(){
        System.out.println("A的Last方法内部");
    }
}

资源对象B类:
public class DeadClassB {
    public void FirstClassB(){
        try {
            Thread.sleep(100);
        } catch (Exception e) {
            System.out.println("进程名:"+Thread.currentThread().getName()+"调用B的first方法");
        }
    }

    public void LastClassB(){
        System.out.println("进入B的Last方法内部");
    }
}

线程对象:
public class DeadLock implements Runnable {

    Object obj = new Object();
    Object obj2 = new Object();
    private boolean flag;//定义一个判断量
    public void lockflag(boolean flag){
        this.flag=flag;
    }

    @Override
    public void run() {
        if(flag){
            //先调用A类的第一个方法,在调用B类的最后一个方法
            synchronized (obj) {
                new DeadClassA().FirstClassA();
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (obj2) {
                    new DeadClassB().LastClassB();
                }
            }

        }
        else{
            //先调用B类的第一个方法,在调用A类的最后一个方法
            synchronized (obj2) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                new DeadClassB().FirstClassB();
                synchronized (obj) {
                    new DeadClassA().LastClassA();
                }
            }
    }

    }   }

测试类:
package Day21_Lock;
/**
 * @author Aoman_Hao
 */
public class DeadLockTest {

    public static void main(String[] args) {

        DeadLock DL = new DeadLock();
        DL.lockflag(true);
        Thread T1 = new Thread(DL);

        DeadLock DL2 = new DeadLock();
        DL2.lockflag(false);
        Thread T2 = new Thread(DL);

        T1.start();
        T2.start();


    }

}

解决死锁用等待唤醒机制
6:递归实现输入任意目录,列出文件以及文件夹

public class Test {

    public static void main(String[] args) {



        File file = new File("D://Java//JavaTest");
        FindFolder(file);

    }

    //递归查询
    public static void FindFolder(File file){
        //读取文件列表
        File[] F_list = file.listFiles();

        //判断是否非空
        if(F_list!=null){
            //循环判断是文件夹的话,进去文件夹,是文件的话,输出文件名
            for(File  filename:F_list){
                //如果是文件夹,递归寻找
                if(filename.isDirectory()){
                    System.out.println("文件夹名:"+filename.getName());
                    FindFolder(filename);
                }else{
                    System.out.println("文件名:"+filename.getName());
                }
            }
        }
    }
}
输出:
文件名:aaa.txt
文件名:Aoman_3.txt
文件名:Aoman_Hao.txt
文件夹名:Test
文件名:ab.txt
文件名:Aoman4122312q.txt
文件名:dwadwadwd.txt