news 2026/4/15 0:21:53

算法题 翻转图像

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
算法题 翻转图像

832. 翻转图像

问题描述

给定一个n x n的二进制矩阵image,对其进行水平翻转后再对每个元素进行反转(0变1,1变0)。

水平翻转:将每一行的元素顺序颠倒
反转:将每个 0 变为 1,每个 1 变为 0

要求原地修改矩阵并返回结果。

示例

输入:image=[[1,1,0],[1,0,1],[0,0,0]]输出:[[1,0,0],[0,1,0],[1,1,1]]解释:-首先水平翻转每一行:[[0,1,1],[1,0,1],[0,0,0]]-然后反转每个元素:[[1,0,0],[0,1,0],[1,1,1]]
输入:image=[[1,1,0,0],[1,0,0,1],[0,1,1,1],[1,0,1,0]]输出:[[1,1,0,0],[0,1,1,0],[0,0,0,1],[1,0,1,0]]

算法思路

双指针

  1. 核心

    • 水平翻转 + 反转可以同时进行
    • 对于每一行,使用双指针从两端向中间移动
    • 交换两个位置的元素时,同时进行反转操作
  2. 反转

    • 使用异或操作:value ^ 1可以实现 0↔1 的翻转
    • 或者使用1 - value

代码实现

方法一:双指针

classSolution{/** * 翻转图像:水平翻转每一行,然后反转每个元素(0变1,1变0) * 使用双指针原地操作,时间复杂度O(n²),空间复杂度O(1) * * @param image n x n 的二进制矩阵 * @return 翻转后的图像 */publicint[][]flipAndInvertImage(int[][]image){intn=image.length;// 遍历每一行for(inti=0;i<n;i++){intleft=0;// 左指针intright=n-1;// 右指针// 双指针向中间移动while(left<right){// 交换左右元素并同时反转// 使用异或操作进行反转:0^1=1, 1^1=0inttemp=image[i][left]^1;image[i][left]=image[i][right]^1;image[i][right]=temp;left++;right--;}// 如果行长度为奇数,处理中间元素(只需反转)if(left==right){image[i][left]^=1;}}returnimage;}}

方法二:分步

classSolution{/** * 分步实现:先水平翻转,再反转每个元素 * * @param image 输入图像 * @return 翻转后的图像 */publicint[][]flipAndInvertImage(int[][]image){intn=image.length;// 水平翻转每一行for(inti=0;i<n;i++){intleft=0;intright=n-1;while(left<right){// 交换元素inttemp=image[i][left];image[i][left]=image[i][right];image[i][right]=temp;left++;right--;}}// 反转每个元素for(inti=0;i<n;i++){for(intj=0;j<n;j++){image[i][j]^=1;// 或者 image[i][j] = 1 - image[i][j];}}returnimage;}}

方法三:使用额外空间

classSolution{/** * 使用额外空间 * * @param image 输入图像 * @return 翻转后的图像 */publicint[][]flipAndInvertImage(int[][]image){intn=image.length;int[][]result=newint[n][n];for(inti=0;i<n;i++){for(intj=0;j<n;j++){// result[i][j] = 反转(image[i][n-1-j])result[i][j]=image[i][n-1-j]^1;}}returnresult;}}

算法分析

  • 时间复杂度:O(n²)
    • 需要处理矩阵中的每个元素一次
    • 双指针中每个元素只被访问一次
  • 空间复杂度:O(1)
    • 原地修改,只使用常数额外空间

算法过程

输入:image = [[1,1,0],[1,0,1],[0,0,0]]

方法一:

处理第0行[1,1,0]

  • left=0, right=2:交换image[0][0]image[0][2]并反转
    • temp = 1^1 = 0
    • image[0][0] = 0^1 = 1
    • image[0][2] = 0
    • 行变为[1,1,0]
  • left=1, right=1:中间元素,image[0][1] ^= 11^1 = 0
  • 最终第0行:[1,0,0]

处理第1行[1,0,1]

  • left=0, right=2:交换并反转
    • temp = 1^1 = 0
    • image[1][0] = 1^1 = 0
    • image[1][2] = 0
    • 行变为[0,0,0]
  • left=1, right=1:中间元素,image[1][1] ^= 10^1 = 1
  • 最终第1行:[0,1,0]

处理第2行[0,0,0]

  • left=0, right=2:交换并反转
    • temp = 0^1 = 1
    • image[2][0] = 0^1 = 1
    • image[2][2] = 1
    • 行变为[1,0,1]
  • left=1, right=1:中间元素,image[2][1] ^= 10^1 = 1
  • 最终第2行:[1,1,1]

最终结果:

[[1,0,0],[0,1,0],[1,1,1]]

测试用例

importjava.util.Arrays;publicclassMain{publicstaticvoidmain(String[]args){Solutionsolution=newSolution();// 测试用例1:标准示例int[][]image1={{1,1,0},{1,0,1},{0,0,0}};int[][]result1=solution.flipAndInvertImage(image1);System.out.println("Test 1: "+Arrays.deepToString(result1));// [[1,0,0],[0,1,0],[1,1,1]]// 测试用例2:4x4矩阵int[][]image2={{1,1,0,0},{1,0,0,1},{0,1,1,1},{1,0,1,0}};int[][]result2=solution.flipAndInvertImage(image2);System.out.println("Test 2: "+Arrays.deepToString(result2));// [[1,1,0,0],[0,1,1,0],[0,0,0,1],[1,0,1,0]]// 测试用例3:全1矩阵int[][]image3={{1,1,1},{1,1,1},{1,1,1}};int[][]result3=solution.flipAndInvertImage(image3);System.out.println("Test 3: "+Arrays.deepToString(result3));// [[0,0,0],[0,0,0],[0,0,0]]// 测试用例4:全0矩阵int[][]image4={{0,0,0},{0,0,0},{0,0,0}};int[][]result4=solution.flipAndInvertImage(image4);System.out.println("Test 4: "+Arrays.deepToString(result4));// [[1,1,1],[1,1,1],[1,1,1]]// 测试用例5:单元素矩阵int[][]image5={{1}};int[][]result5=solution.flipAndInvertImage(image5);System.out.println("Test 5: "+Arrays.deepToString(result5));// [[0]]int[][]image6={{0}};int[][]result6=solution.flipAndInvertImage(image6);System.out.println("Test 6: "+Arrays.deepToString(result6));// [[1]]// 测试用例6:2x2矩阵int[][]image7={{1,0},{0,1}};int[][]result7=solution.flipAndInvertImage(image7);System.out.println("Test 7: "+Arrays.deepToString(result7));// [[1,0],[0,1]]}}

关键点

  1. 原地操作

    • 双指针在交换的同时进行反转,避免了两次遍历
    • 异或操作^ 1是最简洁的反转方式
  2. 奇偶长度

    • 偶数长度:所有元素都通过交换处理
    • 奇数长度:中间元素单独处理(只需反转,无需交换)
  3. 位运算

    • value ^ 11 - value更高效
  4. 边界情况

    • 1x1 矩阵:直接反转唯一元素
    • 全0或全1矩阵:结果分别为全1或全0

常见问题

  1. 异或操作^ 1为什么能实现反转?
    在二进制中,01=1,11=0,正好实现了0和1的互换。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/12 10:24:37

揭秘Open-AutoGLM部署全流程:手把手教你避坑并实现一键部署

第一章&#xff1a;Open-AutoGLM部署概述Open-AutoGLM 是一个开源的自动化通用语言模型管理框架&#xff0c;旨在简化大语言模型的本地化部署、服务调度与推理优化。该框架支持多种主流模型格式&#xff0c;并提供模块化的插件体系&#xff0c;便于开发者根据实际需求进行功能扩…

作者头像 李华
网站建设 2026/4/15 16:17:56

n8n工作流自动化平台:从零开始构建智能自动化流程的完整指南

在当今数字化转型的时代&#xff0c;自动化已经成为企业和个人提升效率的关键。n8n作为一款开源的工作流自动化平台&#xff0c;为初学者提供了一个直观且强大的工具&#xff0c;帮助您轻松构建复杂的自动化流程。无论是简单的文件处理、数据同步&#xff0c;还是复杂的AI驱动决…

作者头像 李华
网站建设 2026/4/13 9:41:51

通信系统中滤波器的模拟电子技术实现:操作指南

通信系统中的模拟滤波器设计实战&#xff1a;从原理到PCB落地在高速无线通信时代&#xff0c;我们每天都在与看不见的电磁波打交道。无论是5G手机、Wi-Fi路由器&#xff0c;还是卫星接收终端&#xff0c;它们背后都离不开一个看似低调却至关重要的角色——模拟滤波器。你有没有…

作者头像 李华
网站建设 2026/4/15 20:17:59

Android Root权限获取全攻略:APatch快速配置指南

想要在Android设备上获得完整Root权限&#xff0c;却担心操作复杂容易出错&#xff1f;今天让我们一起来探索APatch这个强大的Android内核修补工具&#xff0c;它将为你打开一扇通往系统权限管理的新大门。APatch巧妙融合了Magisk的便捷安装方式和KernelSU的强大内核修补能力&a…

作者头像 李华
网站建设 2026/4/14 8:28:07

Tesseract OCR语言训练数据:让图片中的文字“开口说话“的魔法词典

想象一下&#xff0c;你有一本神奇的词典&#xff0c;能让任何图片中的文字自动"开口说话"——这就是Tesseract OCR语言训练数据的魔力所在。今天&#xff0c;就让我带你走进这个充满魔力的世界&#xff0c;看看如何用最简单的方法让计算机读懂图片中的文字。 【免费…

作者头像 李华