爬虫

爬虫可以分为本地爬虫和网络爬虫.

本地爬虫

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo6 {
public static void main(String[] args) {
/*
* 有如下文本, 请按照要求爬取数据.
* Java 自从 95 年问世以来, 经历了很多版本, 目前企业中用的最多的是 Java8 和 Java11,
* 因为这两个是长期支持版本, 下一个长期支持版本是 Java17, 相信在未来不久 Java17 也会逐渐登上历史舞台
* 要求: 找出里面所有的 JavaXX
*/
String str = "Java 自从 95 年问世以来, 经历了很多版本, 目前企业中用的最多的是 Java8 和 Java11, " +
"因为这两个是长期支持版本, 下一个长期支持版本是 Java17, 相信在未来不久 Java17 也会逐渐登上历史舞台";
// 1. 获取正则表达式的对象
Pattern p = Pattern.compile("Java\\d{0,2}");
// 2. 获取文本匹配器的对象
// 拿着 m 去读取 str, 找符合 p 规则的子串
Matcher m = p.matcher(str);
// 拿着文本匹配器从头开始读取,寻找是否有满足规则的子串
// 如果没有, 方法返回 false
// 如果有, 返回 true. 在底层记录子串的起始索引和结束索引 +1
// 0, 4
boolean b = m.find();
// 方法底层会根据 find 方法记录的索引进行字符串的截取
// subString(起始索引, 结束索引); // 包头不包尾
// (0, 4) 但是不包含 4 索引
// 会把截取的小串进行返回.
String s1 = m.group();
System.out.println(s1); // Java
// 第二次在调用 find 的时候, 会继续读取后面的内容
// 读取到第二个满足要求的子串, 方法会继续返回 true
// 并把第二个子串的起始索引和结束索引 +1, 进行记录
b = m.find();
// 第二次调用 group 方法的时候, 会根据 find 方法记录的索引再次截取子串
String s2 = m.group();
System.out.println(s2); // Java8
}
}

这样显然不可行, 需要用循环改写代码.

把原来的代码抽取出来形成一个方法放到外面, 但是并不去使用它. 改写代码:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo6 {
public static void main(String[] args) {
/*
* 有如下文本, 请按照要求爬取数据.
* Java 自从 95 年问世以来, 经历了很多版本, 目前企业中用的最多的是 Java8 和 Java11,
* 因为这两个是长期支持版本, 下一个长期支持版本是 Java17, 相信在未来不久 Java17 也会逐渐登上历史舞台
* 要求: 找出里面所有的 JavaXX
*/
String str = "Java 自从 95 年问世以来, 经历了很多版本, 目前企业中用的最多的是 Java8 和 Java11, " +
"因为这两个是长期支持版本, 下一个长期支持版本是 Java17, 相信在未来不久 Java17 也会逐渐登上历史舞台";
// 1. 获取正则表达式的对象
Pattern p = Pattern.compile("Java\\d{0,2}");
// 2. 获取文本匹配器的对象
// 拿着 m 去读取 str, 找符合 p 规则的子串
Matcher m = p.matcher(str);
// 3. 利用循环获取
while (m.find()) {
String s = m.group();
System.out.println(s);
}
}
private static void method1(String str) {
// 1. 获取正则表达式的对象
Pattern p = Pattern.compile("Java\\d{0,2}");
// 2. 获取文本匹配器的对象
// 拿着 m 去读取 str, 找符合 p 规则的子串
Matcher m = p.matcher(str);
// 拿着文本匹配器从头开始读取,寻找是否有满足规则的子串
// 如果没有, 方法返回 false
// 如果有, 返回 true. 在底层记录子串的起始索引和结束索引 +1
// 0, 4
boolean b = m.find();
// 方法底层会根据 find 方法记录的索引进行字符串的截取
// subString(起始索引, 结束索引); // 包头不包尾
// (0, 4) 但是不包含 4 索引
// 会把截取的小串进行返回.
String s1 = m.group();
System.out.println(s1); // Java
// 第二次在调用 find 的时候, 会继续读取后面的内容
// 读取到第二个满足要求的子串, 方法会继续返回 true
// 并把第二个子串的起始索引和结束索引 +1, 进行记录
b = m.find();
// 第二次调用 group 方法的时候, 会根据 find 方法记录的索引再次截取子串
String s2 = m.group();
System.out.println(s2); // Java8
}
}

运行结果:

Java
Java8
Java11
Java17
Java17

网络爬虫

程序示例:

public class RegexDemo7 {
public static void main(String[] args) throws IOException {
/*
* 扩展需求2:
* https://www.bilibili.com/
*/
// 创建一个 URL 对象
URL url = new URL("https://www.bilibili.com/");
// 连接上这个网址
// 细节: 保证网络是畅通
URLConnection conn = url.openConnection();
// 创建一个对象去读取网络中的数据
BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream()));
String line;
while ((line = br.readLine()) != null) {
System.out.println(line);
}
br.close();
}
}

运行结果:

<!DOCTYPE html><html><head><title>验证码_哔哩哔哩</title><meta name="viewport" content="width=device-width,user-scalable=no,initial-scale=1,maximum-scale=1,minimum-scale=1,viewport-fit=cover"><meta name="spm_prefix" content="333.1291"><script>window._BiliGreyResult={"method":"base","grayVersion":"13372"}</script><script type="text/javascript" src="//www.bilibili.com/gentleman/polyfill.js?features=Promise%2CObject.assign%2CString.prototype.includes%2CNumber.isNaN"></script>
<script>
window._riskdata_ = {
'v_voucher': 'voucher_e356c93d-6a09-4600-9948-a8f7c307fd92'
}
</script>
<script type="text/javascript" src="//s1.hdslb.com/bfs/seed/log/report/log-reporter.js"></script><link href="//s1.hdslb.com/bfs/static/jinkela/risk-captcha/css/risk-captcha.0.8a8937eeb2b2c3216dcd9e09a643c16b6e8482c5.css" rel="stylesheet"></head><body><div id="biliMainHeader"></div><div id="risk-captcha-app"></div><script src="//s1.hdslb.com/bfs/seed/jinkela/risk-captcha-sdk/CaptchaLoader.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/static/jinkela/risk-captcha/1.risk-captcha.8a8937eeb2b2c3216dcd9e09a643c16b6e8482c5.js"></script><script type="text/javascript" src="//s1.hdslb.com/bfs/static/jinkela/risk-captcha/risk-captcha.8a8937eeb2b2c3216dcd9e09a643c16b6e8482c5.js"></script></body></html>

程序示例:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.URL;
import java.net.URLConnection;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo7 {
public static void main(String[] args) throws IOException {
/*
* 扩展需求 2:
* 把链接 https://m.sengzan.com/jiaoyu/29104.html?ivk sa=1025883i 中所有的身份证号码都爬取出来.
*/
// 创建一个 URL 对象
URL url = new URL("https://m.sengzan.com/jiaoyu/29104.html?ivk_sa=1025883i");
// 连接上这个网址
// 细节: 保证网络是畅通
URLConnection conn = url.openConnection();
// 创建一个对象去读取网络中的数据
BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream()));
String line;
// 获取正则表达式的对象 pattern
String regex = "[1-9]\\d{17}";
Pattern pattern = Pattern.compile(regex);
// 在读取的时候每次读一整行
while ((line = br.readLine()) != null) {
// 拿着文本匹配器的对象 matcher 按照 pattern 的规则去读取当前的这一行信息
Matcher matcher = pattern.matcher(line);
while (matcher.find()) {
System.out.println(matcher.group());
}
}
br.close();
}
}

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo8 {
public static void main(String[] args) {
/*
* 需求: 把下面文本中的座机电话, 邮箱, 手机号, 热线都爬取出来.
*
* 来黑马程序员学习Java,
* 手机号: 18512516758, 18512508907 或者联系邮箱: boniu@itcast.cn,
* 座机电话: 01036517895, 010-98951256 邮箱: bozai@itcast.cn,
* 热线电话: 400-618-9090 , 400-618-4000, 4006184000, 4006189090
*
* 手机号的正则表达式: 1[3-9]\d{9}
* 邮箱的正则表达式: \w+@[\w&&[^_]]{2,6}(\.[a-zA-Z]{2,3}){1,2}
* 座机电话的正则表达式: θ\d{2,3}-?[1-9]\d{4,9}
* 热线电话的正则表达式: 400-?[1-9]\\d{2}-?[1-9]\\d{3}
*/
String s = "来黑马程序员学习Java, " +
"手机号: 18512516758, 18512508907 或者联系邮箱: boniu@itcast.cn, " +
"座机电话: 01036517895, 010-98951256 邮箱: bozai@itcast.cn, " +
"热线电话: 400-618-9090 , 400-618-4000, 4006184000, 4006189090";
System.out.println("400-618-9090");
String regex = "(1[3-9]\\d{9})|(\\w+@[\\w&&[^_]]{2,6}(\\.[a-zA-Z]{2,3}){1,2})" +
"|(0\\d{2,3}-?[1-9]\\d{4,9})" +
"|(400-?[1-9]\\d{2}-?[1-9]\\d{3})";
// 1. 获取正则表达式的对象
Pattern p = Pattern.compile(regex);
// 2. 获取文本匹配器的对象
// 利用 m 去读取 s, 会按照 p 的规则找里面的小串
Matcher m = p.matcher(s);
// 3. 利用循环获取每一个数据
while (m.find()) {
String str = m.group();
System.out.println(str);
}
}
}

带条件的爬取

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo9 {
public static void main(String[] args) {
/*
* 有如下文本, 按要求爬取数据.
*
* Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11,
* 因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台
*
* 需求 1: 爬取版本号为 8, 11, 17 的 Java 文本, 但是只要 Java, 不显示版本号.
*/
String s = "Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11, " +
"因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台";
// 1. 定义正则表达式 Java(?=8|11|17)
// ? 理解为前面的数据 Java
// = 表示在 Java 后面要跟随的数据
// 但是在获取的时候, 只获取前半部分
String regex = "Java(?=8|11|17)";
// 或者
String regex1 = "((?i)Java)(?=8|11|17)";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(s);
while (m.find()) {
System.out.println(m.group());
}
}
}

执行结果:

Java
Java
Java
Java

第一个 Java , 也就是 "Java自从95年问世以来" 中的 Java, 不能被匹配到, 因为后面没有跟数字, 不符合要求.

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo9 {
public static void main(String[] args) {
/*
* 有如下文本, 按要求爬取数据.
*
* Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11,
* 因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台
*
* 需求 2: 爬取版本号为 8, 11, 17 的 Java文本. 正确爬取结果为: Java8 Java11 Java17 Java17
*/
String s = "Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11, " +
"因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台";
// 1. 定义正则表达式
// ? 理解为前面的数据 Java
// : 表示在 Java 后面要跟随的数据
// 在获取的时候, 获取全部内容
// 需求 2:
String regex2 = "((?i)Java)(8|11|17)";
// 或者
String regex3 = "((?i)Java)(?:8|11|17)";
Pattern p = Pattern.compile(regex3);
Matcher m = p.matcher(s);
while (m.find()) {
System.out.println(m.group());
}
}
}

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo9 {
public static void main(String[] args) {
/*
* 有如下文本, 按要求爬取数据.
*
* Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11,
* 因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台
*
* 需求 3: 爬取除了版本号为 8, 11, 17 的 Java 文本
*/
String s = "Java自从95年问世以来, 经历了很多版本, 目前企业中用的最多的是Java8和Java11, " +
"因为这两个是长期支持版本, 下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台";
// 1. 定义正则表达式
// ? 理解为前面的数据 Java
// ! 表示在 Java 后面要跟随的数据, 但是要去除
// 需求 3:
String regex4 = "((?i)Java)(?!8|11|17)";
Pattern p = Pattern.compile(regex4);
Matcher m = p.matcher(s);
while (m.find()) {
System.out.println(m.group());
}
}
}

执行结果:

Java

贪婪爬取和非贪婪爬取

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo10 {
public static void main(String[] args) {
/*
* 只写 + 和 * 表示贪婪匹配
*
* +? : 非贪婪匹配
* *? : 非贪婪匹配
*
* 贪婪爬取: 在爬取数据的时候尽可能的多获取数据
* 非贪婪爬取: 在爬取数据的时候尽可能的少获取数据
*
* ab+:
* 贪婪爬取: abbbbbbbbbbbb
* 非贪婪爬取: ab
*/
String s = "Java自从95年问世以来, abbbbbbbbbbbbaaaaaaaaaaaaaaaaaa" +
"经历了很多版木, 目前企业中用的最多的是]ava8和]ava11, 因为这两个是长期支持版木. " +
"下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台";
String regex = "ab+";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(s);
while (m.find()) {
System.out.println(m.group()); // abbbbbbbbbbbb
}
}
}

程序示例:

import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class RegexDemo10 {
public static void main(String[] args) {
/*
* 只写 + 和 * 表示贪婪匹配
*
* +? : 非贪婪匹配
* *? : 非贪婪匹配
*
* 贪婪爬取: 在爬取数据的时候尽可能的多获取数据
* 非贪婪爬取: 在爬取数据的时候尽可能的少获取数据
*
* ab+:
* 贪婪爬取: abbbbbbbbbbbb
* 非贪婪爬取: ab
*/
String s = "Java自从95年问世以来, abbbbbbbbbbbbaaaaaaaaaaaaaaaaaa" +
"经历了很多版木, 目前企业中用的最多的是]ava8和]ava11, 因为这两个是长期支持版木. " +
"下一个长期支持版本是Java17, 相信在未来不久Java17也会逐渐登上历史舞台";
String regex = "ab+?";
Pattern p = Pattern.compile(regex);
Matcher m = p.matcher(s);
while (m.find()) {
System.out.println(m.group()); // ab
}
}
}

正则表达式在字符串方法中的使用


图1 正则表达式在字符串方法中的使用

程序示例:

public class RegexDemo11 {
public static void main(String[] args) {
// public string replaceAll(String regex, String newstr) : 按照正则表达式的规则进行替换
// public string[] split(String regex) : 按照正则表达式的规则切割字符串
/*
* 有一段字符串: 小诗诗dqwefqwfqwfwq12312小丹丹dqwefqwfqwfwq12312小惠惠
* 要求 1: 把字符串中三个姓名之间的字母替换为 vs
* 要求 2: 把字符串中的三个姓名切割出来
*/
String s = "小诗诗dqwefqwfqwfwq12312小丹丹dqwefqwfqwfwq12312小惠惠";
// 细节:
// 方法在底层跟之前一样也会创建文本解析器的对象
// 然后从头开始去读取字符串中的内容, 只要有满足的, 那么就用第一个参数去替换.
String result1 = s.replaceAll("[\\w&&[^_]]+", "vs"); // + 表示前面的正则表达式至少出现一次
System.out.println(result1); // 小诗诗vs小丹丹vs小惠惠
String[] arr = s.split("[\\w&&[^_]]+");
for (int i = 0; i < arr.length; i++) {
System.out.println(arr[i]);
}
}
}

String 类中方法那么多, 怎么看一个方法是否识别正则表达式呢?

在 Java API 中, 如果一个方法的形参取名字叫 regex, 那么这个方法是能识别正则表达式的.


图2

图3

如果不识别正则表达式, 却传递了一个正则表达式, 则该正则表达式被识别为一个普通的字符串.

分组

分组就是小括号.


图4

每组是有组号的,也就是序号.

  • 规则 1: 从 1 开始, 连续不间断.

  • 规则 2: 以左括号为基准, 最左边的是第一组, 其次为第二组, 以此类推.


图5

图6

捕获分组就是把这一组的数据捕获出来, 再用一次.

程序示例:

public class RegexDemo12 {
public static void main(String[] args) {
// 需求 1: 判断一个字符串的开始字符和结束字符是否一致? 只考虑一个字符
// 举例: a123a b456b 17891 &abc& a123b(false)
// \\ 组号: 表示把第 X 组的内容再出来用一次
String regex1 = "(.).+\\1";
System.out.println("a123a".matches(regex1));
System.out.println("b456b".matches(regex1));
System.out.println("17891".matches(regex1));
System.out.println("&abc&".matches(regex1));
System.out.println("a123b".matches(regex1));
System.out.println("--------------------------");
// 需求 2: 判断一个字符串的开始部分和结束部分是否一致? 可以有多个字符
// 举例: abc123abc b456b 123789123 &!@abc&!@ abc123abd(false)
String regex2 = "(.+).+\\1";
System.out.println("abc123abc".matches(regex2));
System.out.println("b456b".matches(regex2));
System.out.println("123789123".matches(regex2));
System.out.println("&!@abc&!@".matches(regex2));
System.out.println("abc123abd".matches(regex2));
System.out.println("---------------------");
// 需求 3: 判断一个字符串的开始部分和结束部分是否一致? 开始部分内部每个字符也需要一致
// 举例: aaa123aaa bbb456bbb 111789111 &&abc&&
// (.): 把首字母看做一组
// \\2: 把首字母拿出来再次使用
// *: 作用于 \\2, 表示后面重复的内容出现 0 次或多次
String regex3 = "((.)\\2*).+\\1";
System.out.println("aaa123aaa".matches(regex3));
System.out.println("bbb456bbb".matches(regex3));
System.out.println("111789111".matches(regex3));
System.out.println("&&abc&&".matches(regex3));
System.out.println("aaa123aab".matches(regex3));
}
}

在正则表达式的外部, 也可以使用组里面的信息.

  • 正则内部使用: \\组号

  • 正则外部使用: $组号

程序示例:

public class RegexDemo13 {
public static void main(String[] args) {
/*
* 需求:
* 将字符串: 我要学学编编编编程程程程程程 替换为: 我要学编程
*/
String str = "我要学学编编编编程程程程程程";
// 需求: 把重复的内容 替换为 单个的
// 学学 学
// 编编编编 编
// 程程程程程程 程
// (.) 表示把重复内容的第一个字符看做一组
// \\1 表示第一个字符再次出现
// + 至少一次
// $1 表示把正则表达式中第一组的内容, 再拿出来用
String result = str.replaceAll("(.)\\1+", "$1");
System.out.println(result); // 我要学编程
}
}

非捕获分组不占用组号.


图7

程序示例:

public class RegexDemo14 {
public static void main(String[] args) {
/*
* 非捕获分组: 分组之后不需要再用本组数据, 仅仅是把数据括起来.
*
* 身份证号码:
* 41080119930228457x
* 510801197609022309
* 15040119810705387X
* 130133197204039024
* 430102197606046442
*/
// 身份证号码的简易正则表达式
// 非捕获分组: 仅仅是把数据括起来 // 特点: 不占用组号
// 这里 \\1 报错原因: (?:) 就是非捕获分组, 此时是不占用组号的.
// (?:) (?=) (?!) 都是非捕获分组
// 更多的使用第一个
// String regex1 = "[1-9]\\d{16}(?:\\d|x|x)\\1"; // 当使用 regex1 时报错: Unresolved back reference
String regex2 = "[1-9]\\d{16}(\\dXx)\\1";
// System.out.println("41080119930228457x".matches(regex1));
System.out.println("41080119930228457x".matches(regex2));
}
}
posted @   有空  阅读(6)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
点击右上角即可分享
微信分享提示