1.2 继承 Thread 类
1.2.1 线程创建
当进程创建的时候,会创建一个 主线程(main方法),还有一个就是 JVM 特有的 GC(垃圾回收)线程。
- main函数 叫 “用户线程/主线程” 是自己写的!
- GC 线程是JVM给我们的,也被称为 “守护线程”!
一共有三种 线程的创建方式。
- Runnable 接口 是目前的主流(也是重点)
- Callable 接口 可能是 以后的 主流(但现在只做了解)
1.2.2 Thread
- 自定义线程类继承 Thread 类
- 重写 run() 方法,编写线程执行体
- 创建线程对象,调用 start() 方法启动线程
package www.muquanyu.lesson01;
public class ThreadDemo extends Thread {
@Override
public void run()
{
//run方法线程体
for(int i = 0;i<20;++i)
{
System.out.println("我在看代码----->"+i);
}
}
//主线程
public static void main(String[] args) {
//开启子线程
ThreadDemo threadDemo = new ThreadDemo();
threadDemo.start();
//主线程的执行代码
for(int i = 0;i<20;++i)
{
System.out.println("我在学习多线程----->"+i);
}
}
}
- 运行结果(证明了 我们所谓的多线程,实际上是 伪造的,只是CPU时间切换过快!)
我在学习多线程----->0
我在看代码----->0
我在学习多线程----->1
我在学习多线程----->2
我在学习多线程----->3
我在学习多线程----->4
我在学习多线程----->5
我在学习多线程----->6
我在学习多线程----->7
我在看代码----->1
我在看代码----->2
我在看代码----->3
我在学习多线程----->8
我在看代码----->4
我在学习多线程----->9
我在看代码----->5
我在学习多线程----->10
我在看代码----->6
我在学习多线程----->11
我在看代码----->7
我在看代码----->8
我在学习多线程----->12
我在学习多线程----->13
我在学习多线程----->14
我在学习多线程----->15
我在学习多线程----->16
我在看代码----->9
我在学习多线程----->17
我在学习多线程----->18
我在学习多线程----->19
我在看代码----->10
我在看代码----->11
我在看代码----->12
我在看代码----->13
我在看代码----->14
我在看代码----->15
我在看代码----->16
我在看代码----->17
我在看代码----->18
我在看代码----->19
如果结果是乱的话,那就证明你的电脑配置还不错。如果不是乱的话,那也证明了 你的 CPU 感觉这样的切换对它不好(配置低,要求太多,b事多。考虑的多。)。
1.3 网图下载(多线程快速下载)
我们需要 用到别人写的 内库。IO流。
先去下载 commons-io-2.11.jar 包!
然后 在 src 文件夹下面 建立个 lib 包,把 commons-io-2.11.0.jar 包复制到 lib 包里。
然后再 右键点击 lib --> add as library
用 commons-io 进行 网络文件的下载,是很容易的。只要没有反爬虫机制,就可以下载下来。
- 利用 FileUtils 类下的 copyURLToFile 方法
copyURLToFile(URL,文件); 是将 网络文件 下载 到 指定文件里面的操作。
因为 这个 方法要求必须抛出 异常,并且 适应的捕获,所以我们 还不如 将它封装起来,变成一个新的类。而提供的 URL,FileName 都可以 进行 二次使用。
//下载器
class WebDownloader{
//下载方法
public void downloader(String url,String fileName)
{
try{
FileUtils.copyURLToFile(new URL(url), new File(fileName));
}catch(IOException e)
{
e.printStackTrace();
System.out.println("IO异常,downloader方法出现问题!");
}
}
}
而我们知道,一般在网络爬虫的时候,都是启动多线程去爬的,因为 那样子 更快。所以我们 就把 下载图片的 代码,写到 run 方法里!
@Override
public void run()
{
super.run();
WebDownloader webDownloader = new WebDownloader();
webDownloader.downloader(url,fileName);
System.out.println("下载了文件名:"+fileName);
}
- 再开启 多个线程,实现 图片的下载。
package www.muquanyu.lesson01;
import org.apache.commons.io.FileUtils;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.net.URL;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class TestThread extends Thread {
//网络图片的地址
private String url;
//保存的文件名
private String fileName;
public TestThread(String url,String fileName)
{
this.url = url;
this.fileName = fileName;
}
@Override
public void run()
{
super.run();
WebDownloader webDownloader = new WebDownloader();
webDownloader.downloader(url,fileName);
System.out.println("下载了文件名:"+fileName);
}
public static void main(String[] args) {
TestThread t1 = new TestThread("https://i0.hdslb.com/bfs/feed-admin/286943cdce0446bb925fc6e065089283991c919b.jpg@880w_388h_1c_95q", "1.jpg");
TestThread t2 = new TestThread("https://i0.hdslb.com/bfs/feed-admin/286943cdce0446bb925fc6e065089283991c919b.jpg@880w_388h_1c_95q", "2.jpg");
TestThread t3 = new TestThread("https://i0.hdslb.com/bfs/feed-admin/286943cdce0446bb925fc6e065089283991c919b.jpg@880w_388h_1c_95q", "3.jpg");
TestThread t4 = new TestThread("http://image.baidu.com/search/index?tn=baiduimage&ipn=r&ct=201326592&cl=2&lm=-1&st=-1&fm=result&fr=&sf=1&fmq=1626391305893_R&pv=&ic=0&nc=1&z=&hd=&latest=©right=&se=1&showtab=0&fb=0&width=&height=&face=0&istype=2&ie=utf-8&sid=&word=%E6%88%91%E7%9A%84", "1.txt");
//URL u = TestThread.class.getResource("1.txt");
t1.start();
t2.start();
t3.start();
t4.start();
try{
//创建 BufferReader 缓冲阅读者,来把 读取出来的文件 存储到 缓冲区里
BufferedReader bufferedReader = new BufferedReader(new FileReader("1.txt"));
//制定正则表达式 规则
String str = "<title>(.*?)</title>";
String s = null;
String sumStr = null;
while((s = bufferedReader.readLine())!=null)
{
sumStr+=s;
}
//创建 这个 规则模式
Pattern p = Pattern.compile(str);
//用匹配器 按照 这个 规则模式 对 输入进来的数据 进行匹配
Matcher m =p.matcher(sumStr);
//判断 是否 有匹配上的?
while(m.find())
{
int i = 1;
//如果有匹配上的,那么 我们就 用 group 进行 获取(注意:group 必须 和 find 连用)
System.out.println(m.group(i));
i++;
}
bufferedReader.close();
}catch (Exception e)
{
System.out.println("出现异常!");
}
}
}
//下载器
class WebDownloader{
//下载方法
public void downloader(String url,String fileName)
{
try{
FileUtils.copyURLToFile(new URL(url), new File(fileName));
}catch(IOException e)
{
e.printStackTrace();
System.out.println("IO异常,downloader方法出现问题!");
}
}
}