实现图片验证码【详细代码】

article/2025/9/20 22:35:38

        实际开发过程中经常遇到要实现图片验证码来防止外部使用脚本刷接口,所以说图片验证码是很有必要的一个小功能。

html

<!--- 注册页面整增加图形验证码功能,这里为了更贴近企业级业务,我们在注册页面整增加图形验证码功能-->
<div class="user-phone"><label for="imageCode"><i class="am-icon-check am-icon-sm"></i></label><!--这里v-model="phoneUserForm.imageCode" 双向绑定 用户根据图形验证码输入的验证码   style="width: 180px;"设置文本的输入长度  --><input type="text" name="" style="width: 140px;" v-model="phoneUserForm.base64ImageCode"id="imageCode" placeholder="图片验证码"><!--@click="getImageCode":src="base64ImageCode" 点击发送axios异步请求到后端,获取到图片过后直接显示在imageCode--><!--@click="getImageCode" vue语法 ==v-on:click--><img id="captcha-image-temp" @click="getImageCode" :src="base64ImageCode"class="captcha-image-temp" alt="点击换图" title="点击换图"style="vertical-align: middle; cursor: pointer;"><button type="button" style="font-size: 10px;color: purple" @click="getImageCode">看不清</button>
</div>

模型数据

new Vue({
data: { /*模型数据*//*** 图形验证码的两个参数* */base64ImageCode: "", // 图片验证码的图片  获取后端经过Base64编码加密后的string字符串,获取到了就会显示在前端,后端传递就会显示imageCodePrefix: "data:image/jpeg;base64,", // 图形验证码显示的前缀,需要加上这个才能显示经过base64编译加密后的String在前端/*** 手机号注册* */phoneUserForm: {base64ImageCode: "", // 点击获取【手机验证码】按钮的时候将 用户输入的图片验证码传递到后端phoneNum: "15555555555", // 用户输入的手机号imageCodeKey: "", // 存储图形验证码的key    即传递到后端的UUID,点击获取验证码的时候进行赋值 【唯一的--避免产生大量无用的验证码】,后端redis将其当作key 生成的图形验证码为valuepassword: "", // 页面输入的密码},})

methods

methods: {/*** 获取UUID,为了获取图片验证码,不会产生大量无用的图片验证码,放置恶意攻击* */createUuid() { // 获取UUID 通用唯一识别码(Universally Unique Identifier)var s = [];var hexDigits = "0123456789abcdefghi";for (var i = 0; i < 36; i++) {s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);}s[14] = "4"; // bits 12-15 of the time_hi_and_version field to 0010s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1); // bits 6-7 of the clock_seq_hi_and_reserved to 01s[8] = s[13] = s[18] = s[23] = "-";var uuid = s.join("");return uuid;},
/*** 获取图片验证码* 页面一加载就会产生UUID-key执行该方法发送异步请求并携带UUID去后端获取图片验证码* */getImageCode() {// 通过localStorage【存储key和value-UUID】通过自定义的键registerImageCodeKey 去获取是否有imageCodeKey【即UUID】// 这个key【registerImageCodeKey】其实一开始是没有,对应的imageCodeKey【UUID】也是没有的。是在下面判断后设定的。let imageCodeKey = localStorage.getItem("registerImageCodeKey");// 如果为false获取这为空字符串就创建UUIDif (!imageCodeKey) {// 调用方法创建一个imageCodeKey【UUID】,这里才是真正的创建UUID,一开始是没有的imageCodeKey = this.createUuid();console.log("前端生成的UUID:" + imageCodeKey); // 判断是否生成UUID*/// 将UUID通过键值对的方式存储到localStorage中 以后通过registerImageCodeKey 去获取属性值imageCodeKey【UUID】localStorage.setItem("registerImageCodeKey", imageCodeKey);}/** 前端页面一加载就发送异步请求去后端 携带imageCodeKey【UUID】* 这里传入的【imageCodeKey】UUID会作为Redis的key进行存储,value就是生成的图形验证码经过base64编码加密后的String* Redis采用键值对的方式进行存储数据在内存中*必须传入到后端,为了避免产生大量无用的验证码,携带过去直接就可以进行覆盖  */this.$http.get("/verifyCode/image/" + imageCodeKey).then(res => {// 直接向后端发送请求获取图形验证码的字符串 并进行拼接,赋值给imageCode// res.data.resultObj 就是 AjaxResult对象中的属性resultObj 即为图像验证码加密后的String// 获取到图形验证码 显示在前端this.base64ImageCode = this.imageCodePrefix + res.data.resultObj;})
},
}

mounted

        如果需要页面一加载就发送请求到后端获取图片验证码,就可以使用mounted钩子函数

//页面加载事件
mounted() { /*钩子函数*/this.getImageCode(); // 这个再点击重新获取验证码时也会触发
}

后端

Controller

/*** I 注解注入一个对象*/@Autowiredprivate IVerifyCodeService iVerifyCodeService;
@RequestMapping("/verifyCode")
public class VerifyCodeController { /*** I 注解注入一个对象*/@Autowiredprivate IVerifyCodeService iVerifyCodeService;/*** I 注解注入一个对象*/@Autowiredprivate IVerifyCodeService iVerifyCodeService;/*** 获取图片验证码的接口* Restful风格是获取的请求 所有使用GetMapping 传递的参数是UUID(key)* 后端可以使用Redis进行将前端传递的UUID作为key 生成的验证码作为value进行存储*/@GetMapping("/image/{key}")@ApiOperation(value = "获取图片验证码的接口")public AjaxResult graphVerifyCode(@PathVariable("key") String key) {/*** 快捷键 ctrl + Alt + t 自动生成try catch。。* */System.out.println("前端传入的UUID:" + key);try {// 获取的是一个经过Base64加密后的StringString base64Str = iVerifyCodeService.graphVerifyCode(key);// 使用单例模式 直接获取一个AjaxResult对象// 并将经过base64加密后的String 设置到AjaxResult的如resultObj属性中return AjaxResult.getAjaxResult().setResultObj(base64Str);} catch (Exception e) {e.printStackTrace();// 直接通过set方法 进行设置提示语return AjaxResult.getAjaxResult().setSuccess(false).setMsg("操作失败");}}
}

service

public interface IVerifyCodeService {/*** 用来获取业务层经过逻辑处理后的经过base64加密后的String*/String graphVerifyCode(String key);

serviceImpl

@Service
public class VerifyCodeServiceImpl implements IVerifyCodeService {/*** 需要注入RedisTemplate* 用redis将前端生成的UUID作为key进行存储,value就是后端图形验证码经过base64编码加密后的String到内存中,能够快速进行获取,提高效率*/@Autowiredprivate RedisTemplate redisTemplate;/*** 注入UserMapper 通过根据电话号码进行查询是否已经注册过*/@Autowiredprivate UserMapper userMapper;
/*** 获取图片的随机验证码*/
@Override
public String graphVerifyCode(String key) {  // key就是传入的UUID// 校验前端传入的UUID是否是一个空值if (!StringUtils.hasLength(key)) { // 如果没有长度 则说明没有keythrow new BusinessException("请输入Key");}// 使用工具类获取4位的随机数验证码String code = StrUtils.getComplexRandomString(4);System.out.println("后台生成的4位图片随机验证码" + code);// 4将Code存储在Redis中(在缓冲中,读取速度比较快),使用前端传递的UUID作为Redis的key,code为值,设置过期时间。//  不允许产生大量无用地图片验证码。【每次都携带UUID,如果再次获取就进行覆盖图形验证码。】,//  对值进行操作 设置 k为identifing_code v为10086 时长为10 单位为秒redisTemplate.opsForValue().set(key, // key位前端传递的UUID 每个浏览器唯一code, // 生成的随机数验证码3, // 设置过期时间TimeUnit.MINUTES // 过期时间的单位);//  5使用自带的2D引擎将图片验证码的code生成一个图片//  6将图片基于Base64编码加密成String,响应给前端 参数为图片显示的高度和宽度,传入code为生成的随机数验证码String graphVerifyStr = VerifyCodeUtils.VerifyCode(105, 42, code);//  7前端拿到base字符串,进行图片的展示,用户输入图片验证码System.out.println("经过加密后的String:" + graphVerifyStr);// 将图片验证码经过base64编码加密后的String返回给前端,前端通过加上前缀就可以进行显示return graphVerifyStr;
}
}

strutils工具类

/*** StrUtils工具类*/
public class StrUtils {/*** 把逗号分隔的字符串转换字符串数组** @param str* @return*/public static String[] splitStr2StrArr(String str, String split) {if (str != null && !str.equals("")) {return str.split(split);}return null;}/*** 把逗号分隔字符串转换List的Long** @param str* @return*/public static List<Long> splitStr2LongArr(String str) {String[] strings = splitStr2StrArr(str, ",");if (strings == null) return null;List<Long> result = new ArrayList<>();for (String string : strings) {result.add(Long.parseLong(string));}return result;}/*** 把逗号分隔字符串转换List的Long** @param str* @return*/public static List<Long> splitStr2LongArr(String str, String split) {String[] strings = splitStr2StrArr(str, split);if (strings == null) return null;List<Long> result = new ArrayList<>();for (String string : strings) {result.add(Long.parseLong(string));}return result;}/*** 获取随机的字符串*/public static String getRandomString(int length) {String str = "0123456789";Random random = new Random();StringBuffer sb = new StringBuffer();for (int i = 0; i < length; i++) {int number = random.nextInt(str.length());sb.append(str.charAt(number));}return sb.toString();}/*** 获取复杂的随机的字符串* 参数为需要生成的随机数个数* <p>**/public static String getComplexRandomString(int length) {
//      String str = "!@#¥%&*abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";String str = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";Random random = new Random();  // 创建一个随机数对象StringBuffer sb = new StringBuffer(); // 创建一个StringBuffer对象,用来存储生成的验证码,线程安全且字符串长度可变for (int i = 0; i < length; i++) { // 需要执行的次数int number = random.nextInt(str.length()); // 生成的随机数作为下标,去str字符串中去取对应的值sb.append(str.charAt(number)); // 通过随机生成的数字,取对应的值拼接到StringBuffer中}return sb.toString();}public static String convertPropertiesToHtml(String properties) {//1:容量:6:32GB_4:样式:12:塑料壳StringBuilder sBuilder = new StringBuilder();String[] propArr = properties.split("_");for (String props : propArr) {String[] valueArr = props.split(":");sBuilder.append(valueArr[1]).append(":").append(valueArr[3]).append("<br>");}return sBuilder.toString();}}

VerifyCodeUtils工具类

/*** 生成图片验证码工具类*/
public class VerifyCodeUtils {//使用到Algerian字体,系统里没有的话需要安装字体,字体只显示大写,去掉了1,0,i,o几个容易混淆的字符public static final String VERIFY_CODES = "23456789ABCDEFGHJKLMNPQRSTUVWXYZ";private static Random random = new Random();/*** 使用系统默认字符源生成验证码** @param verifySize 验证码长度* @return*/public static String generateVerifyCode(int verifySize) {return generateVerifyCode(verifySize, VERIFY_CODES);}/*** 使用指定源生成验证码** @param verifySize 验证码长度* @param sources    验证码字符源* @return*/public static String generateVerifyCode(int verifySize, String sources) {if (sources == null || sources.length() == 0) {sources = VERIFY_CODES;}int codesLen = sources.length();Random rand = new Random(System.currentTimeMillis());StringBuilder verifyCode = new StringBuilder(verifySize);for (int i = 0; i < verifySize; i++) {verifyCode.append(sources.charAt(rand.nextInt(codesLen - 1)));}return verifyCode.toString();}/*** 输出指定验证码图片流**/public static void outputImage(int w, int h, OutputStream os, String code) throws IOException {int verifySize = code.length();BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);Random rand = new Random();Graphics2D g2 = image.createGraphics();g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);Color[] colors = new Color[5];Color[] colorSpaces = new Color[]{Color.WHITE, Color.CYAN,Color.GRAY, Color.LIGHT_GRAY, Color.MAGENTA, Color.ORANGE,Color.PINK, Color.YELLOW};float[] fractions = new float[colors.length];for (int i = 0; i < colors.length; i++) {colors[i] = colorSpaces[rand.nextInt(colorSpaces.length)];fractions[i] = rand.nextFloat();}Arrays.sort(fractions);g2.setColor(Color.GRAY);// 设置边框色g2.fillRect(0, 0, w, h);Color c = getRandColor(200, 250);g2.setColor(c);// 设置背景色g2.fillRect(0, 2, w, h - 4);//绘制干扰线Random random = new Random();g2.setColor(getRandColor(160, 200));// 设置线条的颜色for (int i = 0; i < 20; i++) {int x = random.nextInt(w - 1);int y = random.nextInt(h - 1);int xl = random.nextInt(6) + 1;int yl = random.nextInt(12) + 1;g2.drawLine(x, y, x + xl + 40, y + yl + 20);}// 添加噪点float yawpRate = 0.05f;// 噪声率int area = (int) (yawpRate * w * h);for (int i = 0; i < area; i++) {int x = random.nextInt(w);int y = random.nextInt(h);int rgb = getRandomIntColor();image.setRGB(x, y, rgb);}shear(g2, w, h, c);// 使图片扭曲g2.setColor(getRandColor(100, 160));int fontSize = h - 4;Font font = new Font("Algerian", Font.ITALIC, fontSize);g2.setFont(font);char[] chars = code.toCharArray();for (int i = 0; i < verifySize; i++) {AffineTransform affine = new AffineTransform();affine.setToRotation(Math.PI / 4 * rand.nextDouble() * (rand.nextBoolean() ? 1 : -1), (w / verifySize) * i + fontSize / 2, h / 2);g2.setTransform(affine);g2.drawChars(chars, i, 1, ((w - 10) / verifySize) * i + 5, h / 2 + fontSize / 2 - 10);}g2.dispose();ImageIO.write(image, "jpg", os);}private static Color getRandColor(int fc, int bc) {if (fc > 255)fc = 255;if (bc > 255)bc = 255;int r = fc + random.nextInt(bc - fc);int g = fc + random.nextInt(bc - fc);int b = fc + random.nextInt(bc - fc);return new Color(r, g, b);}private static int getRandomIntColor() {int[] rgb = getRandomRgb();int color = 0;for (int c : rgb) {color = color << 8;color = color | c;}return color;}private static int[] getRandomRgb() {int[] rgb = new int[3];for (int i = 0; i < 3; i++) {rgb[i] = random.nextInt(255);}return rgb;}private static void shear(Graphics g, int w1, int h1, Color color) {shearX(g, w1, h1, color);shearY(g, w1, h1, color);}private static void shearX(Graphics g, int w1, int h1, Color color) {int period = random.nextInt(2);boolean borderGap = true;int frames = 1;int phase = random.nextInt(2);for (int i = 0; i < h1; i++) {double d = (double) (period >> 1)* Math.sin((double) i / (double) period+ (6.2831853071795862D * (double) phase)/ (double) frames);g.copyArea(0, i, w1, 1, (int) d, 0);if (borderGap) {g.setColor(color);g.drawLine((int) d, i, 0, i);g.drawLine((int) d + w1, i, w1, i);}}}private static void shearY(Graphics g, int w1, int h1, Color color) {int period = random.nextInt(40) + 10; // 50;boolean borderGap = true;int frames = 20;int phase = 7;for (int i = 0; i < w1; i++) {double d = (double) (period >> 1)* Math.sin((double) i / (double) period+ (6.2831853071795862D * (double) phase)/ (double) frames);g.copyArea(i, 0, 1, h1, 0, (int) d);if (borderGap) {g.setColor(color);g.drawLine(i, (int) d, i, 0);g.drawLine(i, (int) d + h1, i, h1);}}}/*** 获取随机验证码及其加密图片* 用的就是这个生成图片验证码 基于Base64编码进行加密成的String*/public static String VerifyCode(int w, int h, String code){try {//base64编码器BASE64Encoder encoder = new BASE64Encoder();//准备输出流ByteArrayOutputStream data = new ByteArrayOutputStream();//使用code生成w宽 h高的图片,并将结果图片存入data流中outputImage(w, h, data, code);//使用base64编码成String,返回一个String给前端return encoder.encode(data.toByteArray());} catch (IOException e) {e.printStackTrace();/*** 抛出自定义异常* */throw new BusinessException("生成验证码失败!");}}/*** 直接进行测试* 使用StrUtils调用getComplexRandomString 传入参数为需要即为的验证码* 获取随机的数字 即验证码* 通过VertifyCodeUtils调用VerifyCode方法生成一个加密后的图片验证码* */public static void main(String[] args) throws  Exception{String code = StrUtils.getComplexRandomString(4);System.out.println("生成的验证码:"+code); // s4eMSystem.out.println(VerifyCode(100, 30, "1234"));// 这里显示的是 一个没有前缀的String  前缀为data:image/jpeg;base84,}
}

AjaxResult

        将经过base64加密过后的String进行图片展示,前端通过拼接过后进行显示在前端

/*** 增删改的一个工具类,用来封装是否成功和提示信息** */
public class AjaxResult {// 默认是true 和 操作成功private Boolean success = true;private String msg = "操作成功";/*** 定义一个Object类的属性resultObj* 用来封装上传过后storage存储过后返回给前端的文件路径和文件名* 和图形验证码经过Base64加密后的String* */public Object resultObj;public Object getResultObj() {return resultObj;}// 链式语法改造 调用set方法时会返回一个AjaxResult就可以继续进行调用方法// 参数resultObj就是storage存储过后返回给前端的文件路径和文件名public AjaxResult setResultObj(Object resultObj) {this.resultObj = resultObj;return this;}public AjaxResult() {}/*** 单例模式* 1 构造方法私有化* 2 内部提供一个可供外界调用的一个静态方法* 使用单例模式进行创建AjaxResult对象 每次调用都是用的这个对象* 在类加载的时候就创建好了对象 单例的饿汉模式*/private static AjaxResult a = new AjaxResult();/*** 1构造方法私有化*//* private AjaxResult() {};*//*** 2 内部提供一个可供外界调用的一个静态方法* // this指代的是当前AjaxResult类的对象a   this = a*/public static AjaxResult getAjaxResult() {return a;}/*** 链式编程思维*/public AjaxResult setSuccess(Boolean success) {this.success = success;// 当前操作的是谁返回给谁// this指代的是当前AjaxResult类的对象areturn this;}public AjaxResult setMsg(String msg) {this.msg = msg;// 当前操作的是谁返回给谁return this;}public AjaxResult(Boolean success, String msg) {this.success = success;this.msg = msg;}public Boolean getSuccess() { //  get方法 前端获取bean属性return success;}public String getMsg() {return msg;}
}


http://chatgpt.dhexx.cn/article/qqfdWE4k.shtml

相关文章

图片验证码实现的几种方式

一、Google Kaptcha 1、简介 kaptcha 是一个非常实用的验证码生成工具。有了它&#xff0c;你可以生成各种样式的验证码&#xff0c;因为它是可配置的。kaptcha工作的原理是调用 com.google.code.kaptcha.servlet.KaptchaServlet&#xff0c;生成一个图片。同时将生成的验证码…

ajax请求后台返回数据

功能介绍&#xff1a;最近学习在做一个新闻管理系统&#xff0c;其中有一个模块做的是一个排行榜功能&#xff0c;分为东部联盟和西部联盟&#xff0c;当我点击他的时候&#xff0c;排行的数据会发生变化。由于这一块怎么整个页面中的一小块&#xff0c;所以使用的是局部刷新页…

如何取消ajax请求

之前在面试的时候&#xff0c;被面试官问到了如何取消ajax请求&#xff0c;然鹅并不知道&#xff0c;被刷之后痛定思痛&#xff0c;总结了原生XHR、jquery、axios取消ajax请求的方法。 原生XHR 对于原生XHR对象来说&#xff0c;取消的ajax的关键是调用XHR对象的.abort()方法 …

HTML AJAX请求调用

html ajax请求 ------------------温故而知新,可以装逼矣-------------------- 大佬提出需求&#xff0c;访问html文件&#xff0c;动态响应文章内容 作为一个纯正&#xff08;半吊子&#xff09;的JAVA后端写太多前后端分离的项目&#xff0c;太久没接触前端知识。头都是炸的…

ajax请求五个步骤!

ajax请求五个步骤&#xff01; 今天 咱们讲一讲Ajax请求五个步骤&#xff01; 1.创建XMLHttpRequest异步对象 var xhr; if (window.XMLHttpRequest){// code for IE7, Firefox, Chrome, Opera, Safarixhrnew XMLHttpRequest();} else{// code for IE6, IE5xhrnew ActiveXObje…

jquery(五)Ajax请求

在jQuery中AJAX的写法有3种&#xff0c;$ajax&#xff0c;$post&#xff0c;$get这三种。其中$post和$get是简易写法&#xff0c;高层的实现&#xff0c;在调用他们的时候&#xff0c;会运行底层封装好的$ajax。 ajax写法 $.ajax({url:"http://www.microsoft.com",…

Ajax请求参数

Ajax请求参数 GET请求参数的传递创建服务器通过表单访问服务器 POST请求参数的传递application/x-www-form-urlencoded参数的接收与发送创建服务器通过表单访问服务器 JSON格式数据的发送与接收创建服务器通过表单访问服务器 GET请求参数的传递 设置open()方法中的第1个参数为…

Ajax请求以及发送Ajax请求的方式

1.在写C语言时候&#xff0c;写了好长时间&#xff0c;终于开始运行了&#xff0c;结果出现了这种情况&#xff0c;以前就没见过。 原来是自己的函数名写错了&#xff0c;main写成了amin.所以写代码一定要细心&#xff0c;不然会给你带来很失望的心情。 2.Ajax请求步骤 //1.创…

html的ajax请求

页面中ajax发起请求&#xff0c;controller接收数据并处理 这次ajax的测试以访问html的方式配置 配置yml文件&#xff1a; 然后在 src/main/webapp下创建一个html页面 创建controller&#xff1a; 运行项目就可以访问到index.html&#xff1a; 引入jquery&#xff1a; 修改…

Ajax请求详解

Ajax请求 一、什么是Ajax二、Ajax原理是什么三、Ajax的使用1.创建Ajax核心对象XMLHttpRequest(记得考虑兼容性)2.向服务器发送请求3.服务器响应处理&#xff08;区分同步跟异步两种情况&#xff09;①同步处理②异步处理③GET和POST请求数据区别 四、结束语五、jQuery的ajax函数…

Ajax原理一篇就够了

前言 AJAX即“Asynchronous Javascript And XML”,是指一种创建交互式网页应用的网页开发技术。AJAX 是一种用于创建快速动态网页的技术。它可以令开发者只向服务器获取数据(而不是图片,HTML文档等资源),互联网资源的传输变得前所未有的轻量级和纯粹,这激发了广大开发者的…

Ajax请求

ajax请求简介&#xff1a; ajax是利用前端的技术&#xff0c;向服务器发送一个异步请求 原有的请求在请求后需要刷新整张页面&#xff0c;但是用ajax请求后只需要刷新一部分页面即可&#xff1b; xhr基础属性&#xff1a; Jquery与ajax&#xff1a;使用$.ajax({}) Ajax的html…

Ajax请求的五个步骤

目录 Ajax请求的五个步骤 一、定义 1、什么是Ajax 2、同步与异步的区别 3、ajax的工作原理 二、实现AJAX的基本步骤 1、创建XMLHttpRequest对象 2、创建HTTP请求 3、设置响应HTTP请求状态变化的函数 4、设置获取服务器返回数据的语句 5、发送HTTP请求 6、局部更新 …

详解Ajax请求

目录 1.$.get()函数的语法 2.$.get()发起不带参数的请求 3.$.get()发起带参数的请求 4.$.post()函数的语法 5.$.post()向服务器提交数据 6.$.ajax()函数的语法 7.使用$.ajax()发起GET请求 ​8.使用$.ajax()发起POST请求 1.$.get()函数的语法 jQuery中$.get()函数的功能单…

linux下输入法,中英文自由切换,仍无法输入中文

1、安装中文输入法&#xff1a;yum install "Chinese Support" 2、中英文切换&#xff1a;Super空格 &#xff08;其中Super键在Ctrl和Alt之间&#xff09; 也可以直接点击右上角进行切换 3、设置输入源&#xff1a;如果直接选择汉语的话&#xff0c;即使中英文可以…

Linux配置中文输入法图文教程

原来的设置中&#xff0c;从中文切换到英文或者从英文切换为中文&#xff08;跨语言&#xff09;时&#xff0c;总要使用不同的快捷键。在这个教程中&#xff0c;我们将使用controlshift来完成所有输入法的切换&#xff08;无论是否跨语言&#xff09;。 首先在键盘设置中&…

「 Linux 」“安装中文输入法方法”讲解

一、前言 每重装一次Linux系统都会被中文输入法打扰一下&#xff0c;把安装方法记录下来吧&#xff0c;方便操作。 二、技术实现 1. 点击屏幕右上角的螺丝口&#xff0c;选择“System Settings”&#xff1b; 2. 点击“Language suport”&#xff1a; 3. 弹出语言支持安装窗口…

【Linux】安装中文输入法

目录 1.安装中文输入2.设置输入法快捷键切换 1.安装中文输入 命令&#xff1a;sudo apt-get install ibus-pinyin 输入命令后重启一下&#xff1a;reboot 添加之后就可以在右上角选择了 2.设置输入法快捷键切换 右上角进入设置界面 默认的切换快捷键&#xff1a; 鼠标单击…

Linux不能输入中文

Linux不能输入中文 文章目录 Linux不能输入中文1 目标2 环境3 修改配置3.1 服务器编码配置 1 目标 本文针对的问题是“Ubuntu 安装中文语言包”“Ubuntu Server中文问题”&#xff0c;“Ubuntu更改语言环境”&#xff0c;“Ubuntu locale的设定”&#xff0c;“cannot change …

Linux之中文输入

打开linux系统的应用程序选择系统工具点击设置选择区域和语言窗口 添加汉语中文输入源 使用win空格实现切换&#xff1b;或shift。