返回介绍

solution / 2100-2199 / 2184.Number of Ways to Build Sturdy Brick Wall / README

发布于 2024-06-17 01:03:09 字数 8089 浏览 0 评论 0 收藏 0

2184. 建造坚实的砖墙的方法数

English Version

题目描述

给你两个整数 height 与width ,表示你要建造的砖墙的高和宽。再给你一个下标从 0 开始的数组 bricks ,其中第 i 块砖的高度是 1 ,宽度为 bricks[i] 。每种砖的数量都是 无限 的,并且砖 不可以 进行旋转。

墙的每一行必须正好 width 单位长。为了让墙体 坚实 ,除了在首尾的位置,相邻的行砖缝 不能 在同一个位置。

请你返回建造坚实的砖墙的方法数,由于答案可能很大,需要对 109 + 7 取余

 

示例 1:

输入:height = 2, width = 3, bricks = [1,2]
输出:2
解释:前两图中的两种方法是建造一座坚实砖墙的唯二的方法。注意,第三幅图所展示的不是坚实的砖墙,因为相邻的行在中间的连接点位置相同。

示例 2:

输入:height = 1, width = 1, bricks = [5]
输出:0
解释:无法建造符合题目要求的砖墙,因为仅有的砖的长度比墙还要长。

 

提示:

  • 1 <= height <= 100
  • 1 <= width <= 10
  • 1 <= bricks.length <= 10
  • 1 <= bricks[i] <= 10
  • bricks 中所有数字 互不相同

解法

方法一:DFS + 动态规划

首先通过 DFS 构造出所有合法的排列。然后所有排列进行两两比较,找出每种排列相邻的合法排列,记录在 g 数组中。

然后进行动态规划。

过程是这样的:计算以某种排列结束的所有方案数。

初始化第一排每种排列的方案数为 1;每一排选取某种排列的总方案数为上一排能与自己相邻的排列的方案数之和。

答案为最后一排的方案数之和。

class Solution:
  def buildWall(self, height: int, width: int, bricks: List[int]) -> int:
    def dfs(v):
      if v > width:
        return
      if v == width:
        s.append(t[:])
        return
      for x in bricks:
        t.append(x)
        dfs(v + x)
        t.pop()

    def check(a, b):
      s1, s2 = a[0], b[0]
      i = j = 1
      while i < len(a) and j < len(b):
        if s1 == s2:
          return False
        if s1 < s2:
          s1 += a[i]
          i += 1
        else:
          s2 += b[j]
          j += 1
      return True

    mod = 10**9 + 7
    s = []
    t = []
    dfs(0)
    g = defaultdict(list)
    n = len(s)
    for i in range(n):
      if check(s[i], s[i]):
        g[i].append(i)
      for j in range(i + 1, n):
        if check(s[i], s[j]):
          g[i].append(j)
          g[j].append(i)
    dp = [[0] * n for _ in range(height)]
    for j in range(n):
      dp[0][j] = 1
    for i in range(1, height):
      for j in range(n):
        for k in g[j]:
          dp[i][j] += dp[i - 1][k]
          dp[i][j] %= mod
    return sum(dp[-1]) % mod
class Solution {
  private List<List<Integer>> res = new ArrayList<>();
  private List<Integer> t = new ArrayList<>();
  private static final int MOD = (int) 1e9 + 7;
  private int width;
  private int[] bricks;

  public int buildWall(int height, int width, int[] bricks) {
    this.width = width;
    this.bricks = bricks;
    dfs(0);
    int n = res.size();
    List<Integer>[] g = new List[n];
    Arrays.setAll(g, k -> new ArrayList<>());
    for (int i = 0; i < n; ++i) {
      if (check(res.get(i), res.get(i))) {
        g[i].add(i);
      }
      for (int j = i + 1; j < n; ++j) {
        if (check(res.get(i), res.get(j))) {
          g[i].add(j);
          g[j].add(i);
        }
      }
    }
    int[][] dp = new int[height][n];
    for (int j = 0; j < n; ++j) {
      dp[0][j] = 1;
    }
    for (int i = 1; i < height; ++i) {
      for (int j = 0; j < n; ++j) {
        for (int k : g[j]) {
          dp[i][j] = (dp[i][j] + dp[i - 1][k]) % MOD;
        }
      }
    }
    int ans = 0;
    for (int j = 0; j < n; ++j) {
      ans = (ans + dp[height - 1][j]) % MOD;
    }
    return ans;
  }

  private boolean check(List<Integer> a, List<Integer> b) {
    int s1 = a.get(0);
    int s2 = b.get(0);
    int i = 1, j = 1;
    while (i < a.size() && j < b.size()) {
      if (s1 == s2) {
        return false;
      }
      if (s1 < s2) {
        s1 += a.get(i++);
      } else {
        s2 += b.get(j++);
      }
    }
    return true;
  }

  private void dfs(int v) {
    if (v > width) {
      return;
    }
    if (v == width) {
      res.add(new ArrayList<>(t));
      return;
    }
    for (int x : bricks) {
      t.add(x);
      dfs(v + x);
      t.remove(t.size() - 1);
    }
  }
}
class Solution {
public:
  vector<int> bricks;
  int width;
  int mod = 1e9 + 7;
  vector<vector<int>> res;
  vector<int> t;

  int buildWall(int height, int width, vector<int>& bricks) {
    this->width = width;
    this->bricks = bricks;
    dfs(0);
    t.resize(0);
    int n = res.size();
    vector<vector<int>> g(n);
    for (int i = 0; i < n; ++i) {
      if (check(res[i], res[i])) {
        g[i].push_back(i);
      }
      for (int j = i + 1; j < n; ++j) {
        if (check(res[i], res[j])) {
          g[i].push_back(j);
          g[j].push_back(i);
        }
      }
    }
    vector<vector<int>> dp(height, vector<int>(n));
    for (int j = 0; j < n; ++j) dp[0][j] = 1;
    for (int i = 1; i < height; ++i) {
      for (int j = 0; j < n; ++j) {
        for (int k : g[j]) {
          dp[i][j] += dp[i - 1][k];
          dp[i][j] %= mod;
        }
      }
    }
    int ans = 0;
    for (int j = 0; j < n; ++j) {
      ans += dp[height - 1][j];
      ans %= mod;
    }
    return ans;
  }

  bool check(vector<int>& a, vector<int>& b) {
    int s1 = a[0], s2 = b[0];
    int i = 1, j = 1;
    while (i < a.size() && j < b.size()) {
      if (s1 == s2) return false;
      if (s1 < s2)
        s1 += a[i++];
      else
        s2 += b[j++];
    }
    return true;
  }

  void dfs(int v) {
    if (v > width) return;
    if (v == width) {
      res.push_back(t);
      return;
    }
    for (int x : bricks) {
      t.push_back(x);
      dfs(v + x);
      t.pop_back();
    }
  }
};
func buildWall(height int, width int, bricks []int) int {
  mod := int(1e9) + 7
  res := [][]int{}
  t := []int{}
  var dfs func(int)
  dfs = func(v int) {
    if v > width {
      return
    }
    if v == width {
      res = append(res, slices.Clone(t))
      return
    }
    for _, x := range bricks {
      t = append(t, x)
      dfs(v + x)
      t = t[:len(t)-1]
    }
  }
  check := func(a, b []int) bool {
    s1, s2 := a[0], b[0]
    i, j := 1, 1
    for i < len(a) && j < len(b) {
      if s1 == s2 {
        return false
      }
      if s1 < s2 {
        s1 += a[i]
        i++
      } else {
        s2 += b[j]
        j++
      }
    }
    return true
  }
  dfs(0)
  n := len(res)
  g := make([][]int, n)
  for i := 0; i < n; i++ {
    if check(res[i], res[i]) {
      g[i] = append(g[i], i)
    }
    for j := i + 1; j < n; j++ {
      if check(res[i], res[j]) {
        g[i] = append(g[i], j)
        g[j] = append(g[j], i)
      }
    }
  }
  dp := make([][]int, height)
  for i := range dp {
    dp[i] = make([]int, n)
  }
  for j := 0; j < n; j++ {
    dp[0][j] = 1
  }
  for i := 1; i < height; i++ {
    for j := 0; j < n; j++ {
      for _, k := range g[j] {
        dp[i][j] += dp[i-1][k]
        dp[i][j] %= mod
      }
    }
  }
  ans := 0
  for j := 0; j < n; j++ {
    ans += dp[height-1][j]
    ans %= mod
  }
  return ans
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文