传说,Pascal中有一个函数——fillchar。这是高手们装*用的利器,也是大家的神器(大家好才是真的好)。

神器介绍:

  百度百科说:“Fillchar是Turbo/Borland Pascal的System单元的一个标准过程,它的使用格式是:FillChar(var X; Count: Word; value),它的功能是,把指定变量X在内存段中所占的低Count个字节赋为相同的值value, 其中value是填充的值,只能是Byte、Char或Boolean等单字节类型的值。在Free Pascal中稍加扩展为FillChar(var X; Count: Longint; value), 功能没变。”听起来很没意义,搞不懂究竟是个什么鬼吧?

  其实这就是个内存块赋值函数,举个例子,这里有一段代码:

    fillchar(a,sizeof(a),233);//a:array[1..100] of byte

  等效于:

    for i:=1 to 100 do a[i]=233;

  显而易见,这个函数减少了编码工作量,而且不用在写代码时抬头看数组a的范围。看到这里,你肯定认为你已经懂了,打开了FPC,键入了如下代码:

    fillchar(a,sizeof(a),1);//a:array[1..100] of longint

  回头用Debug会发现,a中的值不为1,而是一个很大的值(16843009)10=(0000001000000100000010000001)2。没错,你被fillchar坑了。它的工作机理并有你想象的这么简单。

  回顾上文,看见红色的两个类型名了吗?再回顾上文,看见“单字节”三个字了吗?byte是个单字节类型,范围是0-255,但longint不是,它是一个4字节类型,范围是-2^31~2^31。而FPC把这个数组仍当做了一个byte等单字节类型的数组来处理。对于一个longint,其在内存中的块分布是这样的:(第1行第n列的数表示第n字节所表示的数,第2行第n列的数表示第n字节所表示数的二进制形式)

  0   0   0   0
00000000 00000000 00000000 00000000

  经过一次fillchar(a,sizeof(a),1)以后,fillchar对每个字节中的数都进行了赋值,于是内存就变成了这样。

  1   1   1   1
00000001 00000001 00000001 00000001

  把下面那行转化为10进制,就是16843009。

  再看一个例子,如果是fillchar(a,sizeof(a),128)呢?

  128   128   128   128
10000000 10000000 10000000 10000000

  呵呵,按刚才那样的转换,结果应该是2155905152。但是Debug又能够让你傻眼,结果是-2139062144。

  你忘记longint是个有符号数了吗?“1”处在符号位上,于是整个数就变成了负数。而负数又是补码来表示。当符号位为1时,补码的补码就是原数的绝对值。转一下,看看是不是这样?

神器应用:

  这里分点来描述:

  • 给数组赋值0,这是最常用的了,代码:fillchar(a,sizeof(a),0);//a可以是任何类型的数组,原因读者可以自己探究
  • 给数组赋值-1,比较常用,代码fillchar(a,sizeof(a),255);//a为longint数组
  • 给数组赋值±INF,赋值INF的值为127,-INF的值为128,要求数组类型为longint
  • 给boolean,char或byte,shortint数组赋初值,这里的初值就是最后存在数组里的值。
  • 其实fillchar不止可以用在数组上,也可以用在一个数上。那么,可以用这个类似BUG的玩意完成二进制重写2、4、8次后值的问题,而且只要一句话,比位运算更简单(虽然这好像没有什么卵用)。

神器的亲戚:

  神器是有亲戚的!

  pascal中,fillchar的亲戚有fillword,filldword,分别可以用在填充integer、longint数组,而且所见即所得——填充后的数与第三个参数一样(只是故意不早告诉你)。

  c++中,它的亲戚有memset,用法应该与fillchar一样(但是支持有符号数作为参数)。

OI中神奇的神器fillchar的更多相关文章

  1. 浅谈OI中的底层优化!

    众所周知,OI中其实就是算法竞赛,所以时间复杂度非常重要,一个是否优秀的算法或许就决定了人生,而在大多数情况下,我们想出的算法或许并不那么尽如人意,所以这时候就需要一中神奇的的东西,就是底层优化: 其 ...

  2. 浅谈分治算法在OI中的应用

    分治虽然是基本思想,但是OI中不会出裸分治让你一眼看出来,往往都是结合到找规律里面. 先来个简单的: 奇妙变换 (magic.pas/c/cpp) [问题描述]   为了奖励牛牛同学帮妈妈解决了大写中 ...

  3. [技术]浅谈OI中矩阵快速幂的用法

    前言 矩阵是高等代数学中的常见工具,也常见于统计分析等应用数学学科中,矩阵的运算是数值分析领域的重要问题. 基本介绍 (该部分为入门向,非入门选手可以跳过) 由 m行n列元素排列成的矩形阵列.矩阵里的 ...

  4. OI中常犯的傻逼错误总结

    OI中常犯的傻逼错误总结 问题 解决方案 文件名出错,包括文件夹,程序文件名,输入输出文件名  复制pdf的名字  没有去掉调试信息  调试时在后面加个显眼的标记  数组开小,超过定义大小,maxn/ ...

  5. GCC&&GDB在OI中的介绍

    序言 这本来是用Word写的,但是后来我换了系统所以只能用markdown迁移然后写了...... $\qquad$本文主要投食给那些在Windows下活了很久然后考试时发现需要用命令行来操作时困惑万 ...

  6. OI中的莫比乌斯反演

    OI中的莫比乌斯反演 莫比乌斯函数 想要学习莫比乌斯反演,首先要学习莫比乌斯函数. 定义 莫比乌斯函数用\(\mu(x)\)表示.如果\(x\)是\(k\)个不同质数的积,则\(\mu(x) = (- ...

  7. 浅谈OI中的提交答案

    在OI中,题目有三类: 传统题 交互题 提交答案题 今天来了解一下第三类 概述 传统题:给你一个题面,你需要交一个程序,评测姬会用你的程序运行你看不到的一些测试点,用输出和正确答案比较 提交答案题:给 ...

  8. OI中组合数的若干求法与CRT

    OI中组合数的若干求法与CRT 只是下决心整理一下子呢~ 说明:本篇文章采用\(\binom{a}{b}\)而不是\(C_{a}^b\),以\(p\)指代模数,\(fac_i\)指代\(i!\),\( ...

  9. OI中字符串读入和处理

    OI中字符串读入和处理 在NOIP的"大模拟"题中,往往要对字符串进行读入并处理,这些字符串有可能包含空格并以\n作为分割,传统的cin >> scanf() 等等,不 ...

随机推荐

  1. Stage3D学习笔记(六):旋转动画效果

    我们这节在上一篇代码的基础上再进一步,让显示的纹理进行旋转. 实现旋转,只需要每帧修改_modelViewMatrix的旋转角度即可,我们需要一个变量来记录旋转: //旋转度数 private var ...

  2. Swif基本语法以及与OC比较三

         (未 经 博 主 同 意,不 得 转 载 !)   ------------------------华丽分割线----------------------- // // main.swift ...

  3. Asp.Net Core- 多样性的配置来源

    我们知道,ConfigurationProvider提供将数据源转换为字典的功能,数据源可以分为很多种,比如:物理文件.数据库.内存变量等等.物理文件又包括很多种类型的文件,比如:xml.json等等 ...

  4. cocos2d-x CCSpriteBatchNode

    转自:http://www.cnblogs.com/jiackyan/archive/2013/04/14/3019880.html 1.先说下渲染批次:这是游戏引擎中一个比较重要的优化指标,指的是一 ...

  5. 【转载】Android Studio 设置内存大小及原理

    http://www.cnblogs.com/justinzhang/p/4274985.html http://tsroad.lofter.com/post/376316_69363ae Andro ...

  6. webViewDidFinishLoad因为网页里的重定向,会调用多次,使用web view.isLoading来解决

    我编码如下,但我发现 webViewDidFinishLoad() 会发生若干次. 如何知道 webViewDidFinishLoad() 最后发生吗? iNavigate = ; - (void)w ...

  7. mysqlnd cannot connect to MySQL 4.1+ using the old insecure authentication的解决方法

    直接命令行操作没有问题,但是PHP连接就会报上面的错误. SET old_passwords =0; USE mysql; UPDATE user SET password =PASSWORD('yo ...

  8. POJ 3080 Blue Jeans (KMP)

    求出公共子序列  要求最长  字典序最小 枚举第一串的所有子串   然后对每一个串做KMP.找到目标子串 学会了   strncpy函数的使用   我已可入灵魂 #include <iostre ...

  9. 《Java并发编程实战》第十五章 原子变量与非堵塞同步机制 读书笔记

    一.锁的劣势 锁定后假设未释放.再次请求锁时会造成堵塞.多线程调度通常遇到堵塞会进行上下文切换,造成很多其它的开销. 在挂起与恢复线程等过程中存在着非常大的开销,而且通常存在着较长时间的中断. 锁可能 ...

  10. 配置Ubuntu开发环境

    前言 新买了一台ThinkPad E431,主要看中了硬盘500G和7200转/分钟的速度,因此准备从x220上把工作环境迁移到新买的笔记本上. 为什么不要公司的电脑,是由于160G的ssd硬盘实在是 ...