leetcode网格题目中的上下左右四个方向和旋转90度的数量关系的代码表示

本文主要是介绍leetcode网格题目中的上下左右四个方向和旋转90度的数量关系的代码表示,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 总结

1.1 方向数组的二维表示方法

  1. 定义方向数组(一般设定的方向比较灵活,比如(-1,0), (1,0)可以表示向上和向下移动也可以相反,看题目需要改成最易懂的样子即可):可以使用二维数组 dirs 来表示四个方向(上、右、下、左),例如:
    int[][] dirs = {{-1, 0}, {0, 1}, {1, 0}, {0, -1}};
    
    这里,{-1, 0} 表示向上移动,{0, 1} 表示向右移动,{1, 0} 表示向下移动,{0, -1} 表示向左移动。

1.2 方向数组的一维表示方法

  1. 一维数组表示:另一种方法是使用一维数组结合模运算来表示方向。例如,可以定义一个一维数组 dxdy,分别表示 x 轴和 y 轴的变化:
    int[] dx = {-1, 0, 1, 0};
    int[] dy = {0, 1, 0, -1};
    
    这里的数组同样表示上、右、下、左四个方向。

1.3 旋转的处理

  1. 初始化方向:假设机器人最初面向北方,可以用一个变量 d 来表示当前方向,初始设为 1(表示向右)。

  2. 右旋转 90 度:向右旋转意味着 d 的值增加 1。可以使用模运算来确保 d 的值不超过方向数组的长度:

    d = (d + 1) % 4;
    
  3. 左旋转 90 度:向左旋转意味着 d 的值减少 1。需要处理负数的情况,以确保 d 仍然是有效的索引:

    d = (d + 3) % 4;  // 或 d = (d - 1 + 4) % 4;
    
  4. 旋转 180 度:旋转 180 度意味着 d 的值增加或减少 2:

    d = (d + 2) % 4;
    

1.4 应用示例

  • 结合示例:您可以提供一个简单的例子来说明如何使用这些方向数组和旋转操作来控制机器人在二维网格上的移动。

  • 图解辅助:辅以图表来说明方向的改变,会使读者更容易理解。

这篇文章不仅对理解相关的编程题目有帮助,而且也能够加深对方向控制和数组操作的理解。

2 岛屿系列问题

利用1.1和1.2中的行走表示法就可以解决

3 旋转系列问题

3.1 第一题

3.1.1 LC1041. 困于环中的机器人

class Solution {public boolean isRobotBounded(String instructions) {int[][] direc = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};int direcIndex = 0;int x = 0, y = 0;int n = instructions.length();for (int idx = 0; idx < n; idx++) {char instruction = instructions.charAt(idx);if (instruction == 'G') {x += direc[direcIndex][0];y += direc[direcIndex][1];} else if (instruction == 'L') {direcIndex += 3;direcIndex %= 4;} else {direcIndex++;direcIndex %= 4;}}return direcIndex != 0 || (x == 0 && y == 0);}
}作者:力扣官方题解
链接:https://leetcode.cn/problems/robot-bounded-in-circle/solutions/2217873/kun-yu-huan-zhong-de-ji-qi-ren-by-leetco-kjya/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

3.1.2 改编题:LC1041. 困于环中的机器人: 原题中的一条字符串指令是原子花执行完之后看是否有环,但是现在认为"每一个instruction指令不再原子化执行,而是每一个instruction中的字符原子化执行", 请问在无限执行instruction中,是否会碰到环?

class Solution {public boolean isRobotBounded(String instructions) {HashMap<Integer,Set<Integer>>mp=new HashMap<>();mp.put(0,new HashSet<Integer>());mp.get(0).add(0);int x=0,y=0,d=1;int[][]dirs=new int[][]{{-1,0},{0,1},{1,0},{0,-1}};int n=instructions.length()*4;StringBuilder sb=new StringBuilder();for(int i=0;i<4;i++){sb.append(instructions);}String ni=sb.toString();for(int i=0;i<n;i++){char c=ni.charAt(i);if(c=='L'){d=(d+3)%4;}else if(c=='R'){d=(d+1)%4;}else{x+=dirs[d][0];y+=dirs[d][1];if(mp.containsKey(x)){if(mp.get(x).contains(y)){System.out.println("i:"+i+", x:"+x+", y:"+y);return true;}else{mp.get(x).add(y);}}else{mp.put(x,new HashSet<Integer>());mp.get(x).add(y);}}}return false;}
}

3.2 LC874. 模拟行走机器人

3.2.1 原题答案

1 代码
    public int robotSim(int[] commands, int[][] obstacles) {int[][]dirs=new int[][]{{-1,0},{0,1},{1,0},{0,-1}};int x=0,y=0,d=1;HashSet<Integer>set=new HashSet<>();for(int i=0;i<obstacles.length;i++){set.add(obstacles[i][0]*60002+obstacles[i][1]);}int ans=0;for(int i=0;i<commands.length;i++){if(commands[i]<0){if(commands[i]==-2){d=(d+3)%4;}else if(commands[i]==-1){d=(d+1)%4;}}else{for(int j=0;j<commands[i];j++){if(set.contains((x+dirs[d][0])*60002+y+dirs[d][1]))break;x+=dirs[d][0];y+=dirs[d][1];ans=Math.max(ans,x*x+y*y);}}}return ans;}
2 Q1: “set.add(obstacles[i][0]*60002+obstacles[i][1]);中, 60001 怎么来的?10001就不行“

obstacle的坐标范围是正负3w。如果10001, (1, 0)和(0, 10001)就会撞到一起,但是如果乘以一个比6w严格大的数,那么每一个二维坐标被一个一维度的id标识时能保证唯一性

3 Q1: 机器人如何以最快的速度知道自己能走的最大步子, 而不是走一步看一步?

3.2.2 在3.2.1中规定机器人一次性走的步子最大是9, 那么可以把所有的障碍点都存到set中,然后枚举每一步,查看是否撞倒了障碍点,这样复杂度还是不高,假如现在机器人每一次走的最大步子是1000,如何让机器人以最小的复杂度知道自己可以向前走多少步呢?

1 思路一:可以用两个二维set存储所有的坐标,一个存储所有横轴上的障碍物,另一个存储纵轴,遍历当前方向上的哈希表,判断是否有在[from, to]的中间值,有则直接截断
class Solution {
private:// 1右 -1左vector<pair<int, int>> MOVE = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};public:int robotSim(vector<int>& commands, vector<vector<int>>& obstacles) {unordered_map<int, set<int>> row;unordered_map<int, set<int>> col;for (auto&& arr : obstacles) {col[arr[0]].insert(arr[1]);row[arr[1]].insert(arr[0]);}int ans = 0;// 起始方向是北int pos = 0;int x = 0, y = 0;for (int num : commands) {// 向右if (num == -1) {pos = (pos + 1) % 4;} // 向左else if (num == -2) {pos = (pos - 1 + 4) % 4;} // moveelse {auto [movX, movY] = MOVE[pos];                // set的参数分分清除// y轴方向移动if (movY) {y = is_between(col[x], y, y + num * movY, movY);} // x轴方向移动else {x = is_between(row[y], x, x + num * movX, movX);}}ans = max(ans, x * x + y * y);}return ans;}int is_between(set<int>& st, int from, int to, int flag = 1) {// lower 和 upper 的使用注意// 正向if (flag > 0) {auto it = st.upper_bound(from);if (it == st.end()) {return to;} else if (*it > to) {return to;} else {return *it - 1;}} // 逆向else {if (st.size() == 0) {return to;}auto it = st.lower_bound(from);it--;if (it == st.end()) {return to;} else if (*it > from) {return to;} else if (*it < to) {return to;} else {return *it + 1;}}}
};
2 思路二:(通过了100%的case)

可以用两个二维treeset存储所有的坐标,一个存储所有横轴上的有序障碍物,另一个存储纵轴,利用二分查找前方距离自己最近的障碍物,看看自己当前可以最多向前移动多少步。

二分查找时需要注意floor,ceil,lower和higher这几个函数的区别:

在使用 Java 的 TreeMapTreeSet 进行二分查找时,确实需要注意 floor, ceil, lowerhigher 这几个函数的区别。它们都是用来在有序集合中查找特定元素的,但行为略有不同:

对于 TreeMap

  • floorKey(K key): 返回小于或等于给定键的最大键。
  • ceilingKey(K key): 返回大于或等于给定键的最小键。
  • lowerKey(K key): 返回严格小于给定键的最大键。
  • higherKey(K key): 返回严格大于给定键的最小键。

对于 TreeSet

  • floor(E e): 返回小于或等于给定元素的最大元素。
  • ceiling(E e): 返回大于或等于给定元素的最小元素。
  • lower(E e): 返回严格小于给定元素的最大元素。
  • higher(E e): 返回严格大于给定元素的最小元素。

注意事项:

  1. 等于情况的处理floorceiling 方法包含等于的情况,而 lowerhigher 方法则不包括等于的情况。

  2. 返回值的类型:在 TreeMap 中,floorKey, ceilingKey, lowerKey, 和 higherKey 返回键的值;而在 TreeSet 中,floor, ceiling, lower, 和 higher 返回元素本身。

  3. 使用场景:根据您的查找需求(是否包括等于的情况,查找的是键还是元素),选择合适的方法。

在进行二分查找或者需要快速找到最接近的元素时,合理利用这些方法可以大大提高效率和代码的可读性。

代码:
import java.util.*;class Solution {public int robotSim(int[] commands, int[][] obstacles) {// 横轴和纵轴上的有序障碍物集合TreeMap<Integer, TreeSet<Integer>> rowMap = new TreeMap<>();TreeMap<Integer, TreeSet<Integer>> colMap = new TreeMap<>();// 初始化障碍物集合for (int[] obstacle : obstacles) {int x = obstacle[0], y = obstacle[1];rowMap.putIfAbsent(x, new TreeSet<>());colMap.putIfAbsent(y, new TreeSet<>());rowMap.get(x).add(y);colMap.get(y).add(x);}int x = 0, y = 0, direction = 0; // 初始坐标和方向(北)int[][] moves = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}}; // 北、东、南、西int maxDistance = 0;for (int cmd : commands) {if (cmd == -1) { // 右转direction = (direction + 1) % 4;} else if (cmd == -2) { // 左转direction = (direction + 3) % 4;} else { // 移动int dx = moves[direction][0];int dy = moves[direction][1];if (dx != 0) {x = getNextPosition(colMap, x, y, dx, cmd);} else {y = getNextPosition(rowMap, y, x, dy, cmd);}maxDistance = Math.max(maxDistance, x * x + y * y);}}return maxDistance;}private int getNextPosition(TreeMap<Integer, TreeSet<Integer>> map, int pos, int fixedPos, int delta, int steps) {TreeSet<Integer> set = map.getOrDefault(fixedPos, new TreeSet<>());if (delta > 0) {Integer higher = set.higher(pos);if (higher != null) {return Math.min(pos + steps, higher - 1);}} else {Integer lower = set.lower(pos);if (lower != null) {return Math.max(pos - steps, lower + 1);}}return pos + steps * delta;}
}

这篇关于leetcode网格题目中的上下左右四个方向和旋转90度的数量关系的代码表示的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/586156

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

jupyter代码块没有运行图标的解决方案

《jupyter代码块没有运行图标的解决方案》:本文主要介绍jupyter代码块没有运行图标的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录jupyter代码块没有运行图标的解决1.找到Jupyter notebook的系统配置文件2.这时候一般会搜索到

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La