开发的程序安装后,如何防止别人拷贝去直接使用?(50分)

  • 主题发起人 主题发起人 thj
  • 开始时间 开始时间
T

thj

Unregistered / Unconfirmed
GUEST, unregistred user!
我用D5+ADO+ACCESS编的程序安装后,直接将*.exe和*.mdb拷到一个文件夹中就可使用,如何防止?
当然,只是防一般的工作人员。
我是初学者,对这方面一窍不通,望详细指教!
 
这种方法许多,如写注册表后再检测就是最常见的。

你的程序在安装的时候在注册表某个写个信息,在启动的时候读一下,然后没有这个键值
或与你的不符就不能运行。

或有常见的用硬盘序列号加密方法,程序在启动时要检测硬盘序列号是否正确,如果不对
则拒绝运行。

这样拷贝则无效了。
 
一种方法:
安装后要求注册,注册通过后读硬盘序列号 …… 加密算法 …… 写注册表 或 ini 文件
如果不要注册,直接在安装时读硬盘序列号加密也行。
 
捆绕我们所有编程人员的问题啊,加密还不如不加,微软都很等厉害,解密也就是不到1天的时间啊
个人认为做点程序数据的保护,不是限制就可以了。多花点时间在编程上面,少花点时间在游戏上面,
别在加密上花时间。
 
通过写注册表的方法,别人只要监视注册表就知道干什么了。这样好了,安装时,在一个非常隐蔽的地方写一个文件,程序启动时检测该文件存在不存在,不存在就退出程序。这样就可以防住一般的人。
 
加密有一个好方法,大家有无办法破sachxp可以正常使用
没有吧!!哈哈,我之前试过破它,游戏是可以正常进入,但是很多功能不能使用
它是通过在线登录来使用自己的帐号的
 
谢谢各位的回答。
我说过,只防普通人员拷。
我对这方面真的不通,请给个详细的例子。ttapeng@163.com
 
1.获得序列号

(~~~~以下函数为转贴,版权归原作者所有!~~~~~)
function GetIdeDiskSerialNumber : String;
type
TSrbIoControl = packed record
HeaderLength : ULONG;
Signature : Array[0..7] of Char;
Timeout : ULONG;
ControlCode : ULONG;
ReturnCode : ULONG;
Length : ULONG;
end;
SRB_IO_CONTROL = TSrbIoControl;
PSrbIoControl = ^TSrbIoControl;

TIDERegs = packed record
bFeaturesReg : Byte; // Used for specifying SMART "commands".
bSectorCountReg : Byte; // IDE sector count register
bSectorNumberReg : Byte; // IDE sector number register
bCylLowReg : Byte; // IDE low order cylinder value
bCylHighReg : Byte; // IDE high order cylinder value
bDriveHeadReg : Byte; // IDE drive/head register
bCommandReg : Byte; // Actual IDE command.
bReserved : Byte; // reserved for future use. Must be zero.
end;
IDEREGS = TIDERegs;
PIDERegs = ^TIDERegs;

TSendCmdInParams = packed record
cBufferSize : DWORD; // Buffer size in bytes
irDriveRegs : TIDERegs; // Structure with drive register values.
bDriveNumber : Byte; // Physical drive number to send command to (0,1,2,3).
bReserved : Array[0..2] of Byte; // Reserved for future expansion.
dwReserved : Array[0..3] of DWORD; // For future use.
bBuffer : Array[0..0] of Byte; // Input buffer.
end;
SENDCMDINPARAMS = TSendCmdInParams;
PSendCmdInParams = ^TSendCmdInParams;

TIdSector = packed record
wGenConfig : Word;
wNumCyls : Word;
wReserved : Word;
wNumHeads : Word;
wBytesPerTrack : Word;
wBytesPerSector : Word;
wSectorsPerTrack : Word;
wVendorUnique : Array[0..2] of Word;
sSerialNumber : Array[0..19] of Char;
wBufferType : Word;
wBufferSize : Word;
wECCSize : Word;
sFirmwareRev : Array[0..7] of Char;
sModelNumber : Array[0..39] of Char;
wMoreVendorUnique : Word;
wDoubleWordIO : Word;
wCapabilities : Word;
wReserved1 : Word;
wPIOTiming : Word;
wDMATiming : Word;
wBS : Word;
wNumCurrentCyls : Word;
wNumCurrentHeads : Word;
wNumCurrentSectorsPerTrack : Word;
ulCurrentSectorCapacity : ULONG;
wMultSectorStuff : Word;
ulTotalAddressableSectors : ULONG;
wSingleWordDMA : Word;
wMultiWordDMA : Word;
bReserved : Array[0..127] of Byte;
end;
PIdSector = ^TIdSector;

const
IDE_ID_FUNCTION = $EC;
IDENTIFY_BUFFER_SIZE = 512;
DFP_RECEIVE_DRIVE_DATA = $0007c088;
IOCTL_SCSI_MINIPORT = $0004d008;
IOCTL_SCSI_MINIPORT_IDENTIFY = $001b0501;
DataSize = sizeof(TSendCmdInParams)-1+IDENTIFY_BUFFER_SIZE;
BufferSize = SizeOf(SRB_IO_CONTROL)+DataSize;
W9xBufferSize = IDENTIFY_BUFFER_SIZE+16;
var
hDevice : THandle;
cbBytesReturned : DWORD;
pInData : PSendCmdInParams;
pOutData : Pointer; // PSendCmdInParams;
Buffer : Array[0..BufferSize-1] of Byte;
srbControl : TSrbIoControl absolute Buffer;

procedure ChangeByteOrder( var Data; Size : Integer );
var ptr : PChar;
i : Integer;
c : Char;
begin
ptr := @Data;
for i := 0 to (Size shr 1)-1 do
begin
c := ptr^;
ptr^ := (ptr+1)^;
(ptr+1)^ := c;
Inc(ptr,2);
end;
end;

begin
Result := '';
FillChar(Buffer,BufferSize,#0);
if Win32Platform=VER_PLATFORM_WIN32_NT then
begin // Windows NT, Windows 2000
// Get SCSI port handle
hDevice := CreateFile( '//./Scsi0:', GENERIC_READ or GENERIC_WRITE,
FILE_SHARE_READ or FILE_SHARE_WRITE, nil, OPEN_EXISTING, 0, 0 );
if hDevice=INVALID_HANDLE_VALUE then Exit;
try
srbControl.HeaderLength := SizeOf(SRB_IO_CONTROL);
System.Move('SCSIDISK',srbControl.Signature,8);
srbControl.Timeout := 2;
srbControl.Length := DataSize;
srbControl.ControlCode := IOCTL_SCSI_MINIPORT_IDENTIFY;
pInData := PSendCmdInParams(PChar(@Buffer)+SizeOf(SRB_IO_CONTROL));
pOutData := pInData;
with pInData^ do
begin
cBufferSize := IDENTIFY_BUFFER_SIZE;
bDriveNumber := 0;
with irDriveRegs do
begin
bFeaturesReg := 0;
bSectorCountReg := 1;
bSectorNumberReg := 1;
bCylLowReg := 0;
bCylHighReg := 0;
bDriveHeadReg := $A0;
bCommandReg := IDE_ID_FUNCTION;
end;
end;
if not DeviceIoControl( hDevice, IOCTL_SCSI_MINIPORT, @Buffer, BufferSize, @Buffer, BufferSize, cbBytesReturned, nil ) then Exit;
finally
CloseHandle(hDevice);
end;
end
else
begin // Windows 95 OSR2, Windows 98
hDevice := CreateFile( '//./SMARTVSD', 0, 0, nil, CREATE_NEW, 0, 0 );
if hDevice=INVALID_HANDLE_VALUE then Exit;
try
pInData := PSendCmdInParams(@Buffer);
pOutData := PChar(@pInData^.bBuffer);
with pInData^ do
begin
cBufferSize := IDENTIFY_BUFFER_SIZE;
bDriveNumber := 0;
with irDriveRegs do
begin
bFeaturesReg := 0;
bSectorCountReg := 1;
bSectorNumberReg := 1;
bCylLowReg := 0;
bCylHighReg := 0;
bDriveHeadReg := $A0;
bCommandReg := IDE_ID_FUNCTION;
end;
end;
if not DeviceIoControl( hDevice, DFP_RECEIVE_DRIVE_DATA, pInData, SizeOf(TSendCmdInParams)-1, pOutData, W9xBufferSize, cbBytesReturned, nil ) then Exit;
finally
CloseHandle(hDevice);
end;
end;
with PIdSector(PChar(pOutData)+16)^ do
begin
ChangeByteOrder(sSerialNumber,SizeOf(sSerialNumber));
SetString(Result,sSerialNumber,SizeOf(sSerialNumber));
end;
end;

2,将序列号加密写ini文件或者注册表,这个简单,不用我写了,注意要放在隐密的地方,当然如果
是防普通人员那不要太紧张兮兮的。
3,程序运行时,读出刚才写的东西,解密后比较。如果与当前磁盘序列号不一致则不予运行。
第二步可以更深入一些,如果可以,你就把序列号写入自身文件(EXE文件),这个理论上是
可行的,网络上关于这个的东东很多,自己找找看。俺不是很懂。

 
获取本机的一些信息,如cup,硬盘,网卡,主板等序列号,
程序运行时就进行检查,由于以上信息都是唯一的,可以防止别人拷贝运行。
 
提供一个思路:
GetSystemDirectory()函数可以获取系统目录,windows/system或windows/system32
把一些本机相关的信息写入一个文件中,伪装成一个DLL,名字自己看着办,
这样一般人肯定是没法用的了。我还加上了登录用户数量的限制,可以用在MIDAS
系统中,呵呵,实际看效果不错。太多的精力放这没意思,需要的人给我留个幽香
 
干吗这么麻烦,人家只是防止普通人员拷贝使用。简单办法:
*.MDB与*.exe不要放在同个目录,可以把*.MDB文件放到C盘Windows下某个目录里。
当然编译时要设置好ADO的连接,并且制作安装文件时要将*.MDB直接装到那个目录。
 
多人接受答案了。
 
后退
顶部