免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
123
最近访问板块 发新帖
楼主: 太平绅士
打印 上一主题 下一主题

求解一个空间位置的算法 [复制链接]

论坛徽章:
0
21 [报告]
发表于 2009-03-29 13:53 |只看该作者
考虑过把二维映射为一维 ,但如果不离散化,会占用太多空间,
如果离散化,就难以支持修改操作 ,最难的地方是如何保存window id,
rmq .树状数组貌似也不太合适...

抛砖引玉,大牛指点...

论坛徽章:
0
22 [报告]
发表于 2009-03-29 14:39 |只看该作者
原帖由 windyrobin 于 2009-3-29 13:42 发表
复习了下算法 ,搞到凌晨2点,写了个简陋的线段树
测试发现, 在我提供的例子中,速度是普通版的1. 8倍 左右,
我用vector存储 window id , 添加、删除修改操做比较低效,
换成set 应该能大幅提高修改操作效 ...


学习了..
这个实现看起来比我的有效率一点, 构造线段树比较快.

我用线段的端点, 作为线段树的节点, 利用std::map作为这些节点的容器,
这样构造了一个变种的动态线段树. 但是初始化的动作太慢了.
查找的效率有一点提高, 不足以弥补构造花费的时间.


贴代码, 仅实现了1维。
最开始使用set保存线段,实现了插入/删除,不过构造线段树的时间更加惨不忍睹。
后来改成vector保存线段,能构造的快一点,仅有插入。



  1. #include <windows.h>

  2. #include <vector>
  3. #include <map>
  4. #include <set>
  5. #include <list>



  6. struct u_line
  7. {
  8.         inline u_line( int x, int y )
  9.                 : m_x( x )
  10.                 , m_y( y )
  11.         {
  12.         }
  13.         int m_x;
  14.         int m_y;

  15.         bool operator<( u_line const &right ) const
  16.         {
  17.                 if( this->m_y < right.m_y )
  18.                         return true;
  19.                 else if( this->m_y > right.m_y )
  20.                         return false;
  21.                 else if( this->m_x < right.m_x )
  22.                         return true;
  23.                 else
  24.                         return false;
  25.         }
  26. };

  27. struct u_point
  28. {
  29.         int start_pos;
  30.         //std::set< u_line > lines;
  31.         std::vector< u_line > lines;
  32. };




  33. void dump_point( int index, std::map< int, u_point > const &points )
  34. {
  35.         printf( "%d=====\n", index );
  36.         for( std::map< int, u_point >::const_iterator itr = points.begin( );
  37.                 itr != points.end( );
  38.                 ++itr )
  39.         {
  40.                 printf( "Point: %d\n", itr->second.start_pos );
  41.                 for( std::vector< u_line >::const_iterator itr_line = itr->second.lines.begin( );
  42.                         itr_line != itr->second.lines.end( );
  43.                         ++itr_line )
  44.                 {
  45.                         printf( "[%d,%d] ", (itr_line)->m_x, (itr_line)->m_y );
  46.                 }
  47.                 printf( "\n" );
  48.         }
  49. }

  50. #if 0
  51. void del_line( std::map< int, u_point >        &points, u_line const &line )
  52. {
  53.         std::map< int, u_point >::iterator itr = points.upper_bound( line.m_x );
  54.         if( itr == points.end( ) || itr == points.begin( ) )
  55.         {
  56.                 printf( "del_line: %d %d, Point: NOT FOUND0\n", line.m_x, line.m_y );
  57.                 return;
  58.         }
  59.        
  60.         --itr;
  61.         if( itr->second.start_pos != line.m_x )
  62.         {
  63.                 printf( "del_line: %d %d, Point: NOT FOUND1\n", line.m_x, line.m_y );
  64.                 return;
  65.         }

  66.         while( itr != points.end( ) && itr->second.start_pos <= line.m_y )
  67.         {
  68.                 std::map< int, u_point >::iterator itr_this = itr;
  69.                 ++itr;

  70.                 if( itr_this->second.start_pos < line.m_y )
  71.                         itr_this->second.lines.erase( line );

  72.                 if( itr_this->second.lines.size( ) == 0
  73.                         && itr_this != points.begin( ) )
  74.                 {
  75.                         std::map< int, u_point >::iterator prev = itr_this;
  76.                         --prev;
  77.                         if( prev->second.lines.size( ) == 0 )
  78.                                 points.erase( itr_this );
  79.                 }
  80.         }

  81.         dump_point( __LINE__, points );
  82. }
  83. #endif

  84. void add_line( std::map< int, u_point >        &points, u_line const &line )
  85. {
  86.         /* <0> Insert end point */
  87.         //printf( "Insert %d\n", line.m_y );
  88.         u_point point1;
  89.         point1.start_pos = line.m_y;

  90.         std::pair< std::map< int, u_point >::iterator, bool > insert1 =
  91.                 points.insert( std::make_pair( line.m_y, point1 ) );

  92.         std::map< int, u_point >::iterator itr1 = insert1.first;

  93.         //dump_point( __LINE__, points );
  94.         if( insert1.second )
  95.         {/* A new point was inserted, any lines on previous point need to insert to the line array of new point */
  96.                 std::map< int, u_point >::iterator prev = itr1;
  97.                 if( prev != points.begin( ) )
  98.                 {
  99.                         --prev;
  100.                         //itr1->second.lines.insert( prev->second.lines.begin( ), prev->second.lines.end( ) );
  101.                         itr1->second.lines = prev->second.lines;
  102.                 }
  103.         }
  104.         //dump_point( __LINE__, points );

  105.         /* <1> Insert start point. */
  106.         //printf( "Insert %d\n", line.m_x );
  107.         u_point point0;
  108.         point0.start_pos = line.m_x;

  109.         std::pair< std::map< int, u_point >::iterator, bool > insert0 =
  110.                 points.insert( std::make_pair( line.m_x, point0 ) );

  111.         std::map< int, u_point >::iterator itr0 = insert0.first;

  112.         //dump_point( __LINE__, points );

  113.         if( insert0.second )
  114.         {/* A new point was inserted, any lines on previous point need to insert to the line array of new point */
  115.                 std::map< int, u_point >::iterator prev = itr0;
  116.                 if( prev != points.begin( ) )
  117.                 {
  118.                         --prev;
  119.                         //itr0->second.lines.insert( prev->second.lines.begin( ), prev->second.lines.end( ) );
  120.                         itr0->second.lines = prev->second.lines;
  121.                 }
  122.         }
  123.         //dump_point( __LINE__, points );


  124.         /* <2> Append line to the new added point */
  125.         for( std::map< int, u_point >::iterator itr = itr0;
  126.                 itr != itr1;
  127.                 ++itr )
  128.                 //itr->second.lines.insert( line );
  129.                 itr->second.lines.push_back( line );

  130.         //dump_point( __LINE__, points );
  131. }


  132. bool search_point( std::map< int, u_point >        &points, int pos, int &found_num )
  133. {
  134.         std::map< int, u_point >::iterator itr = points.upper_bound( pos );
  135.         if( itr == points.end( ) || itr == points.begin( ) )
  136.         {
  137.                 //printf( "Search: %d, Point: NOT FOUND\n", pos );
  138.                 return false;
  139.         }

  140.         --itr;

  141.         //printf( "Search: %d, Point: %d, size = %d\n", pos, itr->second.start_pos, itr->second.lines.size( ) );
  142.         for( std::vector< u_line >::const_iterator itr_line = itr->second.lines.begin( );
  143.                 itr_line != itr->second.lines.end( );
  144.                 ++itr_line )
  145.         {
  146.                 ++found_num;
  147.                 //printf( "[%d,%d] ", (itr_line)->m_x, (itr_line)->m_y );
  148.         }
  149.         //printf( "\n" );

  150.         return true;
  151. }

  152. int main( int argc, char **argv )
  153. {
  154. #define TEST_SIZE 10000
  155. #define MAP_SIZE 65536
  156. #define LOOP_NUM 10000

  157.         /* <0> 直接在vector中查找 */      
  158.         std::vector< u_line > lines;
  159.         {
  160.                 DWORD ticks = ::GetTickCount();
  161.                 for( int i = 0; i < TEST_SIZE; ++i )
  162.                 {
  163.                         u_line line( rand( ) % MAP_SIZE, rand( ) % MAP_SIZE );
  164.                         if( line.m_x > line.m_y )
  165.                                 std::swap( line.m_x, line.m_y );
  166.                         if( line.m_x < line.m_y )
  167.                                 lines.push_back( line );
  168.                 }
  169.                 printf( "create time %d\n", (int)( ::GetTickCount( ) - ticks ) );
  170.         }

  171.         {
  172.                 int found_num = 0;
  173.                 DWORD ticks = ::GetTickCount();
  174.                 for( int i = 0; i < LOOP_NUM; ++i )
  175.                 {
  176.                         int pos = rand( ) % MAP_SIZE;
  177.                         for( std::vector< u_line >::iterator itr = lines.begin( );
  178.                                 itr != lines.end( );
  179.                                 ++itr )
  180.                         {
  181.                                 if( itr->m_x <= pos && itr->m_y > pos )
  182.                                         ++found_num;
  183.                         }
  184.                 }
  185.                 printf( "found_num = %d, search time = %d\n", found_num, (int)( ::GetTickCount( ) - ticks ) );
  186.         }

  187.         /* <1> 利用map构造的线段树中查找 */      
  188.         std::map< int, u_point > points;

  189.         {
  190.                 DWORD ticks = ::GetTickCount();
  191.                 for( int i = 0; i < TEST_SIZE; ++i )
  192.                 {
  193.                         u_line line( rand( ) % MAP_SIZE, rand( ) % MAP_SIZE );
  194.                         if( line.m_x > line.m_y )
  195.                                 std::swap( line.m_x, line.m_y );
  196.                         if( line.m_x < line.m_y )
  197.                                 add_line( points, line );
  198.                 }
  199.                 printf( "create time %d\n", (int)( ::GetTickCount( ) - ticks ) );
  200.         }

  201.         {
  202.                 int found_num = 0;
  203.                 DWORD ticks = ::GetTickCount();
  204.                 for( int i = 0; i < LOOP_NUM; ++i )
  205.                 {
  206.                         int pos = rand( ) % MAP_SIZE;
  207.                         search_point( points, pos, found_num );
  208.                 }
  209.                 printf( "found_num = %d, search time = %d\n", found_num, (int)( ::GetTickCount( ) - ticks ) );
  210.         }

  211.         return 0;
  212. }
复制代码

[ 本帖最后由 太平绅士 于 2009-3-29 14:54 编辑 ]

论坛徽章:
0
23 [报告]
发表于 2009-03-29 18:04 |只看该作者
google KD-Tree
or
baidu 计算几何 点定位

论坛徽章:
0
24 [报告]
发表于 2009-03-29 22:43 |只看该作者
是不是在搞3D窗口管理器啊??
想到一块去了!
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP