算法设计 - 二分法和三分法,洛谷P3382

news2025/1/21 12:13:20

二分法

二分查找:找目标值位置

二分法是一种适用于特殊场景下的分治算法。

这里的特殊场景指的是,二分法需要作用在一个具有单调性的区间内。

比如,我们熟知的二分查找,就是一种二分法的具体实现,二分查找必须在一个升序或者降序的数组内,才能正确地找到目标值。

下面举个例子,演示下二分查找的过程:

有升序数组 arr = [1, 3, 5, 7, 9, 11, 13],请找出元素3在数组中的索引位置?

我们首先要为二分查找定义一个初始的查找范围[L, R],通常情况下 L = 0, R = arr.length-1,如下图所示,即两个指针分别指向数组的首尾:

 这里其实不好看单调性,我们将其转化为单调函数f(x) = 2x + 1,其中x是元素索引位置,f(x)是元素值,目标值为target

有了L,R后,我们就需要求出其中间位置mid = (L + R)/  2

然后比较f(mid)和target的大小:

  • 若 f(mid) > target,那么当前f(x)是单调递增的,因此target的位置 < mid
  • 若 f(mid) < target,那么当前f(x)是单调递增的,因此target的位置 > mid
  • 若 f(mid) == target,那么当前f(x)是单调递增的,因此target的位置 == mid

我们根据f(mid)和target的大小,就能知道target的索引位置和mid的关系,那么知道了target索引位置和mid的关系,有什么用呢?

答案是,可以缩小二分查找的范围。

  • 若 f(mid) > target,那么当前f(x)是单调递增的,因此target的位置 < mid,那么下次二分查找的右边界R就可以左移到mid-1
  • 若 f(mid) < target,那么当前f(x)是单调递增的,因此target的位置 > mid,那么下次二分查找的左边界就可以右移到mid+1

比如本题target=3,此时f(mid) = 7,则f(mid) > target,由于f(x)是单调递增的,可得target的位置 < mid,因此下次二分查找的右边界R就可以左移到mid-1

即如下图所示

这里需要思考的是,为什么R不是左移到mid(即索引3),而是左移到了mid-1(即索引2)?

其实这个原因很简单,我们已经知道了f(mid) > target了,即说明mid位置不可能是所求目标值target的位置,因此我们下次二分查找的区间没有必要包含此时的mid位置。

进入新的二分区间L,R后,我们继续取中间位置 mid = (L + R) / 2

此时可以发现,f(mid) == target,那么此时mid位置就是目标值target的位置,可以直接返回。 

如果用代码实现上面二分查找逻辑的话,如下:

Java

public class Main {
  public static void main(String[] args) {
    //    int[] arr = {1, 3, 5, 7, 9, 11, 13};
    int[] arr = {13, 11, 9, 7, 5, 3, 1};
    int target = 3;
    System.out.println(binarySearch(arr, target));
  }

  // 二分查找
  public static int binarySearch(int[] arr, int target) {
    int l = 0;
    int r = arr.length - 1;

    // 是否是单调递增的数组
    boolean isIncremental = arr[l] < arr[r];

    while (l <= r) {
      int mid = (l + r) / 2;
      int midVal = arr[mid];

      if (midVal > target) {
        if (isIncremental) r = mid - 1;
        else l = mid + 1;
      } else if (midVal < target) {
        if (isIncremental) l = mid + 1;
        else r = mid - 1;
      } else {
        return mid;
      }
    }

    return -1;
  }
}

JS

// 二分查找
function binarySearch(arr, target) {
  let l = 0;
  let r = arr.length - 1;

  // 单调性确认
  const isIncremental = arr[l] < arr[r];

  while (l <= r) {
    const mid = Math.floor((l + r) / 2);
    const midVal = arr[mid];

    if (midVal > target) {
      isIncremental ? (r = mid - 1) : (l = mid + 1);
    } else if (midVal < target) {
      isIncremental ? (l = mid + 1) : (r = mid - 1);
    } else {
      return mid;
    }
  }

  return -1;
}

// 测试
const target = 3;

const arr = [1, 3, 5, 7, 9, 11, 13];
console.log(binarySearch(arr, target));

arr.reverse();
console.log(binarySearch(arr, target));

Python

# 二分查找
def binarySearch(arr, target):
    l = 0
    r = len(arr) - 1

    # 单调性确认
    isIncremental = arr[l] < arr[r]

    while l <= r:
        mid = (l + r) // 2
        midVal = arr[mid]

        if midVal > target:
            if isIncremental:
                r = mid - 1
            else:
                l = mid + 1
        elif midVal < target:
            if isIncremental:
                l = mid + 1
            else:
                r = mid - 1
        else:
            return mid

    return -1


# 测试
arr = [1, 3, 5, 7, 9, 11, 13]
target = 3
print(binarySearch(arr, target))

arr.reverse()
print(binarySearch(arr, target))

二分查找:找目标值有序插入位置

上面算法实现中,我们可以发现,如果找不到目标值的位置,算法直接返回了-1,即表示数组中没有目标值元素。

但是有时候,我们会有一个需求,那就是如果数组中不存在目标值,那么就返回目标值在数组中的有序插入位置。

什么意思呢?

比如,arr = [1, 3, 5, 7, 9, 11, 13],现在目标值是4,那么我们应该将目标值插入到数组哪个位置,才能保证数组有序性不被破坏呢?

答案很明显,目标值4的插入位置是索引2。即插入后,arr = [1, 3, 4, 5, 7, 9, 11, 13]

下面是基于之前的二分查找逻辑,找目标值4位置的演示过程

 

 

最后L == R时,还可以进入while循环,此时mid == L == R,但是依旧 f(mid) > target,此时由于单调递增,因此target的位置应该在mid的左侧,即R = mid - 1

此时R < L,退出循环。

我们可以发现此时L指向的位置就是target的插入位置。

大家有兴趣的话,可以继续尝试下单调递减数组,最终结论是一样的。

因此,其实前面二分查找算法如果最终找不到目标值位置,那么最后L指针的位置其实就是目标值target的有序插入位置。

那么我们该如何返回这个有序插入位置呢? 

根据Java的Arrays.binarySearch设计,有序插入位置返回为 -L-1。

比如上面例子中L=2,那么binarySearch就要返回-3。

为什么要这么设计呢?

如果数组中可以找到目标值,那么目标值索引可能是0~arr.length-1中任意一个。

因此,数组中如果找不到目标值,那么此时我们不能直接目标值的有序插入位置,这会产生冲突,即搞不清楚binarySearch返回值是目标值的索引位置,还是有序插入位置。

而为了避免冲突,有序插入位置都设计为负数。即从-1开始。比如有序插入位置L=0,那么binarySearch就返回-1,即-L-1。

因此,前面binarySearch方法的实现,可以新增一个返回有序插入位置的功能:

Java

public class Main {
  public static void main(String[] args) {
    int[] arr = {1, 3, 5, 7, 9, 11, 13};
    int target = 4;

    int idx = binarySearch(arr, target);
    if (idx < 0) {
      System.out.println(-idx - 1);
    }
  }

  // 二分查找
  public static int binarySearch(int[] arr, int target) {
    int l = 0;
    int r = arr.length - 1;

    // 是否是单调递增的数组
    boolean isIncremental = arr[l] < arr[r];

    while (l <= r) {
      int mid = (l + r) / 2;
      int midVal = arr[mid];

      if (midVal > target) {
        if (isIncremental) r = mid - 1;
        else l = mid + 1;
      } else if (midVal < target) {
        if (isIncremental) l = mid + 1;
        else r = mid - 1;
      } else {
        return mid;
      }
    }

    // 若查找目标值,则返回目标值在数组中的有序插入位置l,为了避免产生冲突,返回-l-1
    return -l - 1;
  }
}

JS

// 二分查找
function binarySearch(arr, target) {
  let l = 0;
  let r = arr.length - 1;

  // 单调性确认
  const isIncremental = arr[l] < arr[r];

  while (l <= r) {
    const mid = Math.floor((l + r) / 2);
    const midVal = arr[mid];

    if (midVal > target) {
      isIncremental ? (r = mid - 1) : (l = mid + 1);
    } else if (midVal < target) {
      isIncremental ? (l = mid + 1) : (r = mid - 1);
    } else {
      return mid;
    }
  }

  // 若查找目标值,则返回目标值在数组中的有序插入位置l,为了避免产生冲突,返回-l-1
  return -l - 1;
}

// 测试
const target = 4;

const arr = [1, 3, 5, 7, 9, 11, 13];

const idx = binarySearch(arr, target);
if (idx < 0) {
  console.log(-idx - 1);
}

Python

# 二分查找
def binarySearch(arr, target):
    l = 0
    r = len(arr) - 1

    # 单调性确认
    isIncremental = arr[l] < arr[r]

    while l <= r:
        mid = (l + r) // 2
        midVal = arr[mid]

        if midVal > target:
            if isIncremental:
                r = mid - 1
            else:
                l = mid + 1
        elif midVal < target:
            if isIncremental:
                l = mid + 1
            else:
                r = mid - 1
        else:
            return mid

    # 若查找目标值,则返回目标值在数组中的有序插入位置l,为了避免产生冲突,返回-l-1
    return -l-1


# 测试
arr = [1, 3, 5, 7, 9, 11, 13]
target = 4

idx = binarySearch(arr, target)
if idx < 0:
    print(-idx-1)

三分法

三分法的应用场景

通过前面对二分法的研究,我们可以发现二分法必须在一个单调性区间内工作。

那么如果某区间不是一个单调性的,

比如有区间[l, r],其中[l, x]满足单调递增,而[x, r]满足单调递减, 即一个凸函数,即如下图所示

或者,有区间[l, r],其中[l, x]是单调递减的,而[x, r]是单调递增的,即一个凹函数,如下图所示

此时二分法可以找到极值点吗?

  • 极值点就是凸函数的最大值点,或者凹函数的最小值点。

答案是不可以的,因为二分法只能在单调区间内工作,而极值点处于一个非单调区间内,因此二分法无法正确找到凹凸函数的极值点。

此时我们就需要借助三分法来实现凹凸函数找极值点。

三分法找极值点有两种策略:

  • 三等份
  • 近似三等份

三等份

前面研究二分法时,我们是找L,R的中间位置mid,并比较f(mid)和target的大小,来确定target的位置在mid的左侧还是右侧,或者就是mid本身。

而对于凹凸函数而言,我们需要找到L,R区间的三等份点。

什么是三等份点?即可以将[L, R]区间均分为三等份的两个点,

比如下图mL,mR就是三等份点

[L,R]区间被mL和mR点均分为了L~mL,mL~mR,mR~R三个等份区间。 

如果 f(mL) <= f(mR),那么对于凸函数而言,极值点必然在mL的右侧,但是极值点和mR的位置关系是不确定的,如下图所示

 

 反之,如果 f(mL) >= f(mR),那么对于凸函数而言,极值点必然在mR的左侧,但是极值点和mL的位置关系不确定。

因此,对于凸函数而言:

如果 f(mL) <= f(mR),那么可以确定极值点位置在mL的右侧,即此时缩小三分区间时,可以将L右移到mL位置。

如果 f(mL) >= f(mR),那么可以确定极值点位置在mR的左侧,即此时缩小三分区间时,可以将R左移到mR位置。

当新的[L, R]区间确认后,则可以继续进行三等份点确认,然后重复上面逻辑。

那么何时结束呢?

三分法和二分法的区别在于,三分法的L < R总是成立,为什么呢?

因为上面缩小区间时,L是直接移动到mL位置,或者R直接移动到mR位置。大家可以看下这个视频,从04:19开始

【【4K算法详解】【二分与三分】从二分法到牛顿法,领着你的思维带你观望方程求解与数值优化算法】

此时就需要一个精度,即当L和R之间的距离小于等于某个精度时,就可以认为当前L或R就是所求的极值点位置。这里的精度通常用eps表示。

我们用代码代码实现三分法找极值

Java

public class Main {
  public static void main(String[] args) {
    // 测试
    System.out.println(trichotomy(-100, 10));
  }

  // 凸函数 f(x) = -x^2
  public static double f(double x) {
    return -x * x;
  }

  // 求凸函数极值
  public static double trichotomy(double l, double r) {
    // 精度
    double eps = 0.00001;

    while (r - l >= eps) {
      double thridPart = (r - l) / 3;

      // 靠左三等份点
      double ml = l + thridPart;
      // 靠右三等份点
      double mr = r - thridPart;

      // 凸函数l,r移动逻辑
      if (f(ml) < f(mr)) {
        l = ml;
      } else {
        r = mr;
      }
    }

    return l;
  }
}

JS

// 凸函数 f(x) = -x^2
function f(x) {
  return -(x ** 2);
}

// 求凸函数极值
function trichotomy(l, r) {
  // 精度
  const eps = 0.00001;

  while (r - l >= eps) {
    const thridPart = (r - l) / 3;
    // 靠左三等份点
    const ml = l + thridPart;
    // 靠右三等份点
    const mr = r - thridPart;

    // 凸函数l,r移动逻辑
    if (f(ml) < f(mr)) {
      l = ml;
    } else {
      r = mr;
    }
  }

  return l;
}

// 测试
console.log(trichotomy(-100, 10));

Python

# 凸函数 f(x) = -x^2
def f(x):
    return -(x ** 2)


# 求凸函数极值
def trichotomy(l, r):
    # 精度
    eps = 0.00001

    while r - l >= eps:
        thridPart = (r - l) / 3

        # 靠左三等份点
        ml = l + thridPart
        # 靠右三等份点
        mr = r - thridPart

        # 凸函数l,r移动逻辑
        if f(ml) < f(mr):
            l = ml
        else:
            r = mr

    return l


# 测试
print(trichotomy(-100, 10))

 

近似三等份

上面算法是将[L,R]区间均分为三等份,而更优的策略是直接找[L,R]的中间点mid,然后只根据mid点就能确定极值点的位置。

怎么办到的呢?如下图所示mid是L,R的中间点。

 此时我们可以找一个很小的精度accuracy,然后比较两个位置点的关系:

  • mid + accuracy
  • mid - accuracy

对于凸函数而言:

  • 如果 f(mid - accuracy)  <  f(mid + accuracy),那么说明mid点处于凸函数的上升区间中,即极值点位置在mid的右侧,下次缩小区间时,应该让L = mid
  • 如果 f(mid - accuracy) > f(mid + accuracy),那么说明mid点处于凸函数的下降区间中,即极值点位置在mid的左侧,下次缩小区间时,应该让R = mid

实现代码如下

Java

public class Main {
  public static void main(String[] args) {
    // 测试
    System.out.println(trichotomy(-100, 10));
  }

  // 凸函数 f(x) = -x^2
  public static double f(double x) {
    return -x * x;
  }

  // 求凸函数极值
  public static double trichotomy(double l, double r) {
    // 精度
    double eps = 0.00001;
    double accuracy = 0.000000001;

    while (r - l >= eps) {
      double mid = (r + l) / 2;

      // 凸函数l,r移动逻辑
      if (f(mid - accuracy) < f(mid + accuracy)) {
        l = mid;
      } else {
        r = mid;
      }
    }

    return l;
  }
}

JS

// 凸函数 f(x) = -x^2
function f(x) {
  return -(x ** 2);
}

// 求凸函数极值
function trichotomy(l, r) {
  // 精度
  const eps = 0.00001;
  const acc = 0.0000000001;

  while (r - l >= eps) {
    const mid = (r + l) / 2;

    if (f(mid - acc) < f(mid + acc)) {
      l = mid;
    } else {
      r = mid;
    }
  }

  return l;
}

// 测试
console.log(trichotomy(-100, 10));

Python

# 凸函数 f(x) = -x^2
def f(x):
    return -(x ** 2)


# 求凸函数极值
def trichotomy(l, r):
    # 精度
    eps = 0.00001
    acc = 0.0000000001

    while r - l >= eps:
        mid = (r + l) / 2

        if f(mid - acc) < f(mid + acc):
            l = mid
        else:
            r = mid

    return l


# 测试
print(trichotomy(-100, 10))

洛谷P3328 【模板】三分法

题目链接

P3382 【模板】三分法 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

三等份求解

Java

import java.util.Scanner;

public class Main {
  static int n;
  static double l;
  static double r;
  static double[] a;

  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);

    n = sc.nextInt();
    l = sc.nextDouble();
    r = sc.nextDouble();

    a = new double[n + 1];
    for (int i = 0; i <= n; i++) {
      a[i] = sc.nextDouble();
    }

    System.out.println(getResult());
  }

  public static double getResult() {
    while (r - l >= 0.000001) {
      double ml = l + (r - l) / 3.0;
      double mr = r - (r - l) / 3.0;

      if (f(ml) < f(mr)) l = ml;
      else r = mr;
    }
    return l;
  }

  public static double f(double x) {
    double ans = 0;
    for (int i = n; i >= 0; i--) {
      ans += Math.pow(x, i) * a[n - i];
    }
    return ans;
  }
}

JS

/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");

const rl = readline.createInterface({
  input: process.stdin,
  output: process.stdout,
});

const lines = [];
let n, l, r, a;
rl.on("line", (line) => {
  lines.push(line);

  if (lines.length == 2) {
    [n, l, r] = lines[0].split(" ").map(Number);
    a = lines[1].split(" ").map(Number);
    console.log(getResult());
  }
});

const eps = 1e-5;

function getResult() {
  while (r - l >= eps) {
    let k = (r - l) / 3;
    let ml = l + k;
    let mr = r - k;

    if (f(ml) > f(mr)) r = mr;
    else l = ml;
  }

  return l;
}

function f(x) {
  let ans = 0;
  for (let i = n; i >= 0; i--) {
    ans += Math.pow(x, i) * a[n - i];
  }
  return ans;
}

Python

# 输入获取
n, l, r = map(float, input().split())
n = int(n)
a = list(map(float, input().split()))


def f(x):
    ans = 0
    for i in range(n, -1, -1):
        ans += pow(x, i) * a[n - i]
    return ans


# 算法入口
def getResult(l, r):
    eps = 1e-5
    while r - l >= eps:
        k = (r - l) / 3
        ml = l + k
        mr = r - k

        if f(ml) < f(mr):
            l = ml
        else:
            r = mr
    return l


print(getResult(l, r))

近似三等份求解

Java

import java.util.Scanner;

public class Main {
  static int n;
  static double l;
  static double r;
  static double[] a;

  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);

    n = sc.nextInt();
    l = sc.nextDouble();
    r = sc.nextDouble();

    a = new double[n + 1];
    for (int i = 0; i <= n; i++) {
      a[i] = sc.nextDouble();
    }

    System.out.println(getResult());
  }

  static double eps = 1e-5;

  public static double getResult() {
    while (r - l >= eps) {
      double mid = (l + r) / 2.0;

      if (f(mid - eps) < f(mid + eps)) {
        l = mid;
      } else {
        r = mid;
      }
    }
    return l;
  }

  public static double f(double x) {
    double ans = 0;
    for (int i = n; i >= 0; i--) {
      ans += Math.pow(x, i) * a[n - i];
    }
    return ans;
  }
}

JS

/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");

const rl = readline.createInterface({
  input: process.stdin,
  output: process.stdout,
});

const lines = [];
let n, l, r, a;
rl.on("line", (line) => {
  lines.push(line);

  if (lines.length == 2) {
    [n, l, r] = lines[0].split(" ").map(Number);
    a = lines[1].split(" ").map(Number);
    console.log(getResult());
  }
});

function getResult() {
  const eps = 1e-5;
  while (r - l >= eps) {
    const mid = (r + l) / 2;
    if (f(mid - eps) < f(mid + eps)) l = mid;
    else r = mid;
  }

  return l;
}

function f(x) {
  let ans = 0;
  for (let i = n; i >= 0; i--) {
    ans += Math.pow(x, i) * a[n - i];
  }
  return ans;
}

Python

# 输入获取
n, l, r = map(float, input().split())
n = int(n)
a = list(map(float, input().split()))


def f(x):
    ans = 0
    for i in range(n, -1, -1):
        ans += pow(x, i) * a[n - i]
    return ans


# 算法入口
def getResult(l, r):
    eps = 1e-5
    while r - l >= eps:
        mid = (l + r) / 2

        if f(mid - eps) < f(mid + eps):
            l = mid
        else:
            r = mid
    return l


print(getResult(l, r))

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/414316.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【云原生】阿里云ACK部署MySQL 数据持久化存储

思路图 一、概述 ACK介绍 阿里云容器服务Kubernetes版(Alibaba Cloud Container Service for Kubernetes,简称容器服务ACK)是全球首批通过Kubernetes一致性认证的服务平台,提供高性能的容器应用管理服务,支持企业级Kubernetes容器化应用的生命周期管理,轻松高效地在云端…

MAE论文阅读《Masked Autoencoders Are Scalable Vision Learners》

文章目录动机方法写作方面参考Paper: https://arxiv.org/pdf/2111.06377.pdf 动机 首先简要介绍下BERT&#xff0c;NLP领域的BERT是基于Transformer架构&#xff0c;并采取无监督预训练的方式去训练模型。它提出的预训练方法在本质上是一种masked autoencoding&#xff0c;也就…

【CSS】定位 ⑤ ( 子元素绝对定位 父元素相对定位 | 代码示例 )

文章目录一、子元素绝对定位 父元素相对定位二、代码示例一、子元素绝对定位 父元素相对定位 绝对定位 要和 带有定位 的 父容器 搭配使用 ; 子元素 使用绝对定位 , 父元素要使用 相对定位 ; 子元素使用 绝对定位 , 在布局中不会保留其位置 , 子元素完全依赖 父容器 的位置 ,…

阿里巴巴春招的后端面经来啦~

操作系统 一个操作系统&#xff0c;我们在衡量它的内存占用的时候&#xff0c;它一般会有哪些内存的部分&#xff1f; 读者答&#xff1a;堆和栈 补充&#xff1a; 这个其实是问你对free命令的理解。 主机的内存做一些清理的动作。你知道这里面会涉及到对哪些内存区域进行操…

大数据任务调度器 —— Azkaban 3.0 进阶应用

文章目录任务依赖任务重试JavaProcess 进程任务提交条件执行自定义条件预定义条件定时执行邮件警告前言&#xff1a;Azkaban 搭建以及基础介绍&#xff0c;查看这篇博客 —— 大数据任务调度器 —— Azkaban 3.0 部署与简单应用 任务依赖 Azkaban 中的任务依赖设置十分简单&…

OpenCV中的相机标定

之前在https://blog.csdn.net/fengbingchun/article/details/130039337 中介绍了相机的内参和外参&#xff0c;这里通过OpenCV中的接口实现对内参和外参的求解。 估计相机参数的过程称为相机标定(camera calibration)。相机标定是使用已知的真实世界模式(例如棋盘)来估计…

从风靡全球到风口,他们是如何做到的?

大家好&#xff0c;我是湖北鑫优尚电子商务。我们是从2019年下半年就注意到了TikTok的风口&#xff0c;并在2020年初&#xff0c;团队就开始投入TikTok的运营及商业变现。从2019年起&#xff0c;围绕TikTok一系列的产品功能迭代和商业模式开放的声音一直不绝于耳&#xff0c;直…

零基础可以学习数据分析吗,有没有好的培训机构推荐?

数据分析从沿海火到了中西部的软件园&#xff0c;从传统互联网企业火到了新经济领域&#xff0c;火到了第一二产业。数字化成为这个时代的标签&#xff0c;而数据也成为了最有价值的资源&#xff0c;更多企业重视数据&#xff1b;因为有了真实数据的支撑&#xff0c;所有的决策…

快速排序(挖坑法 前后指针法 非递归版本)

上一次我们说了快速排序的hoare的版本&#xff0c;但是该版本有很多问题&#xff0c;首先是需要控制很多边界&#xff0c;比较复杂一点 其次就是上一次的快速排序还是有很多的其他问题 我们试着想一下&#xff0c;如果我们用快速排序拍有序数组&#xff0c;那会怎么样&#x…

数字化和自动化的潮流:外贸企业如何应对?

当今世界正面临着数字化和自动化的浪潮&#xff0c;这些技术的应用正在深刻地改变着各个领域的生产、交易和管理方式。尤其在外贸行业&#xff0c;数字化和自动化的应用已成为一种趋势和必然。在本文中&#xff0c;我们将探讨数字化和自动化对外贸行业的影响和挑战&#xff0c;…

Windows环境下实现设计模式——解释器模式(JAVA版)

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天总结一下Windows环境下如何编程实现解释器模式&#xff08;设计模式&#xff09;。 不知道大家有没有这样的感觉&#xff0c;看了一大堆编程和设计模式的书&#xff0c;却还是很难理解设计模式&#xff…

巧用千寻位置GNSS软件|CAD功能全解析

千寻位置GNSS软件中的CAD功能&#xff0c;用于已有 CAD的图形的导入和编辑&#xff0c;并且可以对 CAD图形已有线条进行线放样&#xff0c;在日常测绘工作中十分常见。下面向各位介绍CAD功能的使用技巧。点击【测量】->【CAD】&#xff0c;进入 CAD功能如图 5.3-1所示。以下…

三、线程状态【3/12】【多线程】

线程的状态3. 线程的状态3.1 观察线程的所有状态3.2 线程状态和状态转移的意义3.3 观察线程的状态和转移3. 线程的状态 3.1 观察线程的所有状态 线程的状态是一个枚举类型 Thread.State public class ThreadState {public static void main(String[] args) {for (Thread.State…

项目7:(aliyun)实现短信的发送和验证微服务和上传文件删除文件微服务

项目7&#xff1a;实现短信的发送和验证 1.对gulimall-common补充 2.短信验证的流程&#xff08;aliyun的sms&#xff09; 3.具体接口的编写&#xff08;新建微服务service-sms&#xff09; 4.上传和删除文件流程&#xff08;aliyun的oss&#xff09; 5.具体接口的编写&am…

区块链智能合约开发学习

最近正在肝区块链知识学习&#xff0c;入手学习智能合约的开发&#xff0c;由于网上资料实在是太少了&#xff0c;好不容易东拼西凑完成了智能合约的开发、编译、部署、web3js调用&#xff08;网页页面&#xff09;和web3j调用&#xff08;java调用&#xff09;&#xff0c;赶紧…

Linux 内存回收,思维导图记录

最近天天跟内存斗智斗勇&#xff0c;整理下学习的记录 一些图片 参考 Tuning Linux Kernel Parameters For PostgreSQL Optimization PostgreSQL recommendations - Documentation for BMC Client Management 12.6 - BMC Documentation PostgreSQL load tuning on Red Hat E…

【vSphere | Python】vSphere Automation SDK for Python Ⅵ—— VM Guest Processes APIs

目录12. VM APIs12.1 VM Guest Processes APIsProcesses 进程Operations 操作&#xff08;1&#xff09;List Guest Processes&#xff08;2&#xff09;Get Guest Processes&#xff08;3&#xff09;Create Guest Processes&#xff08;4&#xff09;Delete Guest Processes参…

PaddleHub 更改模型默认下载位置

文章目录1.PaddleHub介绍2.PaddleHub安装3.PaddleHub使用中出现的问题4.更改PaddleHub模型的默认下载位置5. PaddleHub的简单使用1.PaddleHub介绍 PaddleHub 是基于 PaddlePaddle 开发的预训练模型管理工具&#xff0c;可以借助预训练模型更便捷地开展迁移学习工作&#xff0c…

docker内部执行nvidia-smi无任何显示的解决方法

docker内部执行nvidia-smi无任何显示的解决方法 贺志国 2023.4.11 今天在Docker内部编译程序&#xff0c;发现与CUDA相关的代码居然没有编译&#xff0c;于是在Docker内部执行Nvidia显卡驱动检测命令nvidia-smi&#xff0c;执行完毕后&#xff0c;无任何输出&#xff0c;也没…

计算机视觉面试题-网络结构相关问题总结-未完待续

VGG卷积核为什么取33 &#xff1f; VGG使用33卷积核的优势是什么? Resnet 主要解决什么问题 为什么会有ResNet&#xff1f; 深度网络退化的原因 Resnet的针对网络退化提出的残差网络 Resnet网络结构 Resnet网络结构中如何实现的下采样 Resnet50网络结构Resnet特点 vgg16与 res…