【实战】算法思路总结

面试过程中,总是被拷打,信心都要没了。但是也慢慢摸索出一些思路,希望对大家有帮助。

(需要多用一下ACM模式,力扣模式提供好了模板,自己在IDEA里面写的话,还是会有些陌生)

0、基本Java类型

1、用双指针思路去解决链表问题

定义一个单链表

class ListNode{

        int val;

        ListNode next;

        ListNode(int x){

                val = x;

                next = null;

        }

}

力扣21 - 合并两个有序链表

/**

public class ListNode {

        int val;

        ListNode next;

        ListNode() {

        }

        ListNode(int val) {

                 this.val = val;

        }

        ListNode(int val, ListNode next) {

                 this.val = val; this.next = next;

        }

}

 */

class Solution {

    public ListNode mergeTwoLists(ListNode list1, ListNode list2) {

        //虚拟头结点

        ListNode dummy = new ListNode(-1), p = dummy;

        ListNode p1 = list1 ,p2 = list2;

        while(p1 != null && p2 != null){

                //比较p1和p2两个指针,把值较小的节点接到p指针

            if( p1.val > p2.val){

            p.next = p2;

            p2 = p2.next;

        }else{

            p.next = p1;

            p1 = p1.next;

        }

        p = p.next;

        }

        if(p1 != null){

            p.next = p1;

        }

        if(p2 != null){

            p.next = p2;

        }

        return dummy.next;

    }

}

力扣19 - 删除倒数第k个节点

/**

 * Definition for singly-linked list.

 * public class ListNode {

 *     int val;

 *     ListNode next;

 *     ListNode() {}

 *     ListNode(int val) { this.val = val; }

 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }

 * }

 */

class Solution {

    public ListNode removeNthFromEnd(ListNode head, int n) {

        ListNode dummy = new ListNode(-1);

        dummy.next = head;

        ListNode x = findFromEnd(dummy, n+1);

        x.next = x.next.next;

        return dummy.next;

    }

    private ListNode findFromEnd(ListNode head, int k) {

        ListNode p1 = head;

        for (int i = 0; i< k; i++){

            p1 = p1.next;

        }

        ListNode p2 = head;

        while( p1 != null){

            p2 = p2.next;

            p1 = p1.next;

        }

        return p2;

    }

}

这段代码的目的是从单向链表中移除倒数第 n 个节点。为了实现这一点,代码使用了一个辅助的虚拟头节点(dummy node)和双指针方法。以下是详细的解释:

链表节点类的定义

首先是链表节点的定义,类名为 ListNode

public class ListNode {

        int val;

        ListNode next; ListNode() {} ListNode(int val) { this.val = val; }

        ListNode(int val, ListNode next) { this.val = val; this.next = next; }

}

removeNthFromEnd 方法的解释

这是移除倒数第 n 个节点的方法:

class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        ListNode dummy = new ListNode(-1);
        dummy.next = head;

        ListNode x = findFromEnd(dummy, n + 1);

        x.next = x.next.next;

        return dummy.next;
    }

    private ListNode findFromEnd(ListNode head, int k) {
        ListNode p1 = head;

        for (int i = 0; i < k; i++) {
            p1 = p1.next;
        }

        ListNode p2 = head;

        while (p1 != null) {
            p2 = p2.next;
            p1 = p1.next;
        }

        return p2;
    }
}
 

具体思路

1. 使用虚拟头节点
ListNode dummy = new ListNode(-1);
dummy.next = head;
2. 找到倒数第 n+1 个节点
ListNode x = findFromEnd(dummy, n + 1);

调用 findFromEnd 方法,找到倒数第 n+1 个节点。之所以要找到倒数第 n+1 个节点,是为了方便我们进行节点删除操作,因为我们需要操作的是前一个节点的 next 指针。

3. 删除节点
x.next = x.next.next;

将倒数第 n+1 个节点的 next 指针指向其下下个节点,从而删除倒数第 n 个节点。

4. 返回结果
return dummy.next;

返回虚拟头节点的 next 指针,即新的链表头。

findFromEnd 方法的解释

这个方法的目的是找到从链表末尾数起的第 k 个节点。

private ListNode findFromEnd(ListNode head, int k) {
    ListNode p1 = head;

    for (int i = 0; i < k; i++) {
        p1 = p1.next;
    }

    ListNode p2 = head;

    while (p1 != null) {
        p2 = p2.next;
        p1 = p1.next;
    }

    return p2;
}

1. 初始化指针

ListNode p1 = head;

首先初始化指针 p1 指向链表头节点。

2. 移动 p1 指针

for (int i = 0; i < k; i++) { p1 = p1.next; }

p1 指针向前移动 k 步。

3. 初始化 p2 指针

ListNode p2 = head;

初始化另一个指针 p2 指向链表头节点。

4. 同步移动 p1p2 指针

while (p1 != null) {

   p2 = p2.next;

   p1 = p1.next;

}

同步移动 p1p2 指针,直到 p1 到达链表末尾。此时 p2 指向的就是从末尾数起的第 k 个节点。

5. 返回 p2 指针

return p2;

返回 p2 指针,即倒数第 k 个节点。

总结

这段代码通过创建一个虚拟头节点,并使用双指针方法来找到并删除倒数第 n 个节点。反复利用 findFromEnd 方法来找到关键节点,并调整链表指针以删除目标节点。这种方法有效地解决了移除链表倒数第 n 个节点的问题,同时简化了边界条件的处理。

力扣876 - 链表的中间结点

/**

 * Definition for singly-linked list.

 * public class ListNode {

 *     int val;

 *     ListNode next;

 *     ListNode() {}

 *     ListNode(int val) { this.val = val; }

 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }

 * }

 */

class Solution {

    public ListNode middleNode(ListNode head) {

        ListNode slow = head;

        ListNode fast = head;

        while(fast != null && fast.next != null){

            slow = slow.next;

            fast = fast.next.next;

        }

        return slow;

    }

}

力扣142 - 环形链表

class Solution {
    public ListNode detectCycle(ListNode head) {
        ListNode fast, slow;
        fast = slow = head;
        while (fast != null && fast.next != null) {
            fast = fast.next.next;
            slow = slow.next;
            if (fast == slow) break;
        }
        // 上面的代码类似 hasCycle 函数
        if (fast == null || fast.next == null) {
            // fast 遇到空指针说明没有环
            return null;
        }

        // 重新指向头结点
        slow = head;
        // 快慢指针同步前进,相交点就是环起点
        while (slow != fast) {
            fast = fast.next;
            slow = slow.next;
        }
        return slow;
    }
}
 

160 - 相交链表

/**

 * Definition for singly-linked list.

 * public class ListNode {

 *     int val;

 *     ListNode next;

 *     ListNode(int x) {

 *         val = x;

 *         next = null;

 *     }

 * }

 */

public class Solution {

    public ListNode getIntersectionNode(ListNode headA, ListNode headB) {

        ListNode p1 = headA, p2 = headB;

        while(p1 != p2){

            if(p1 == null) p1 = headB;

            else p1 = p1.next;

            if(p2 ==null) p2 = headA;

            else p2 = p2.next;

        }

        return p2;

       

    }

}

2、双指针解决数组问题

class Solution {

    public int removeDuplicates(int[] nums) {

        int slow = 0, fast = 0;

        while(fast < nums.length){

            if(nums[slow] != nums[fast]){

                slow++;

                nums[slow] = nums[fast];

            }

            fast++;

        }

        return slow+1;

    }

}

class Solution {

    public int removeElement(int[] nums, int val) {

        int readIndex = 0;

        int writeIndex = 0;

        while(readIndex < nums.length){

            if(nums[readIndex] != val){

                nums[writeIndex] = nums[readIndex];

                writeIndex++;

            }

            readIndex++;

        }

        return writeIndex;

    }

}

未完待续。。。。

### CCS烧录代码失败的原因分析 在使用TI毫米波雷达开发过程中,遇到`cannot find file "libsleep_xwr68xx.aer4f"`的报错可能源于以下几个方面: #### 1. 文件路径配置不正确 如果工程中缺少指定库文件 `libsleep_xwr68xx.aer4f` 或者其路径未被正确定义,则可能导致此问题。通常情况下,该文件应位于项目的链接器设置中的库目录下[^1]。 #### 2. 工程依赖缺失 某些必要的源文件或头文件未能正确添加到项目中也可能引发类似的错误。例如,在另一个案例中提到,“DSP2833x_MemCopy.c” 这样的核心组件如果没有加入工程则会触发编译阶段的问题[^2]。 #### 3. 构建工具链异常 构建脚本执行不当或者相关参数设定有误同样会影响最终生成物的质量。比如通过命令行调用 mkimage 实用程序时若选项不符合预期目标架构的要求就会造成不可预见的结果[^3]。 --- ### 解决方案 针对上述几种可能性提供如下建议措施来修复CCS烧录过程中的这些障碍: #### 配置正确的库文件位置 确认所有必需的静态/动态链接库均已存在于当前工作区内的适当子文件夹之中,并且它们已被纳入至Linker Settings下的Library Files列表里。 ```plaintext Project -> Properties -> C/C++ Build -> Settings -> TI Linker -> Library files ``` #### 添加遗漏的关键模块 仔细核查是否有任何重要的实现单元尚未导入工程项目结构之内。特别是像"DSP2833x_MemCopy.c"这样的基础功能部件应当显式声明并关联起来以便顺利完成整个应用程序的整体组装流程。 #### 调整制作镜像的具体指令序列 对于涉及外部固件映射创建的操作而言,确保所使用的各项开关以及输入输出数据形式均严格匹配实际硬件平台的需求是非常关键的一环。下面给出了一条标准样例供参考对比之用: ```bash ./mkimage -A arm -O U-Boot -C none -T script -d debrick-nand.txt debrick.scr ``` > **注意**: 上述例子适用于特定类型的启动加载程序环境定制场景, 用户需依据自身具体情况做出相应调整. --- ### 总结说明 综上所述,当遭遇类似于无法定位某个具体资源之类的状况时,可以从三个方面入手排查即验证资源配置准确性、补充潜在丢失的部分以及优化自动化生产环节的各项细节处理方式从而达到彻底消除此类技术难题的目的。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值