在Java中如何编写简单的验证码生成程序_Java随机数项目解析

发布时间 - 2026-01-30 00:00:00    点击率:
Java生成验证码核心是用BufferedImage和Graphics2D自制图像:选52个易识别字符生成4–5位码,存入session的"captcha_code";绘图时加错切、噪点、随机灰度色,尺寸120×40;响应头禁用缓存,输出PNG并flush。

Java 中生成简单验证码,核心是「随机选字符 + 绘制成图」,不是靠第三方库硬堆功能,而是用 BufferedImageGraphics2D 自己画。关键不在“能不能”,而在“怎么避免被轻易识别”和“怎么让前端能拿到”。

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建站篡改攻击?