⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 glplib06.c

📁 著名的大规模线性规划求解器源码GLPK.C语言版本,可以修剪.内有详细帮助文档.
💻 C
字号:
/* glplib06.c (64-bit arithmetic) *//************************************************************************  This code is part of GLPK (GNU Linear Programming Kit).**  Copyright (C) 2000,01,02,03,04,05,06,07,08,2009 Andrew Makhorin,*  Department for Applied Informatics, Moscow Aviation Institute,*  Moscow, Russia. All rights reserved. E-mail: <mao@mai2.rcnet.ru>.**  GLPK is free software: you can redistribute it and/or modify it*  under the terms of the GNU General Public License as published by*  the Free Software Foundation, either version 3 of the License, or*  (at your option) any later version.**  GLPK is distributed in the hope that it will be useful, but WITHOUT*  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY*  or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public*  License for more details.**  You should have received a copy of the GNU General Public License*  along with GLPK. If not, see <http://www.gnu.org/licenses/>.***********************************************************************/#include "glplib.h"/************************************************************************  NAME**  xlset - expand integer to long integer**  SYNOPSIS**  #include "glplib.h"*  xlong_t xlset(int x);**  RETURNS**  The routine xlset returns x expanded to long integer. */xlong_t xlset(int x){     xlong_t t;      t.lo = x, t.hi = (x >= 0 ? 0 : -1);      return t;}/************************************************************************  NAME**  xlneg - negate long integer**  SYNOPSIS**  #include "glplib.h"*  xlong_t xlneg(xlong_t x);**  RETURNS**  The routine xlneg returns the difference  0 - x. */xlong_t xlneg(xlong_t x){     if (x.lo)         x.lo = - x.lo, x.hi = ~x.hi;      else         x.hi = - x.hi;      return x;}/************************************************************************  NAME**  xladd - add long integers**  SYNOPSIS**  #include "glplib.h"*  xlong_t xladd(xlong_t x, xlong_t y);**  RETURNS**  The routine xladd returns the sum x + y. */xlong_t xladd(xlong_t x, xlong_t y){     if ((unsigned int)x.lo <= 0xFFFFFFFF - (unsigned int)y.lo)         x.lo += y.lo, x.hi += y.hi;      else         x.lo += y.lo, x.hi += y.hi + 1;      return x;}/************************************************************************  NAME**  xlsub - subtract long integers**  SYNOPSIS**  #include "glplib.h"*  xlong_t xlsub(xlong_t x, xlong_t y);**  RETURNS**  The routine xlsub returns the difference x - y. */xlong_t xlsub(xlong_t x, xlong_t y){     return         xladd(x, xlneg(y));}/************************************************************************  NAME**  xlcmp - compare long integers**  SYNOPSIS**  #include "glplib.h"*  int xlcmp(xlong_t x, xlong_t y);**  RETURNS**  The routine xlcmp returns the sign of the difference x - y. */int xlcmp(xlong_t x, xlong_t y){     if (x.hi >= 0 && y.hi <  0) return +1;      if (x.hi <  0 && y.hi >= 0) return -1;      if ((unsigned int)x.hi < (unsigned int)y.hi) return -1;      if ((unsigned int)x.hi > (unsigned int)y.hi) return +1;      if ((unsigned int)x.lo < (unsigned int)y.lo) return -1;      if ((unsigned int)x.lo > (unsigned int)y.lo) return +1;      return 0;}/************************************************************************  NAME**  xlmul - multiply long integers**  SYNOPSIS**  #include "glplib.h"*  xlong_t xlmul(xlong_t x, xlong_t y);**  RETURNS**  The routine xlmul returns the product x * y. */xlong_t xlmul(xlong_t x, xlong_t y){     unsigned short xx[8], yy[4];      xx[4] = (unsigned short)x.lo;      xx[5] = (unsigned short)(x.lo >> 16);      xx[6] = (unsigned short)x.hi;      xx[7] = (unsigned short)(x.hi >> 16);      yy[0] = (unsigned short)y.lo;      yy[1] = (unsigned short)(y.lo >> 16);      yy[2] = (unsigned short)y.hi;      yy[3] = (unsigned short)(y.hi >> 16);      bigmul(4, 4, xx, yy);      x.lo = (unsigned int)xx[0] | ((unsigned int)xx[1] << 16);      x.hi = (unsigned int)xx[2] | ((unsigned int)xx[3] << 16);      return x;}/************************************************************************  NAME**  xldiv - divide long integers**  SYNOPSIS**  #include "glplib.h"*  xldiv_t xldiv(xlong_t x, xlong_t y);**  RETURNS**  The routine xldiv returns a structure of type xldiv_t containing*  members quot (the quotient) and rem (the remainder), both of type*  xlong_t. */xldiv_t xldiv(xlong_t x, xlong_t y){     xldiv_t t;      int m, sx, sy;      unsigned short xx[8], yy[4];      /* sx := sign(x) */      sx = (x.hi < 0);      /* sy := sign(y) */      sy = (y.hi < 0);      /* x := |x| */      if (sx) x = xlneg(x);      /* y := |y| */      if (sy) y = xlneg(y);      /* compute x div y and x mod y */      xx[0] = (unsigned short)x.lo;      xx[1] = (unsigned short)(x.lo >> 16);      xx[2] = (unsigned short)x.hi;      xx[3] = (unsigned short)(x.hi >> 16);      yy[0] = (unsigned short)y.lo;      yy[1] = (unsigned short)(y.lo >> 16);      yy[2] = (unsigned short)y.hi;      yy[3] = (unsigned short)(y.hi >> 16);      if (yy[3])         m = 4;      else if (yy[2])         m = 3;      else if (yy[1])         m = 2;      else if (yy[0])         m = 1;      else         xerror("xldiv: divide by zero\n");      bigdiv(4 - m, m, xx, yy);      /* remainder in x[0], x[1], ..., x[m-1] */      t.rem.lo = (unsigned int)xx[0], t.rem.hi = 0;      if (m >= 2) t.rem.lo |= (unsigned int)xx[1] << 16;      if (m >= 3) t.rem.hi = (unsigned int)xx[2];      if (m >= 4) t.rem.hi |= (unsigned int)xx[3] << 16;      if (sx) t.rem = xlneg(t.rem);      /* quotient in x[m], x[m+1], ..., x[4] */      t.quot.lo = (unsigned int)xx[m], t.quot.hi = 0;      if (m <= 3) t.quot.lo |= (unsigned int)xx[m+1] << 16;      if (m <= 2) t.quot.hi = (unsigned int)xx[m+2];      if (m <= 1) t.quot.hi |= (unsigned int)xx[m+3] << 16;      if (sx ^ sy) t.quot = xlneg(t.quot);      return t;}/************************************************************************  NAME**  xltod - convert long integer to double**  SYNOPSIS**  #include "glplib.h"*  double xltod(xlong_t x);**  RETURNS**  The routine xltod returns x converted to double. */double xltod(xlong_t x){     double s, z;      if (x.hi >= 0)         s = +1.0;      else         s = -1.0, x = xlneg(x);      if (x.hi >= 0)         z = 4294967296.0 * (double)x.hi + (double)(unsigned int)x.lo;      else      {  xassert(x.hi == 0x80000000 && x.lo == 0x00000000);         z = 9223372036854775808.0; /* 2^63 */      }      return s * z;}char *xltoa(xlong_t x, char *s){     /* convert long integer to character string */      static const char *d = "0123456789";      xldiv_t t;      int neg, len;      if (x.hi >= 0)         neg = 0;      else         neg = 1, x = xlneg(x);      if (x.hi >= 0)      {  len = 0;         while (!(x.hi == 0 && x.lo == 0))         {  t = xldiv(x, xlset(10));            xassert(0 <= t.rem.lo && t.rem.lo <= 9);            s[len++] = d[t.rem.lo];            x = t.quot;         }         if (len == 0) s[len++] = d[0];         if (neg) s[len++] = '-';         s[len] = '\0';         strrev(s);      }      else         strcpy(s, "-9223372036854775808"); /* -2^63 */      return s;}/**********************************************************************/#if 0#include "glprng.h"#define N_TEST 1000000/* number of tests */static xlong_t myrand(RNG *rand){     xlong_t x;      int k;      k = rng_unif_rand(rand, 4);      xassert(0 <= k && k <= 3);      x.lo = rng_unif_rand(rand, 65536);      if (k == 1 || k == 3)      {  x.lo <<= 16;         x.lo += rng_unif_rand(rand, 65536);      }      if (k <= 1)         x.hi = 0;      else         x.hi = rng_unif_rand(rand, 65536);      if (k == 3)      {  x.hi <<= 16;         x.hi += rng_unif_rand(rand, 65536);      }      if (rng_unif_rand(rand, 2)) x = xlneg(x);      return x;}int main(void){     RNG *rand;      xlong_t x, y;      xldiv_t z;      int test;      rand = rng_create_rand();      for (test = 1; test <= N_TEST; test++)      {  x = myrand(rand);         y = myrand(rand);         if (y.lo == 0 && y.hi == 0) y.lo = 1;         /* z.quot := x div y, z.rem := x mod y */         z = xldiv(x, y);         /* x must be equal to y * z.quot + z.rem */         xassert(xlcmp(x, xladd(xlmul(y, z.quot), z.rem)) == 0);      }      xprintf("%d tests successfully passed\n", N_TEST);      rng_delete_rand(rand);      return 0;}#endif/* eof */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -