青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 68  文章 - 57  trackbacks - 0
<2025年11月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
30123456

常用鏈接

留言簿(8)

隨筆分類(74)

隨筆檔案(68)

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

好久不寫了啊,最近打算重新啟用這個blog。就先寫這個題目吧,滿經典的。
【題目大意】
  m * n的區域內,每個整數坐標都有整點,問以這些整點為端點能夠形成多少個三角形。(0 < m, n <= 1000)

【題目分析】
  這個題目乍一看挺簡單的,但是想做對還是要仔細的思考下的。補集轉化的思想,求出所有共線的三元組,然后用總數減掉就是答案了,關鍵就是如何求共線三元組。x坐標相同和y坐標相同的比較好計算,在一條斜線的就不好算了,畫個圖發現,即使斜率相同的線,經過的格點數可能各不相同。思路當然還是枚舉y / x(不同的y / x確定了不同的矩形區域),之后如何有效的計算,我采用的方法可能有些麻煩,有點類容斥的方法。以斜率為a / b為例,(a, b) = g,那么m * n的區域內一定有(m - a + 1) * (n - b + 1)個那么大的矩形,這樣的矩形經過的格點數是(g + 1);然后因為同等斜率小一點的矩形(a - a / g, b - b / g)也是存在的,個數同樣可以統計出來,但是有些大矩形包括了,要去掉;因此就采用這種思想,先求出大矩形的個數,然后依次往下減,就可以避免重復計數了。雖然這樣復雜度有點高,不過極限數據還是比較快的跑出來了。
  說的可能不太清楚,具體代碼如下:
 1 #include <cstdio>
 2 #include <iostream>
 3 using namespace std;
 4 const int N = 1024;
 5 
 6 bool tag[N][N];
 7 long long calc(long long n)
 8 {
 9     if (n <= 2)
10         return 0;
11     return n * (n - 1* (n - 2/ 6;
12 }
13 int gcd(int a, int b)
14 {
15     return b == 0 ? a : gcd(b, a % b);
16 }
17 
18 int main()
19 {
20     int m, n, g, a, b, timer = 1, cnt[N], ta, tb;
21     long long ans;
22 
23     while (scanf("%d %d"&m, &n) == 2)
24     {
25         memset(tag, 0sizeof(tag));
26         if (m == 0 && n == 0)
27             break;
28         ans = calc((m + 1* (n + 1)) - calc(m + 1* (n + 1- calc(n + 1* (m + 1);
29         for (int i = m; i >= 1; i--)
30             for (int j = n; j >= 1; j--)
31             {
32                 g = gcd(i, j);
33                 a = i / g, b = j / g;
34                 if (tag[a][b])      continue;
35                 memset(cnt, 0sizeof(cnt));
36                 tag[a][b] = 1;
37                 a = i, b = j;
38                 ta = i / g, tb = j / g;
39                 for (int k = g; k >= 2; k--)
40                 {
41                     cnt[k] += (m - a + 1* (n - b + 1);
42                     ans -= calc(k + 1* cnt[k] * 2;
43                     for (int t = 1; t <= k - 2; t++)
44                         cnt[k-t] -= (t + 1* cnt[k];
45                     a -= ta, b -= tb;
46                 }
47             }
48         cout << "Case " << timer++ << "" << ans << endl;
49     }
50 
51     return 0;
52 }
53 
posted @ 2009-10-15 19:54 sdfond 閱讀(392) | 評論 (0)編輯 收藏

這個題目去年就過了,用得是狀態壓縮dp,不過沒用dfs預處理,當時做得不是很明白,還是參考網上的一個代碼做的。
現在重新做了一下這個題目,請教了icecream,學會了一個很簡練的做法,而且比較好理解還好寫。
首先還是狀態的表示,用0表示沒有放木塊,用1表示放了木塊。此外,對于一個橫放的木塊,對應的兩位都用1表示;對于一個豎放的木塊,第一行用1表示,第二行用0表示。這只是一種設狀態的方式,當然還有別的設法,但是這種方法在接下來就會發現優點。

狀態表示完就要處理轉移了,如何判斷一個轉移是否合法比較難辦,用一個dfs卻可以簡潔的解決這個問題。
對于上一行到下一行的轉移,規定上一行一定填滿,這樣有三種方式:
    dfs(col + 1, (s1 << 1) | 1, s2 << 1, n);
    dfs(col + 1, s1 << 1, (s2 << 1) | 1, n);
    dfs(col + 2, (s1 << 2) | 3, (s2 << 2) | 3, n);
第一種上面是1,那么下面一定是0,表示是一個豎放的木塊。
第二種上面是0,就是說這個位置一定是一個豎放木塊的下半截,那么下一行肯定是要另起一行了,放一個豎放或者橫放的木塊都必須是1。
第三種相當于上下兩行各放一個橫木塊。
實現的時候我用了一個vector記錄每個狀態所有可行的轉移,這樣在dp的時候可以加快一些效率。

還有一個問題需要考慮,那就是初值和最終的結果。如果尋找合法狀態,依然比較麻煩,假設共有n行,可以分別在這n行上下新加兩行。下面一行都是1,由于第n行肯定要填滿,這樣新加的全1的行就相當于頂住了第n行使其沒有凸出(有凸出那么第n+1行有0),而通過第n行到第n+1行轉移保留了所有合法狀態;同理最上面加的那行保證第一行沒有凸出。最后第n+1行對應全1的狀態就是最終的結果了。通過新加行巧妙地解決了初值和終值。

實現的時候也需要注意一下,在TSP問題中,外層循環是狀態,內層是點,之所以這樣寫因為在枚舉點的時候,可能會從比當前編號大的點轉移,但是由于無論怎樣轉移過來的狀態肯定比當前狀態?。ㄈサ袅?),所以先從小到大枚舉狀態就保證轉移過來的狀態一定是算過的。而這個題目里面正好反過來,因為狀態可能從比當前狀態大的狀態轉移過來,而行數肯定是從編號小的行轉移,因此先枚舉行就能保證轉移過來的狀態一定是更新過的。


#include <cstdio>
#include 
<vector>
#include 
<algorithm>
using namespace std;
const int N = 11;

vector
<int> g[1<<N];
long long dp[N+2][1<<N];

void dfs(int col, int s1, int s2, int n)
{
    
if (col >= n)
    {
        
if (s1 < (1 << n) && s2 < (1 << n))
            g[s2].push_back(s1);
        
return;
    }
    dfs(col 
+ 1, (s1 << 1| 1, s2 << 1, n);
    dfs(col 
+ 1, s1 << 1, (s2 << 1| 1, n);
    dfs(col 
+ 2, (s1 << 2| 3, (s2 << 2| 3, n);
}
long long calc(int m, int n)
{
    
if (m < n)  swap(m, n);
    dfs(
000, n);
    
int state = 1 << n;

    dp[
0][0= 1;
    
for (int i = 1; i <= m + 1; i++)
        
for (int s = 0; s < state; s++)
            
for (int j = 0; j < g[s].size(); j++)
                dp[i][s] 
+= dp[i-1][g[s][j]];
    
return dp[m+1][state-1];
}

int main()
{
    
int m, n;

    
while (scanf("%d %d"&m, &n) == 2)
    {
        
if (m == 0 && n == 0)
            
break;
        
for (int i = 0; i < (1 << N); i++)
            g[i].clear();
        memset(dp, 
0sizeof(dp));
        printf(
"%I64d\n", calc(m, n));
    }

    
return 0;
}
posted @ 2009-07-31 08:12 sdfond 閱讀(2343) | 評論 (1)編輯 收藏
【題目大意】
  定義兩個三元組I(xi, yi, zi)和J(xj, yj, zj),他們的差為D(I, J) = max{xi - xj, yi - yj, zi - zj} - min{xi - xj, yi - yj, zi - zj},給定n個三元組(n <= 200000),求任意兩個三元組的差的和。

【題目分析】
  數據范圍非常大,枚舉必然不可,需要數學方法。這個題目巧妙之處在于,模型經過了層層的包裝,要想一下子有想法還真不容易。既然不能枚舉了,這個max和min操作就不好辦了,應該設法去掉。max{a, b, c} - min{a, b, c} = |a - b| + |b - c| + | c - a| / 2,這個公式應該不難想到,但是這只是第一步,因為引進了絕對值,依然不好做。可以先算出分子,最后再除2。接下來需要一個等價變換,以a - b為例,a - b = xi - xj - yi + yj = (xi - yi) - (xj - yj),同理把b - c、c - a都寫成這種形式。這一步變換看似作用不大,但是假設我們算出所有的xi - yi之后(i = 0... n - 1),將其排序,會發現,對于第i個xi - yi,它前面的都比它小,后面的都比它大。而實際上,由于求任意兩個三元組的差,肯定xi - yi會和任意的xj - yj都作差的,加了絕對值后,它對最后的結果就會貢獻i個(xi - yi),n - i - 1個-(xi - yi)。同樣的方法算出所有的(yi - zi)和(zi - xi),結果就能夠求出來了。算法復雜度O(n * logn)。

【題目總結】
  這是一道不錯的題目,首先考察了公式的變形,需要改寫max - min操作,之后的等價變換和排序的思想都非常值得借鑒。
題目代碼:
#include <cstdio>
#include 
<algorithm>
using namespace std;
const int N = 200010;

int x[N], y[N], z[N];
int main()
{
    
int n, a, b, c;

    
while (scanf("%d"&n) == 1 && n)
    {
        
for (int i = 0; i < n; i++)
        {
            scanf(
"%d %d %d"&a, &b, &c);
            x[i] 
= a - b;
            y[i] 
= b - c;
            z[i] 
= c - a;
        }
        sort(x, x 
+ n);
        sort(y, y 
+ n);
        sort(z, z 
+ n);
        
long long ans = 0;
        
for (int i = 0; i < n; i++)
            ans 
+= (2 * i + 1 - n) * (long long)(x[i] + y[i] + z[i]);
        printf(
"%I64d\n", ans / 2);
    }

    
return 0;
}
posted @ 2009-07-14 10:34 sdfond 閱讀(315) | 評論 (0)編輯 收藏

  題目大意是給定n個點的坐標(n <= 10000),問把這些點移動到一橫行并且一個挨著一個(具體位置任意)的最少移動步數(其中每次只能向上下左右移動一個坐標)。
  這個題目體現了轉化的思想。首先考慮這樣的問題:一個數軸上有n個坐標,問把這n個坐標移動到一個點上最少移動步數,其中每次移動一個格子。根據中位數的定義,把所有坐標排序后第n / 2個坐標是中位數,把所有坐標移動到這上面移動次數最小。證明很容易想到,因為如果不這樣的話,把目標坐標往左平移還是往右平移,勢必造成左半部的坐標集體變化1,右半部的坐標也集體變化1,如果左右半部坐標的個數不同,那么顯然就不是最優的了。
  接下來考慮題目,題目中x和y的移動是孤立的,可以分開討論。y的移動方法和上面討論的情況一樣,現在考慮x的移動。x的移動要求最終是一個挨著一個的,x排好序之后,假設最終所有點以x0為左端點依次排開,對應的點分別為x0, x1...那么問題的答案就等于把這n個坐標依次對應的挪到x0到xn-1上的步數。如果我們把這n個目標點分別都移動到x0上,那么問題就轉化成了中位數問題了。考慮把xi移動到x0上,要花費i步,為了保證問題是等價變換的,應該把xi在原坐標中對應的xi'也相應的向左移動i步,這樣xi'移動到xi的代價就是不變的。設xi'左移i步后的新位置是xi'',那么問題就轉化成:把x0''到xn-1''這n個點移動到一個坐標的最小步數,用中位數的方法就可以做出來了。
  這個題目的巧妙之處在于把一個未知問題轉化成一個已知問題。轉化的思想在數學中用的很多,應該多多練習。

題目代碼:

#include <cstdio>
#include 
<algorithm>
using namespace std;
const int N = 10010;

int main()
{
    
int x[N], y[N], n;

    
while (scanf("%d"&n) == 1)
    {
        
for (int i = 0; i < n; i++)
            scanf(
"%d %d"&x[i], &y[i]);
        sort(x, x 
+ n);
        sort(y, y 
+ n);
        
for (int i = 0; i < n; i++)
            x[i] 
-= i;
        sort(x, x 
+ n);
        
int ans = 0;
        
for (int i = 0; i < n / 2; i++)
            ans 
+= x[n-i-1- x[i] + y[n-1-i] - y[i];
        printf(
"%d\n", ans);
    }

    
return 0;
}
posted @ 2009-06-25 10:13 sdfond 閱讀(276) | 評論 (0)編輯 收藏
  給定n個數求這n個數劃分成互不相交的m段的最大m子段和。
  經典的動態規劃優化的問題。設f(i, j)表示前i個數劃分成j段,且包括第i個數的最大m子段和,那么有dp方程:
    f(i, j) = max { f(i - 1, j) + v[i], max {f(k, j - 1) + v[i]}(k = j - 1 ... i - 1) }
  也就是說第i個數要么自己劃到第j段,要么和前一個數一起劃到第j段里面,轉移是O(n)的,總復雜度O(n * n * m)。
  可以引入一個輔助數組來優化轉移。設g(i, j)表示前i個數劃分成j段的最大子段和(注意第i個數未必在j段里面),那么遞推關系如下:
    g(i, j) = max{g(i - 1, j), f(i, j)},分是否加入第i個數來轉移
  這樣f的遞推關系就變成:
    f(i, j) = max{f(i - 1, j), g(i - 1, j - 1)} + v[i],轉移變成了O(1)
  這樣最后的結果就是g[n][m],通過引入輔助數組巧妙的優化了轉移。實現的時候可以用一維數組,速度很快。

附HDU 1024題目代碼:
#include <cstdio>
#include 
<algorithm>
using namespace std;
const int N = 1000010, INF = 0x3fffffff;

int f[N], g[N], a[N];

int max_sum(int m, int n)
{
    
int i, j, t;
    
for (i = 1; i <= n; i++)
    {
        t 
= min(i, m);
        
for (j = 1; j <= t; j++)
        {
            f[j] 
= max(f[j], g[j-1]) + a[i];
            g[j
-1>?= f[j-1];
        }
        g[j
-1>?= f[j-1];
    }
    
return g[m];
}

int main()
{
    
int m, n;

    
while (scanf("%d %d"&m, &n) == 2 && m && n)
    {
        
for (int i = 1; i <= n; i++)
        {
            f[i] 
= g[i] = -INF;
            scanf(
"%d"&a[i]);
        }
        printf(
"%d\n", max_sum(m, n));
    }

    
return 0;
}
posted @ 2009-06-19 11:18 sdfond 閱讀(4907) | 評論 (4)編輯 收藏
僅列出標題
共14頁: First 3 4 5 6 7 8 9 10 11 Last 
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲一区二区免费视频| 尤物视频一区二区| 国产夜色精品一区二区av| 亚洲精品午夜精品| 久久九九精品99国产精品| 91久久久久久久久久久久久| 亚洲免费在线视频| 国产精品日韩精品| 小辣椒精品导航| 亚洲欧美日韩国产| 欧美高清视频一二三区| 欧美日本韩国一区| 日韩视频在线永久播放| 亚洲精品小视频| 国产精品一区二区久久久久| 午夜日韩福利| 久久久久久久一区二区| 亚洲精品日韩在线| 先锋影音国产精品| 日韩写真视频在线观看| 亚洲精品一区在线观看| 国产美女精品视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 在线播放亚洲| 99精品国产福利在线观看免费| 国产日韩精品一区二区三区| 欧美激情中文字幕一区二区| 欧美日韩综合在线免费观看| 免费国产一区二区| 国产精品久久久久国产精品日日| 久久久精品一区| 国产精品网站一区| 日韩一级成人av| 国产精品99久久久久久www| 久久香蕉精品| 噜噜噜噜噜久久久久久91 | 欧美成va人片在线观看| 亚洲人成高清| 亚洲国产精品久久久久秋霞蜜臀| 亚洲资源在线观看| 午夜在线不卡| 国产毛片精品视频| 亚洲一区二区三区影院| 亚洲一区二区精品| 欧美色图天堂网| 亚洲综合欧美日韩| 欧美一级片在线播放| 国产亚洲欧美日韩美女| 久久久精品999| 欧美国产精品中文字幕| 亚洲国产婷婷| 国产精品va在线| 欧美一区高清| 欧美黑人一区二区三区| 久久一区中文字幕| 免费在线欧美视频| 亚洲剧情一区二区| 国产精品亚洲美女av网站| 欧美一区二视频| 亚洲精品国精品久久99热一| 亚洲视频在线一区| 在线观看一区欧美| 欧美午夜电影网| 乱人伦精品视频在线观看| 亚洲欧洲三级| 久久精品国产免费| 一区二区三区四区国产| 国内揄拍国内精品久久| 欧美色欧美亚洲另类二区| 亚洲一区在线播放| 亚洲精品黄色| 欧美成人小视频| 亚洲精品一二三| 免费av成人在线| 亚洲小说欧美另类社区| 另类尿喷潮videofree| 亚洲天堂偷拍| 亚洲日本va在线观看| 激情成人综合| 国产精品专区h在线观看| 欧美不卡激情三级在线观看| 欧美一区二区视频在线| 亚洲欧美一区二区原创| 亚洲视频www| 中文日韩欧美| 一区二区三区欧美在线观看| 亚洲精品乱码久久久久| 欧美黄在线观看| 亚洲片国产一区一级在线观看| 亚洲黑丝在线| 亚洲三级视频| 亚洲欧美久久久| 久久久久国产精品麻豆ai换脸| 久久久999精品免费| 久久人人爽人人爽| 欧美va天堂| 99国产精品国产精品久久| 亚洲永久免费精品| 久久久综合激的五月天| 欧美破处大片在线视频| 国产精品嫩草久久久久| 国内精品视频在线播放| 亚洲国产精品激情在线观看| 亚洲丝袜av一区| 欧美激情第4页| 亚洲字幕在线观看| 欧美精品国产精品日韩精品| 国产亚洲一区在线播放| 亚洲日韩欧美视频一区| 欧美一级二区| 亚洲视频大全| 在线一区视频| 亚洲国产精品久久久久秋霞不卡| 亚洲一区免费网站| 欧美激情视频在线播放 | 久久久久久穴| 国产精品成人aaaaa网站| 玉米视频成人免费看| 久久久噜噜噜久久中文字免| 日韩一级大片| 久久精品视频在线看| 国产婷婷色一区二区三区四区| 一区二区欧美精品| 亚洲欧洲精品成人久久奇米网| 久久精品女人的天堂av| 国产婷婷精品| 久久亚洲影音av资源网| 欧美亚洲网站| 精品av久久707| 你懂的视频一区二区| 久久精品一区四区| 亚洲精品国产精品乱码不99| 欧美激情日韩| 欧美日韩mp4| 欧美另类变人与禽xxxxx| 亚洲成色最大综合在线| 欧美v国产在线一区二区三区| 亚洲欧美综合网| 亚洲黄色精品| 一区二区久久久久| 欧美日韩免费在线观看| 午夜精品理论片| 蜜桃伊人久久| 亚洲一区在线直播| 午夜精品理论片| 亚洲精品美女久久7777777| 亚洲欧美日韩网| 亚洲国产精品va在线观看黑人| 日韩视频免费看| 亚洲国产成人精品女人久久久| 一区二区三区成人| 久久精品国产99国产精品澳门| 国产午夜精品麻豆| 亚洲欧洲一区二区三区| 国产一区二区三区四区五区美女| 亚洲国产日韩一级| 一区二区在线观看视频在线观看 | 老牛嫩草一区二区三区日本| 欧美激情精品久久久| 免费成人激情视频| 国产日韩在线一区| 亚洲欧美日韩区 | 午夜影院日韩| 欧美视频一区在线观看| 久久婷婷久久一区二区三区| 欧美性猛交xxxx乱大交退制版| 亚洲日本国产| 亚洲一区二区在线播放| 欧美日本精品| 野花国产精品入口| 亚洲少妇中出一区| 国产欧美日韩精品a在线观看| 亚洲在线一区| 久久综合狠狠综合久久激情| 亚洲高清在线精品| 欧美激情偷拍| 亚洲欧美久久久| 亚洲黄色毛片| 欧美一区二区成人6969| 亚洲一区二区三区三| 午夜免费久久久久| 在线高清一区| 欧美日韩在线看| 久久久久国产一区二区| 亚洲精品国产精品国自产在线| 亚洲欧美激情在线视频| 精品999久久久| 国产精品国产三级国产普通话蜜臀| 亚洲在线第一页| 亚洲二区在线| 久久深夜福利免费观看| 在线一区二区日韩| 亚洲黄色高清| 在线成人性视频| 国产精品乱码一区二区三区| 欧美日韩高清在线| 欧美成人高清| 欧美激情按摩| 美女91精品| 欧美日韩在线观看一区二区|