在Java中如何编写简单的验证码生成程序_Java随机数项目解析
发布时间 - 2026-01-30 00:00:00 点击率:次Java生成验证码核心是用BufferedImage和Graphics2D自制图像:选52个易识别字符生成4–5位码,存入session的"captcha_code";绘图时加错切、噪点、随机灰度色,尺寸120×40;响应头禁用缓存,输出PNG并flush。
Java 中生成简单验证码,核心是「随机选字符 + 绘制成图」,不是靠第三方库硬堆功能,而是用 BufferedImage 和 Graphics2D 自己画。关键不在“能不能”,而在“怎么避免被轻易识别”和“怎么让前端能拿到”。
用 Random 生成 4–6 位字母数字组合
别用 Math.random(),它不支持设置种子,不利于测试;也别直接拼接 "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ" 然后 charAt(rand.nextInt(len)) —— 默认会包含易混淆字符(如 0/O、1/l/I),实际部署时容易被用户投诉。
- 推荐字符集:去掉
'0','O','1','l','I',剩下约 52 个安全字符 - 长度固定为 4 或 5 位,太短不安全,太长用户输错率飙升
- 务必在生成后把验证码字符串存入
HttpSession,键名统一用"captcha_code",前端校验时才对得上
用 BufferedImage 绘制带干扰的图片
纯白底 + 黑字的图,OCR 一扫一个准。必须加干扰:噪点、斜线、轻微扭曲、字符间距随机化。但注意——Graphics2D.rotate() 直接旋转会导致字符边缘锯齿严重,且坐标难算;更稳妥的是用 shear(错切)+ translate 微调位置。
- 图片尺寸建议设为
width=120,height=40,留出左右边距,避免字符贴边 - 每画一个字符前,用
g2d.setColor(new Color(50 + rand.nextInt(100), 50 + rand.nextInt(100), 50 + rand.nextInt(100)))随机灰度色,别全黑 - 画完字符后,用双重循环在图上随机打 30–50 个单像素点(
g2d.fillRect(x, y, 1, 1)),颜色从背景色附近扰动
Servlet 中输出 PNG 并禁用缓存
很多人写完发现浏览器反复加载同一张图,或者验证码明明换了但前端还显示旧的——八成是没设响应头。response.setContentType("image/png") 只管 MIME 类型,不管缓存逻辑。
- 必须加上:
response.setHeader("Cache-Control", "no-store") - 同时加:
response.setHeader("Pragma", "no-cache")和response.setDateHeader("Expires", 0) - 输出流要用
ImageIO.write(bufferedImage, "png", response.getOutputStream()),别用toString()或手动转 Base64 - 最后务必调用
response.getOutputStream().flush(),否则部分容器(如 Tomcat 9+)可能卡住不返回
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
HttpSession session = request.getSession(true);
String code = generateCaptchaText(); // 自定义方法
session.setAttribute("captcha_code", code);
BufferedImage image = new BufferedImage(120, 40, BufferedImage.TYPE_INT_RGB);
Graphics2D g2d = image.createGraphics();
g2d.setColor(Color.WHITE);
g2d.fillRect(0, 0, 120, 40);
Random rand = new Random();
for (int i = 0; i < code.length(); i++) {
g2d.setColor(new Color(50 + rand.nextInt(100), 50 + rand.nextInt(100), 50 + rand.nextInt(100)));
g2
d.setFont(new Font("Arial", Font.BOLD, 24));
g2d.translate(5 + i * 22, 28);
g2d.shear(0.1 - rand.nextDouble() * 0.2, 0);
g2d.drawString(code.charAt(i) + "", 0, 0);
g2d.shear(0, 0);
g2d.translate(-5 - i * 22, -28);
}
for (int i = 0; i < 40; i++) {
int x = rand.nextInt(120);
int y = rand.nextInt(40);
g2d.setColor(new Color(rand.nextInt(100), rand.nextInt(100), rand.nextInt(100)));
g2d.fillRect(x, y, 1, 1);
}
g2d.dispose();
response.setContentType("image/png");
response.setHeader("Cache-Control", "no-store");
response.setHeader("Pragma", "no-cache");
response.setDateHeader("Expires", 0);
ImageIO.write(image, "png", response.getOutputStream());
response.getOutputStream().flush();}
真正难的不是画图,是让这个图在 Nginx 反向代理后仍不被缓存、在移动端缩放时不糊、在 IE11 下也能正常渲染——这些细节不写进代码里,上线后就会变成凌晨三点的告警。
# java
# 前端
# nginx
# 浏览器
# session
# tomcat
# stream
# 验证码生成
# red
# servlet
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Linux系统命令中tree命令详解
最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?
佛山企业网站制作公司有哪些,沟通100网上服务官网?
详解CentOS6.5 安装 MySQL5.1.71的方法
Laravel如何使用Gate和Policy进行授权?(权限控制)
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
Laravel怎么实现支付功能_Laravel集成支付宝微信支付
Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法
Laravel Asset编译怎么配置_Laravel Vite前端构建工具使用
如何在IIS管理器中快速创建并配置网站?
C语言设计一个闪闪的圣诞树
如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
北京的网站制作公司有哪些,哪个视频网站最好?
Python自然语言搜索引擎项目教程_倒排索引查询优化案例
Laravel如何生成API文档?(Swagger/OpenAPI教程)
如何在 Pandas 中基于一列条件计算另一列的分组均值
再谈Python中的字符串与字符编码(推荐)
Swift中switch语句区间和元组模式匹配
Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
Laravel如何使用API Resources格式化JSON响应_Laravel数据资源封装与格式化输出
魔方云NAT建站如何实现端口转发?
INTERNET浏览器怎样恢复关闭标签页_INTERNET浏览器标签恢复快捷键与方法【指南】
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
简历在线制作网站免费版,如何创建个人简历?
IOS倒计时设置UIButton标题title的抖动问题
Bootstrap CSS布局之列表
Laravel如何发送邮件_Laravel Mailables构建与发送邮件的简明教程
Laravel如何实现登录错误次数限制_Laravel自带LoginThrottles限流配置【方法】
详解Nginx + Tomcat 反向代理 如何在高效的在一台服务器部署多个站点
教你用AI将一段旋律扩展成一首完整的曲子
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】
Laravel如何自定义错误页面(404, 500)?(代码示例)
JavaScript中如何操作剪贴板_ClipboardAPI怎么用
Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】
原生JS获取元素集合的子元素宽度实例
如何在阿里云高效完成企业建站全流程?
Android使用GridView实现日历的简单功能
PHP 500报错的快速解决方法
Laravel如何实现用户注册和登录?(Auth脚手架指南)
如何用西部建站助手快速创建专业网站?
php静态变量怎么调试_php静态变量作用域调试技巧【解答】
JavaScript如何实现错误处理_try...catch如何捕获异常?
Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能
Laravel如何处理JSON字段的查询和更新_Laravel JSON列操作与查询技巧
如何有效防御Web建站篡改攻击?


